forked from auracaster/bumble_mirror
Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| bdad225033 | |||
| 8eeb58e467 | |||
| 91971433d2 | |||
| afb21220e2 |
@@ -41,3 +41,30 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
inv build
|
inv build
|
||||||
inv build.mkdocs
|
inv build.mkdocs
|
||||||
|
build-rust:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
python-version: [ "3.8", "3.9", "3.10" ]
|
||||||
|
fail-fast: false
|
||||||
|
steps:
|
||||||
|
- name: Check out from Git
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: Set up Python ${{ matrix.python-version }}
|
||||||
|
uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: ${{ matrix.python-version }}
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
python -m pip install --upgrade pip
|
||||||
|
python -m pip install ".[build,test,development,documentation]"
|
||||||
|
- name: Install Rust toolchain
|
||||||
|
uses: actions-rust-lang/setup-rust-toolchain@v1
|
||||||
|
with:
|
||||||
|
components: clippy,rustfmt
|
||||||
|
- name: Rust Lints
|
||||||
|
run: cd rust && cargo fmt --check && cargo clippy --all-targets -- --deny warnings
|
||||||
|
- name: Rust Build
|
||||||
|
run: cd rust && cargo build --all-targets
|
||||||
|
- name: Rust Tests
|
||||||
|
run: cd rust && cargo test
|
||||||
@@ -9,3 +9,4 @@ __pycache__
|
|||||||
# generated by setuptools_scm
|
# generated by setuptools_scm
|
||||||
bumble/_version.py
|
bumble/_version.py
|
||||||
.vscode/launch.json
|
.vscode/launch.json
|
||||||
|
/.idea
|
||||||
|
|||||||
+230
-221
@@ -1445,8 +1445,14 @@ class HCI_Object:
|
|||||||
@staticmethod
|
@staticmethod
|
||||||
def init_from_fields(hci_object, fields, values):
|
def init_from_fields(hci_object, fields, values):
|
||||||
if isinstance(values, dict):
|
if isinstance(values, dict):
|
||||||
for field_name, _ in fields:
|
for field in fields:
|
||||||
setattr(hci_object, field_name, values[field_name])
|
if isinstance(field, list):
|
||||||
|
# The field is an array, up-level the array field names
|
||||||
|
for sub_field_name, _ in field:
|
||||||
|
setattr(hci_object, sub_field_name, values[sub_field_name])
|
||||||
|
else:
|
||||||
|
field_name = field[0]
|
||||||
|
setattr(hci_object, field_name, values[field_name])
|
||||||
else:
|
else:
|
||||||
for field_name, field_value in zip(fields, values):
|
for field_name, field_value in zip(fields, values):
|
||||||
setattr(hci_object, field_name, field_value)
|
setattr(hci_object, field_name, field_value)
|
||||||
@@ -1456,133 +1462,161 @@ class HCI_Object:
|
|||||||
parsed = HCI_Object.dict_from_bytes(data, offset, fields)
|
parsed = HCI_Object.dict_from_bytes(data, offset, fields)
|
||||||
HCI_Object.init_from_fields(hci_object, parsed.keys(), parsed.values())
|
HCI_Object.init_from_fields(hci_object, parsed.keys(), parsed.values())
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def parse_field(data, offset, field_type):
|
||||||
|
# The field_type may be a dictionary with a mapper, parser, and/or size
|
||||||
|
if isinstance(field_type, dict):
|
||||||
|
if 'size' in field_type:
|
||||||
|
field_type = field_type['size']
|
||||||
|
elif 'parser' in field_type:
|
||||||
|
field_type = field_type['parser']
|
||||||
|
|
||||||
|
# Parse the field
|
||||||
|
if field_type == '*':
|
||||||
|
# The rest of the bytes
|
||||||
|
field_value = data[offset:]
|
||||||
|
return (field_value, len(field_value))
|
||||||
|
if field_type == 1:
|
||||||
|
# 8-bit unsigned
|
||||||
|
return (data[offset], 1)
|
||||||
|
if field_type == -1:
|
||||||
|
# 8-bit signed
|
||||||
|
return (struct.unpack_from('b', data, offset)[0], 1)
|
||||||
|
if field_type == 2:
|
||||||
|
# 16-bit unsigned
|
||||||
|
return (struct.unpack_from('<H', data, offset)[0], 2)
|
||||||
|
if field_type == '>2':
|
||||||
|
# 16-bit unsigned big-endian
|
||||||
|
return (struct.unpack_from('>H', data, offset)[0], 2)
|
||||||
|
if field_type == -2:
|
||||||
|
# 16-bit signed
|
||||||
|
return (struct.unpack_from('<h', data, offset)[0], 2)
|
||||||
|
if field_type == 3:
|
||||||
|
# 24-bit unsigned
|
||||||
|
padded = data[offset : offset + 3] + bytes([0])
|
||||||
|
return (struct.unpack('<I', padded)[0], 3)
|
||||||
|
if field_type == 4:
|
||||||
|
# 32-bit unsigned
|
||||||
|
return (struct.unpack_from('<I', data, offset)[0], 4)
|
||||||
|
if field_type == '>4':
|
||||||
|
# 32-bit unsigned big-endian
|
||||||
|
return (struct.unpack_from('>I', data, offset)[0], 4)
|
||||||
|
if isinstance(field_type, int) and 4 < field_type <= 256:
|
||||||
|
# Byte array (from 5 up to 256 bytes)
|
||||||
|
return (data[offset : offset + field_type], field_type)
|
||||||
|
if callable(field_type):
|
||||||
|
new_offset, field_value = field_type(data, offset)
|
||||||
|
return (field_value, new_offset - offset)
|
||||||
|
|
||||||
|
raise ValueError(f'unknown field type {field_type}')
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def dict_from_bytes(data, offset, fields):
|
def dict_from_bytes(data, offset, fields):
|
||||||
result = collections.OrderedDict()
|
result = collections.OrderedDict()
|
||||||
for (field_name, field_type) in fields:
|
for field in fields:
|
||||||
# The field_type may be a dictionary with a mapper, parser, and/or size
|
if isinstance(field, list):
|
||||||
if isinstance(field_type, dict):
|
# This is an array field, starting with a 1-byte item count.
|
||||||
if 'size' in field_type:
|
item_count = data[offset]
|
||||||
field_type = field_type['size']
|
|
||||||
elif 'parser' in field_type:
|
|
||||||
field_type = field_type['parser']
|
|
||||||
|
|
||||||
# Parse the field
|
|
||||||
if field_type == '*':
|
|
||||||
# The rest of the bytes
|
|
||||||
field_value = data[offset:]
|
|
||||||
offset += len(field_value)
|
|
||||||
elif field_type == 1:
|
|
||||||
# 8-bit unsigned
|
|
||||||
field_value = data[offset]
|
|
||||||
offset += 1
|
offset += 1
|
||||||
elif field_type == -1:
|
for _ in range(item_count):
|
||||||
# 8-bit signed
|
for sub_field_name, sub_field_type in field:
|
||||||
field_value = struct.unpack_from('b', data, offset)[0]
|
value, size = HCI_Object.parse_field(
|
||||||
offset += 1
|
data, offset, sub_field_type
|
||||||
elif field_type == 2:
|
)
|
||||||
# 16-bit unsigned
|
result.setdefault(sub_field_name, []).append(value)
|
||||||
field_value = struct.unpack_from('<H', data, offset)[0]
|
offset += size
|
||||||
offset += 2
|
continue
|
||||||
elif field_type == '>2':
|
|
||||||
# 16-bit unsigned big-endian
|
|
||||||
field_value = struct.unpack_from('>H', data, offset)[0]
|
|
||||||
offset += 2
|
|
||||||
elif field_type == -2:
|
|
||||||
# 16-bit signed
|
|
||||||
field_value = struct.unpack_from('<h', data, offset)[0]
|
|
||||||
offset += 2
|
|
||||||
elif field_type == 3:
|
|
||||||
# 24-bit unsigned
|
|
||||||
padded = data[offset : offset + 3] + bytes([0])
|
|
||||||
field_value = struct.unpack('<I', padded)[0]
|
|
||||||
offset += 3
|
|
||||||
elif field_type == 4:
|
|
||||||
# 32-bit unsigned
|
|
||||||
field_value = struct.unpack_from('<I', data, offset)[0]
|
|
||||||
offset += 4
|
|
||||||
elif field_type == '>4':
|
|
||||||
# 32-bit unsigned big-endian
|
|
||||||
field_value = struct.unpack_from('>I', data, offset)[0]
|
|
||||||
offset += 4
|
|
||||||
elif isinstance(field_type, int) and 4 < field_type <= 256:
|
|
||||||
# Byte array (from 5 up to 256 bytes)
|
|
||||||
field_value = data[offset : offset + field_type]
|
|
||||||
offset += field_type
|
|
||||||
elif callable(field_type):
|
|
||||||
offset, field_value = field_type(data, offset)
|
|
||||||
else:
|
|
||||||
raise ValueError(f'unknown field type {field_type}')
|
|
||||||
|
|
||||||
|
field_name, field_type = field
|
||||||
|
field_value, field_size = HCI_Object.parse_field(data, offset, field_type)
|
||||||
result[field_name] = field_value
|
result[field_name] = field_value
|
||||||
|
offset += field_size
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def serialize_field(field_value, field_type):
|
||||||
|
# The field_type may be a dictionary with a mapper, parser, serializer,
|
||||||
|
# and/or size
|
||||||
|
serializer = None
|
||||||
|
if isinstance(field_type, dict):
|
||||||
|
if 'serializer' in field_type:
|
||||||
|
serializer = field_type['serializer']
|
||||||
|
if 'size' in field_type:
|
||||||
|
field_type = field_type['size']
|
||||||
|
|
||||||
|
# Serialize the field
|
||||||
|
if serializer:
|
||||||
|
field_bytes = serializer(field_value)
|
||||||
|
elif field_type == 1:
|
||||||
|
# 8-bit unsigned
|
||||||
|
field_bytes = bytes([field_value])
|
||||||
|
elif field_type == -1:
|
||||||
|
# 8-bit signed
|
||||||
|
field_bytes = struct.pack('b', field_value)
|
||||||
|
elif field_type == 2:
|
||||||
|
# 16-bit unsigned
|
||||||
|
field_bytes = struct.pack('<H', field_value)
|
||||||
|
elif field_type == '>2':
|
||||||
|
# 16-bit unsigned big-endian
|
||||||
|
field_bytes = struct.pack('>H', field_value)
|
||||||
|
elif field_type == -2:
|
||||||
|
# 16-bit signed
|
||||||
|
field_bytes = struct.pack('<h', field_value)
|
||||||
|
elif field_type == 3:
|
||||||
|
# 24-bit unsigned
|
||||||
|
field_bytes = struct.pack('<I', field_value)[0:3]
|
||||||
|
elif field_type == 4:
|
||||||
|
# 32-bit unsigned
|
||||||
|
field_bytes = struct.pack('<I', field_value)
|
||||||
|
elif field_type == '>4':
|
||||||
|
# 32-bit unsigned big-endian
|
||||||
|
field_bytes = struct.pack('>I', field_value)
|
||||||
|
elif field_type == '*':
|
||||||
|
if isinstance(field_value, int):
|
||||||
|
if 0 <= field_value <= 255:
|
||||||
|
field_bytes = bytes([field_value])
|
||||||
|
else:
|
||||||
|
raise ValueError('value too large for *-typed field')
|
||||||
|
else:
|
||||||
|
field_bytes = bytes(field_value)
|
||||||
|
elif isinstance(field_value, (bytes, bytearray)) or hasattr(
|
||||||
|
field_value, 'to_bytes'
|
||||||
|
):
|
||||||
|
field_bytes = bytes(field_value)
|
||||||
|
if isinstance(field_type, int) and 4 < field_type <= 256:
|
||||||
|
# Truncate or pad with zeros if the field is too long or too short
|
||||||
|
if len(field_bytes) < field_type:
|
||||||
|
field_bytes += bytes(field_type - len(field_bytes))
|
||||||
|
elif len(field_bytes) > field_type:
|
||||||
|
field_bytes = field_bytes[:field_type]
|
||||||
|
else:
|
||||||
|
raise ValueError(f"don't know how to serialize type {type(field_value)}")
|
||||||
|
|
||||||
|
return field_bytes
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def dict_to_bytes(hci_object, fields):
|
def dict_to_bytes(hci_object, fields):
|
||||||
result = bytearray()
|
result = bytearray()
|
||||||
for (field_name, field_type) in fields:
|
for field in fields:
|
||||||
# The field_type may be a dictionary with a mapper, parser, serializer,
|
if isinstance(field, list):
|
||||||
# and/or size
|
# The field is an array. The serialized form starts with a 1-byte
|
||||||
serializer = None
|
# item count. We use the length of the first array field as the
|
||||||
if isinstance(field_type, dict):
|
# array count, since all array fields have the same number of items.
|
||||||
if 'serializer' in field_type:
|
item_count = len(hci_object[field[0][0]])
|
||||||
serializer = field_type['serializer']
|
result += bytes([item_count]) + b''.join(
|
||||||
if 'size' in field_type:
|
b''.join(
|
||||||
field_type = field_type['size']
|
HCI_Object.serialize_field(
|
||||||
|
hci_object[sub_field_name][i], sub_field_type
|
||||||
# Serialize the field
|
)
|
||||||
field_value = hci_object[field_name]
|
for sub_field_name, sub_field_type in field
|
||||||
if serializer:
|
)
|
||||||
field_bytes = serializer(field_value)
|
for i in range(item_count)
|
||||||
elif field_type == 1:
|
|
||||||
# 8-bit unsigned
|
|
||||||
field_bytes = bytes([field_value])
|
|
||||||
elif field_type == -1:
|
|
||||||
# 8-bit signed
|
|
||||||
field_bytes = struct.pack('b', field_value)
|
|
||||||
elif field_type == 2:
|
|
||||||
# 16-bit unsigned
|
|
||||||
field_bytes = struct.pack('<H', field_value)
|
|
||||||
elif field_type == '>2':
|
|
||||||
# 16-bit unsigned big-endian
|
|
||||||
field_bytes = struct.pack('>H', field_value)
|
|
||||||
elif field_type == -2:
|
|
||||||
# 16-bit signed
|
|
||||||
field_bytes = struct.pack('<h', field_value)
|
|
||||||
elif field_type == 3:
|
|
||||||
# 24-bit unsigned
|
|
||||||
field_bytes = struct.pack('<I', field_value)[0:3]
|
|
||||||
elif field_type == 4:
|
|
||||||
# 32-bit unsigned
|
|
||||||
field_bytes = struct.pack('<I', field_value)
|
|
||||||
elif field_type == '>4':
|
|
||||||
# 32-bit unsigned big-endian
|
|
||||||
field_bytes = struct.pack('>I', field_value)
|
|
||||||
elif field_type == '*':
|
|
||||||
if isinstance(field_value, int):
|
|
||||||
if 0 <= field_value <= 255:
|
|
||||||
field_bytes = bytes([field_value])
|
|
||||||
else:
|
|
||||||
raise ValueError('value too large for *-typed field')
|
|
||||||
else:
|
|
||||||
field_bytes = bytes(field_value)
|
|
||||||
elif isinstance(field_value, (bytes, bytearray)) or hasattr(
|
|
||||||
field_value, 'to_bytes'
|
|
||||||
):
|
|
||||||
field_bytes = bytes(field_value)
|
|
||||||
if isinstance(field_type, int) and 4 < field_type <= 256:
|
|
||||||
# Truncate or Pad with zeros if the field is too long or too short
|
|
||||||
if len(field_bytes) < field_type:
|
|
||||||
field_bytes += bytes(field_type - len(field_bytes))
|
|
||||||
elif len(field_bytes) > field_type:
|
|
||||||
field_bytes = field_bytes[:field_type]
|
|
||||||
else:
|
|
||||||
raise ValueError(
|
|
||||||
f"don't know how to serialize type {type(field_value)}"
|
|
||||||
)
|
)
|
||||||
|
continue
|
||||||
|
|
||||||
result += field_bytes
|
(field_name, field_type) = field
|
||||||
|
result += HCI_Object.serialize_field(hci_object[field_name], field_type)
|
||||||
|
|
||||||
return bytes(result)
|
return bytes(result)
|
||||||
|
|
||||||
@@ -1617,48 +1651,73 @@ class HCI_Object:
|
|||||||
return str(value)
|
return str(value)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def format_fields(hci_object, keys, indentation='', value_mappers=None):
|
def stringify_field(
|
||||||
if not keys:
|
field_name, field_type, field_value, indentation, value_mappers
|
||||||
return ''
|
):
|
||||||
|
value_mapper = None
|
||||||
|
if isinstance(field_type, dict):
|
||||||
|
# Get the value mapper from the specifier
|
||||||
|
value_mapper = field_type.get('mapper')
|
||||||
|
|
||||||
# Measure the widest field name
|
# Check if there's a matching mapper passed
|
||||||
max_field_name_length = max(
|
if value_mappers:
|
||||||
(len(key[0] if isinstance(key, tuple) else key) for key in keys)
|
value_mapper = value_mappers.get(field_name, value_mapper)
|
||||||
|
|
||||||
|
# Map the value if we have a mapper
|
||||||
|
if value_mapper is not None:
|
||||||
|
field_value = value_mapper(field_value)
|
||||||
|
|
||||||
|
# Get the string representation of the value
|
||||||
|
return HCI_Object.format_field_value(
|
||||||
|
field_value, indentation=indentation + ' '
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def format_fields(hci_object, fields, indentation='', value_mappers=None):
|
||||||
|
if not fields:
|
||||||
|
return ''
|
||||||
|
|
||||||
# Build array of formatted key:value pairs
|
# Build array of formatted key:value pairs
|
||||||
fields = []
|
field_strings = []
|
||||||
for key in keys:
|
for field in fields:
|
||||||
value_mapper = None
|
if isinstance(field, list):
|
||||||
if isinstance(key, tuple):
|
for sub_field in field:
|
||||||
# The key has an associated specifier
|
sub_field_name, sub_field_type = sub_field
|
||||||
key, specifier = key
|
item_count = len(hci_object[sub_field_name])
|
||||||
|
for i in range(item_count):
|
||||||
|
field_strings.append(
|
||||||
|
(
|
||||||
|
f'{sub_field_name}[{i}]',
|
||||||
|
HCI_Object.stringify_field(
|
||||||
|
sub_field_name,
|
||||||
|
sub_field_type,
|
||||||
|
hci_object[sub_field_name][i],
|
||||||
|
indentation,
|
||||||
|
value_mappers,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
continue
|
||||||
|
|
||||||
# Get the value mapper from the specifier
|
field_name, field_type = field
|
||||||
if isinstance(specifier, dict):
|
field_value = hci_object[field_name]
|
||||||
value_mapper = specifier.get('mapper')
|
field_strings.append(
|
||||||
|
(
|
||||||
# Get the value for the field
|
field_name,
|
||||||
value = hci_object[key]
|
HCI_Object.stringify_field(
|
||||||
|
field_name, field_type, field_value, indentation, value_mappers
|
||||||
# Check if there's a matching mapper passed
|
),
|
||||||
if value_mappers:
|
),
|
||||||
value_mapper = value_mappers.get(key, value_mapper)
|
|
||||||
|
|
||||||
# Map the value if we have a mapper
|
|
||||||
if value_mapper is not None:
|
|
||||||
value = value_mapper(value)
|
|
||||||
|
|
||||||
# Get the string representation of the value
|
|
||||||
value_str = HCI_Object.format_field_value(
|
|
||||||
value, indentation=indentation + ' '
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Add the field to the formatted result
|
# Measure the widest field name
|
||||||
key_str = color(f'{key + ":":{1 + max_field_name_length}}', 'cyan')
|
max_field_name_length = max(len(s[0]) for s in field_strings)
|
||||||
fields.append(f'{indentation}{key_str} {value_str}')
|
sep = ':'
|
||||||
|
return '\n'.join(
|
||||||
return '\n'.join(fields)
|
f'{indentation}'
|
||||||
|
f'{color(f"{field_name + sep:{1 + max_field_name_length}}", "cyan")} {field_value}'
|
||||||
|
for field_name, field_value in field_strings
|
||||||
|
)
|
||||||
|
|
||||||
def __bytes__(self):
|
def __bytes__(self):
|
||||||
return self.to_bytes()
|
return self.to_bytes()
|
||||||
@@ -3769,9 +3828,7 @@ class HCI_LE_Set_Extended_Advertising_Parameters_Command(HCI_Command):
|
|||||||
'advertising_data',
|
'advertising_data',
|
||||||
{
|
{
|
||||||
'parser': HCI_Object.parse_length_prefixed_bytes,
|
'parser': HCI_Object.parse_length_prefixed_bytes,
|
||||||
'serializer': functools.partial(
|
'serializer': HCI_Object.serialize_length_prefixed_bytes,
|
||||||
HCI_Object.serialize_length_prefixed_bytes
|
|
||||||
),
|
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
@@ -3819,9 +3876,7 @@ class HCI_LE_Set_Extended_Advertising_Data_Command(HCI_Command):
|
|||||||
'scan_response_data',
|
'scan_response_data',
|
||||||
{
|
{
|
||||||
'parser': HCI_Object.parse_length_prefixed_bytes,
|
'parser': HCI_Object.parse_length_prefixed_bytes,
|
||||||
'serializer': functools.partial(
|
'serializer': HCI_Object.serialize_length_prefixed_bytes,
|
||||||
HCI_Object.serialize_length_prefixed_bytes
|
|
||||||
),
|
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
@@ -3849,73 +3904,21 @@ class HCI_LE_Set_Extended_Scan_Response_Data_Command(HCI_Command):
|
|||||||
|
|
||||||
|
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
@HCI_Command.command(fields=None)
|
@HCI_Command.command(
|
||||||
|
[
|
||||||
|
('enable', 1),
|
||||||
|
[
|
||||||
|
('advertising_handles', 1),
|
||||||
|
('durations', 2),
|
||||||
|
('max_extended_advertising_events', 1),
|
||||||
|
],
|
||||||
|
]
|
||||||
|
)
|
||||||
class HCI_LE_Set_Extended_Advertising_Enable_Command(HCI_Command):
|
class HCI_LE_Set_Extended_Advertising_Enable_Command(HCI_Command):
|
||||||
'''
|
'''
|
||||||
See Bluetooth spec @ 7.8.56 LE Set Extended Advertising Enable Command
|
See Bluetooth spec @ 7.8.56 LE Set Extended Advertising Enable Command
|
||||||
'''
|
'''
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def from_parameters(cls, parameters):
|
|
||||||
enable = parameters[0]
|
|
||||||
num_sets = parameters[1]
|
|
||||||
advertising_handles = []
|
|
||||||
durations = []
|
|
||||||
max_extended_advertising_events = []
|
|
||||||
offset = 2
|
|
||||||
for _ in range(num_sets):
|
|
||||||
advertising_handles.append(parameters[offset])
|
|
||||||
durations.append(struct.unpack_from('<H', parameters, offset + 1)[0])
|
|
||||||
max_extended_advertising_events.append(parameters[offset + 3])
|
|
||||||
offset += 4
|
|
||||||
|
|
||||||
return cls(
|
|
||||||
enable, advertising_handles, durations, max_extended_advertising_events
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self, enable, advertising_handles, durations, max_extended_advertising_events
|
|
||||||
):
|
|
||||||
super().__init__(HCI_LE_SET_EXTENDED_ADVERTISING_ENABLE_COMMAND)
|
|
||||||
self.enable = enable
|
|
||||||
self.advertising_handles = advertising_handles
|
|
||||||
self.durations = durations
|
|
||||||
self.max_extended_advertising_events = max_extended_advertising_events
|
|
||||||
|
|
||||||
self.parameters = bytes([enable, len(advertising_handles)]) + b''.join(
|
|
||||||
[
|
|
||||||
struct.pack(
|
|
||||||
'<BHB',
|
|
||||||
advertising_handles[i],
|
|
||||||
durations[i],
|
|
||||||
max_extended_advertising_events[i],
|
|
||||||
)
|
|
||||||
for i in range(len(advertising_handles))
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
fields = [('enable:', self.enable)]
|
|
||||||
for i, advertising_handle in enumerate(self.advertising_handles):
|
|
||||||
fields.append(
|
|
||||||
(f'advertising_handle[{i}]: ', advertising_handle)
|
|
||||||
)
|
|
||||||
fields.append((f'duration[{i}]: ', self.durations[i]))
|
|
||||||
fields.append(
|
|
||||||
(
|
|
||||||
f'max_extended_advertising_events[{i}]:',
|
|
||||||
self.max_extended_advertising_events[i],
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
return (
|
|
||||||
color(self.name, 'green')
|
|
||||||
+ ':\n'
|
|
||||||
+ '\n'.join(
|
|
||||||
[color(field[0], 'cyan') + ' ' + str(field[1]) for field in fields]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
@HCI_Command.command(
|
@HCI_Command.command(
|
||||||
@@ -4066,7 +4069,10 @@ class HCI_LE_Set_Extended_Scan_Parameters_Command(HCI_Command):
|
|||||||
color(self.name, 'green')
|
color(self.name, 'green')
|
||||||
+ ':\n'
|
+ ':\n'
|
||||||
+ '\n'.join(
|
+ '\n'.join(
|
||||||
[color(field[0], 'cyan') + ' ' + str(field[1]) for field in fields]
|
[
|
||||||
|
color(' ' + field[0], 'cyan') + ' ' + str(field[1])
|
||||||
|
for field in fields
|
||||||
|
]
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -4242,7 +4248,10 @@ class HCI_LE_Extended_Create_Connection_Command(HCI_Command):
|
|||||||
color(self.name, 'green')
|
color(self.name, 'green')
|
||||||
+ ':\n'
|
+ ':\n'
|
||||||
+ '\n'.join(
|
+ '\n'.join(
|
||||||
[color(field[0], 'cyan') + ' ' + str(field[1]) for field in fields]
|
[
|
||||||
|
color(' ' + field[0], 'cyan') + ' ' + str(field[1])
|
||||||
|
for field in fields
|
||||||
|
]
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -5205,7 +5214,7 @@ class HCI_Number_Of_Completed_Packets_Event(HCI_Event):
|
|||||||
def __str__(self):
|
def __str__(self):
|
||||||
lines = [
|
lines = [
|
||||||
color(self.name, 'magenta') + ':',
|
color(self.name, 'magenta') + ':',
|
||||||
color(' number_of_handles: ', 'cyan')
|
color(' number_of_handles: ', 'cyan')
|
||||||
+ f'{len(self.connection_handles)}',
|
+ f'{len(self.connection_handles)}',
|
||||||
]
|
]
|
||||||
for i, connection_handle in enumerate(self.connection_handles):
|
for i, connection_handle in enumerate(self.connection_handles):
|
||||||
|
|||||||
@@ -0,0 +1,2 @@
|
|||||||
|
/target
|
||||||
|
/.idea
|
||||||
Generated
+1235
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,49 @@
|
|||||||
|
[package]
|
||||||
|
name = "bumble"
|
||||||
|
description = "Rust API for the Bumble Bluetooth stack"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
license = "Apache-2.0"
|
||||||
|
homepage = "https://google.github.io/bumble/index.html"
|
||||||
|
repository = "https://github.com/google/bumble"
|
||||||
|
documentation = "https://docs.rs/crate/bumble"
|
||||||
|
authors = ["Marshall Pierce <marshallpierce@google.com>"]
|
||||||
|
keywords = ["bluetooth", "ble"]
|
||||||
|
categories = ["api-bindings", "network-programming"]
|
||||||
|
rust-version = "1.69.0"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
pyo3 = { version = "0.18.3", features = ["macros"] }
|
||||||
|
pyo3-asyncio = { version = "0.18.0", features = ["tokio-runtime"] }
|
||||||
|
tokio = { version = "1.28.2" }
|
||||||
|
nom = "7.1.3"
|
||||||
|
strum = "0.25.0"
|
||||||
|
strum_macros = "0.25.0"
|
||||||
|
hex = "0.4.3"
|
||||||
|
itertools = "0.11.0"
|
||||||
|
lazy_static = "1.4.0"
|
||||||
|
thiserror = "1.0.41"
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
tokio = { version = "1.28.2", features = ["full"] }
|
||||||
|
tempfile = "3.6.0"
|
||||||
|
nix = "0.26.2"
|
||||||
|
anyhow = "1.0.71"
|
||||||
|
pyo3 = { version = "0.18.3", features = ["macros", "anyhow"] }
|
||||||
|
pyo3-asyncio = { version = "0.18.0", features = ["tokio-runtime", "attributes", "testing"] }
|
||||||
|
clap = { version = "4.3.3", features = ["derive"] }
|
||||||
|
owo-colors = "3.5.0"
|
||||||
|
log = "0.4.19"
|
||||||
|
env_logger = "0.10.0"
|
||||||
|
rusb = "0.9.2"
|
||||||
|
rand = "0.8.5"
|
||||||
|
|
||||||
|
# test entry point that uses pyo3_asyncio's test harness
|
||||||
|
[[test]]
|
||||||
|
name = "pytests"
|
||||||
|
path = "pytests/pytests.rs"
|
||||||
|
harness = false
|
||||||
|
|
||||||
|
[features]
|
||||||
|
anyhow = ["pyo3/anyhow"]
|
||||||
|
pyo3-asyncio-attributes = ["pyo3-asyncio/attributes"]
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
# What is this?
|
||||||
|
|
||||||
|
Rust wrappers around the [Bumble](https://github.com/google/bumble) Python API.
|
||||||
|
|
||||||
|
Method calls are mapped to the equivalent Python, and return types adapted where
|
||||||
|
relevant.
|
||||||
|
|
||||||
|
See the `examples` directory for usage.
|
||||||
|
|
||||||
|
# Usage
|
||||||
|
|
||||||
|
Set up a virtualenv for Bumble, or otherwise have an isolated Python environment
|
||||||
|
for Bumble and its dependencies.
|
||||||
|
|
||||||
|
Due to Python being
|
||||||
|
[picky about how its sys path is set up](https://github.com/PyO3/pyo3/issues/1741,
|
||||||
|
it's necessary to explicitly point to the virtualenv's `site-packages`. Use
|
||||||
|
suitable virtualenv paths as appropriate for your OS, as seen here running
|
||||||
|
the `battery_client` example:
|
||||||
|
|
||||||
|
```
|
||||||
|
PYTHONPATH=..:~/.virtualenvs/bumble/lib/python3.10/site-packages/ \
|
||||||
|
cargo run --example battery_client -- \
|
||||||
|
--transport android-netsim --target-addr F0:F1:F2:F3:F4:F5
|
||||||
|
```
|
||||||
|
|
||||||
|
Run the corresponding `battery_server` Python example, and launch an emulator in
|
||||||
|
Android Studio (currently, Canary is required) to run netsim.
|
||||||
|
|
||||||
|
# Development
|
||||||
|
|
||||||
|
Run the tests:
|
||||||
|
|
||||||
|
```
|
||||||
|
PYTHONPATH=.. cargo test
|
||||||
|
```
|
||||||
|
|
||||||
|
Check lints:
|
||||||
|
|
||||||
|
```
|
||||||
|
cargo clippy --all-targets
|
||||||
|
```
|
||||||
@@ -0,0 +1,112 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Counterpart to the Python example `battery_server.py`.
|
||||||
|
//!
|
||||||
|
//! Start an Android emulator from Android Studio, or otherwise have netsim running.
|
||||||
|
//!
|
||||||
|
//! Run the server from the project root:
|
||||||
|
//! ```
|
||||||
|
//! PYTHONPATH=. python examples/battery_server.py \
|
||||||
|
//! examples/device1.json android-netsim
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! Then run this example from the `rust` directory:
|
||||||
|
//!
|
||||||
|
//! ```
|
||||||
|
//! PYTHONPATH=..:/path/to/virtualenv/site-packages/ \
|
||||||
|
//! cargo run --example battery_client -- \
|
||||||
|
//! --transport android-netsim \
|
||||||
|
//! --target-addr F0:F1:F2:F3:F4:F5
|
||||||
|
//! ```
|
||||||
|
|
||||||
|
use bumble::wrapper::{
|
||||||
|
device::{Device, Peer},
|
||||||
|
profile::BatteryServiceProxy,
|
||||||
|
transport::Transport,
|
||||||
|
PyObjectExt,
|
||||||
|
};
|
||||||
|
use clap::Parser as _;
|
||||||
|
use log::info;
|
||||||
|
use owo_colors::OwoColorize;
|
||||||
|
use pyo3::prelude::*;
|
||||||
|
|
||||||
|
#[pyo3_asyncio::tokio::main]
|
||||||
|
async fn main() -> PyResult<()> {
|
||||||
|
env_logger::builder()
|
||||||
|
.filter_level(log::LevelFilter::Info)
|
||||||
|
.init();
|
||||||
|
|
||||||
|
let cli = Cli::parse();
|
||||||
|
|
||||||
|
let transport = Transport::open(cli.transport).await?;
|
||||||
|
|
||||||
|
let device = Device::with_hci(
|
||||||
|
"Bumble",
|
||||||
|
"F0:F1:F2:F3:F4:F5",
|
||||||
|
transport.source()?,
|
||||||
|
transport.sink()?,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
device.power_on().await?;
|
||||||
|
|
||||||
|
let conn = device.connect(&cli.target_addr).await?;
|
||||||
|
let mut peer = Peer::new(conn)?;
|
||||||
|
for mut s in peer.discover_services().await? {
|
||||||
|
s.discover_characteristics().await?;
|
||||||
|
}
|
||||||
|
let battery_service = peer
|
||||||
|
.create_service_proxy::<BatteryServiceProxy>()?
|
||||||
|
.ok_or(anyhow::anyhow!("No battery service found"))?;
|
||||||
|
|
||||||
|
let mut battery_level_char = battery_service
|
||||||
|
.battery_level()?
|
||||||
|
.ok_or(anyhow::anyhow!("No battery level characteristic"))?;
|
||||||
|
info!(
|
||||||
|
"{} {}",
|
||||||
|
"Initial Battery Level:".green(),
|
||||||
|
battery_level_char
|
||||||
|
.read_value()
|
||||||
|
.await?
|
||||||
|
.extract_with_gil::<u32>()?
|
||||||
|
);
|
||||||
|
battery_level_char
|
||||||
|
.subscribe(|_py, args| {
|
||||||
|
info!(
|
||||||
|
"{} {:?}",
|
||||||
|
"Battery level update:".green(),
|
||||||
|
args.get_item(0)?.extract::<u32>()?,
|
||||||
|
);
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
// wait until user kills the process
|
||||||
|
tokio::signal::ctrl_c().await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(clap::Parser)]
|
||||||
|
#[command(author, version, about, long_about = None)]
|
||||||
|
struct Cli {
|
||||||
|
/// Bumble transport spec.
|
||||||
|
///
|
||||||
|
/// <https://google.github.io/bumble/transports/index.html>
|
||||||
|
#[arg(long)]
|
||||||
|
transport: String,
|
||||||
|
|
||||||
|
/// Address to connect to
|
||||||
|
#[arg(long)]
|
||||||
|
target_addr: String,
|
||||||
|
}
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
use anyhow::anyhow;
|
||||||
|
use bumble::{
|
||||||
|
adv::{AdvertisementDataBuilder, CommonDataType},
|
||||||
|
wrapper::{
|
||||||
|
device::Device,
|
||||||
|
logging::{bumble_env_logging_level, py_logging_basic_config},
|
||||||
|
transport::Transport,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
use clap::Parser as _;
|
||||||
|
use pyo3::PyResult;
|
||||||
|
use rand::Rng;
|
||||||
|
use std::path;
|
||||||
|
|
||||||
|
#[pyo3_asyncio::tokio::main]
|
||||||
|
async fn main() -> PyResult<()> {
|
||||||
|
env_logger::builder()
|
||||||
|
.filter_level(log::LevelFilter::Info)
|
||||||
|
.init();
|
||||||
|
|
||||||
|
let cli = Cli::parse();
|
||||||
|
|
||||||
|
if cli.log_hci {
|
||||||
|
py_logging_basic_config(bumble_env_logging_level("DEBUG"))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let transport = Transport::open(cli.transport).await?;
|
||||||
|
|
||||||
|
let mut device = Device::from_config_file_with_hci(
|
||||||
|
&cli.device_config,
|
||||||
|
transport.source()?,
|
||||||
|
transport.sink()?,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
let mut adv_data = AdvertisementDataBuilder::new();
|
||||||
|
|
||||||
|
adv_data
|
||||||
|
.append(
|
||||||
|
CommonDataType::CompleteLocalName,
|
||||||
|
"Bumble from Rust".as_bytes(),
|
||||||
|
)
|
||||||
|
.map_err(|e| anyhow!(e))?;
|
||||||
|
|
||||||
|
// Randomized TX power
|
||||||
|
adv_data
|
||||||
|
.append(
|
||||||
|
CommonDataType::TxPowerLevel,
|
||||||
|
&[rand::thread_rng().gen_range(-100_i8..=20) as u8],
|
||||||
|
)
|
||||||
|
.map_err(|e| anyhow!(e))?;
|
||||||
|
|
||||||
|
device.set_advertising_data(adv_data)?;
|
||||||
|
device.power_on().await?;
|
||||||
|
|
||||||
|
println!("Advertising...");
|
||||||
|
device.start_advertising(true).await?;
|
||||||
|
|
||||||
|
// wait until user kills the process
|
||||||
|
tokio::signal::ctrl_c().await?;
|
||||||
|
|
||||||
|
println!("Stopping...");
|
||||||
|
device.stop_advertising().await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(clap::Parser)]
|
||||||
|
#[command(author, version, about, long_about = None)]
|
||||||
|
struct Cli {
|
||||||
|
/// Bumble device config.
|
||||||
|
///
|
||||||
|
/// See, for instance, `examples/device1.json` in the Python project.
|
||||||
|
#[arg(long)]
|
||||||
|
device_config: path::PathBuf,
|
||||||
|
/// Bumble transport spec.
|
||||||
|
///
|
||||||
|
/// <https://google.github.io/bumble/transports/index.html>
|
||||||
|
#[arg(long)]
|
||||||
|
transport: String,
|
||||||
|
|
||||||
|
/// Log HCI commands
|
||||||
|
#[arg(long)]
|
||||||
|
log_hci: bool,
|
||||||
|
}
|
||||||
@@ -0,0 +1,185 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Counterpart to the Python example `run_scanner.py`.
|
||||||
|
//!
|
||||||
|
//! Device deduplication is done here rather than relying on the controller's filtering to provide
|
||||||
|
//! for additional features, like the ability to make deduplication time-bounded.
|
||||||
|
|
||||||
|
use bumble::{
|
||||||
|
adv::CommonDataType,
|
||||||
|
wrapper::{
|
||||||
|
core::AdvertisementDataUnit, device::Device, hci::AddressType, transport::Transport,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
use clap::Parser as _;
|
||||||
|
use itertools::Itertools;
|
||||||
|
use owo_colors::{OwoColorize, Style};
|
||||||
|
use pyo3::PyResult;
|
||||||
|
use std::{
|
||||||
|
collections,
|
||||||
|
sync::{Arc, Mutex},
|
||||||
|
time,
|
||||||
|
};
|
||||||
|
|
||||||
|
#[pyo3_asyncio::tokio::main]
|
||||||
|
async fn main() -> PyResult<()> {
|
||||||
|
env_logger::builder()
|
||||||
|
.filter_level(log::LevelFilter::Info)
|
||||||
|
.init();
|
||||||
|
|
||||||
|
let cli = Cli::parse();
|
||||||
|
|
||||||
|
let transport = Transport::open(cli.transport).await?;
|
||||||
|
|
||||||
|
let mut device = Device::with_hci(
|
||||||
|
"Bumble",
|
||||||
|
"F0:F1:F2:F3:F4:F5",
|
||||||
|
transport.source()?,
|
||||||
|
transport.sink()?,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
// in practice, devices can send multiple advertisements from the same address, so we keep
|
||||||
|
// track of a timestamp for each set of data
|
||||||
|
let seen_advertisements = Arc::new(Mutex::new(collections::HashMap::<
|
||||||
|
Vec<u8>,
|
||||||
|
collections::HashMap<Vec<AdvertisementDataUnit>, time::Instant>,
|
||||||
|
>::new()));
|
||||||
|
|
||||||
|
let seen_adv_clone = seen_advertisements.clone();
|
||||||
|
device.on_advertisement(move |_py, adv| {
|
||||||
|
let rssi = adv.rssi()?;
|
||||||
|
let data_units = adv.data()?.data_units()?;
|
||||||
|
let addr = adv.address()?;
|
||||||
|
|
||||||
|
let show_adv = if cli.filter_duplicates {
|
||||||
|
let addr_bytes = addr.as_le_bytes()?;
|
||||||
|
|
||||||
|
let mut seen_adv_cache = seen_adv_clone.lock().unwrap();
|
||||||
|
let expiry_duration = time::Duration::from_secs(cli.dedup_expiry_secs);
|
||||||
|
|
||||||
|
let advs_from_addr = seen_adv_cache
|
||||||
|
.entry(addr_bytes)
|
||||||
|
.or_insert_with(collections::HashMap::new);
|
||||||
|
// we expect cache hits to be the norm, so we do a separate lookup to avoid cloning
|
||||||
|
// on every lookup with entry()
|
||||||
|
let show = if let Some(prev) = advs_from_addr.get_mut(&data_units) {
|
||||||
|
let expired = prev.elapsed() > expiry_duration;
|
||||||
|
*prev = time::Instant::now();
|
||||||
|
expired
|
||||||
|
} else {
|
||||||
|
advs_from_addr.insert(data_units.clone(), time::Instant::now());
|
||||||
|
true
|
||||||
|
};
|
||||||
|
|
||||||
|
// clean out anything we haven't seen in a while
|
||||||
|
advs_from_addr.retain(|_, instant| instant.elapsed() <= expiry_duration);
|
||||||
|
|
||||||
|
show
|
||||||
|
} else {
|
||||||
|
true
|
||||||
|
};
|
||||||
|
|
||||||
|
if !show_adv {
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
|
let addr_style = if adv.is_connectable()? {
|
||||||
|
Style::new().yellow()
|
||||||
|
} else {
|
||||||
|
Style::new().red()
|
||||||
|
};
|
||||||
|
|
||||||
|
let (type_style, qualifier) = match adv.address()?.address_type()? {
|
||||||
|
AddressType::PublicIdentity | AddressType::PublicDevice => (Style::new().cyan(), ""),
|
||||||
|
_ => {
|
||||||
|
if addr.is_static()? {
|
||||||
|
(Style::new().green(), "(static)")
|
||||||
|
} else if addr.is_resolvable()? {
|
||||||
|
(Style::new().magenta(), "(resolvable)")
|
||||||
|
} else {
|
||||||
|
(Style::new().default_color(), "")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
println!(
|
||||||
|
">>> {} [{:?}] {qualifier}:\n RSSI: {}",
|
||||||
|
addr.as_hex()?.style(addr_style),
|
||||||
|
addr.address_type()?.style(type_style),
|
||||||
|
rssi,
|
||||||
|
);
|
||||||
|
|
||||||
|
data_units.into_iter().for_each(|(code, data)| {
|
||||||
|
let matching = CommonDataType::for_type_code(code).collect::<Vec<_>>();
|
||||||
|
let code_str = if matching.is_empty() {
|
||||||
|
format!("0x{}", hex::encode_upper([code.into()]))
|
||||||
|
} else {
|
||||||
|
matching
|
||||||
|
.iter()
|
||||||
|
.map(|t| format!("{}", t))
|
||||||
|
.join(" / ")
|
||||||
|
.blue()
|
||||||
|
.to_string()
|
||||||
|
};
|
||||||
|
|
||||||
|
// use the first matching type's formatted data, if any
|
||||||
|
let data_str = matching
|
||||||
|
.iter()
|
||||||
|
.filter_map(|t| {
|
||||||
|
t.format_data(&data).map(|formatted| {
|
||||||
|
format!(
|
||||||
|
"{} {}",
|
||||||
|
formatted,
|
||||||
|
format!("(raw: 0x{})", hex::encode_upper(&data)).dimmed()
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.next()
|
||||||
|
.unwrap_or_else(|| format!("0x{}", hex::encode_upper(&data)));
|
||||||
|
|
||||||
|
println!(" [{}]: {}", code_str, data_str)
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
})?;
|
||||||
|
|
||||||
|
device.power_on().await?;
|
||||||
|
// do our own dedup
|
||||||
|
device.start_scanning(false).await?;
|
||||||
|
|
||||||
|
// wait until user kills the process
|
||||||
|
tokio::signal::ctrl_c().await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(clap::Parser)]
|
||||||
|
#[command(author, version, about, long_about = None)]
|
||||||
|
struct Cli {
|
||||||
|
/// Bumble transport spec.
|
||||||
|
///
|
||||||
|
/// <https://google.github.io/bumble/transports/index.html>
|
||||||
|
#[arg(long)]
|
||||||
|
transport: String,
|
||||||
|
|
||||||
|
/// Filter duplicate advertisements
|
||||||
|
#[arg(long, default_value_t = false)]
|
||||||
|
filter_duplicates: bool,
|
||||||
|
|
||||||
|
/// How long before a deduplicated advertisement that hasn't been seen in a while is considered
|
||||||
|
/// fresh again, in seconds
|
||||||
|
#[arg(long, default_value_t = 10, requires = "filter_duplicates")]
|
||||||
|
dedup_expiry_secs: u64,
|
||||||
|
}
|
||||||
@@ -0,0 +1,342 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Rust version of the Python `usb_probe.py`.
|
||||||
|
//!
|
||||||
|
//! This tool lists all the USB devices, with details about each device.
|
||||||
|
//! For each device, the different possible Bumble transport strings that can
|
||||||
|
//! refer to it are listed. If the device is known to be a Bluetooth HCI device,
|
||||||
|
//! its identifier is printed in reverse colors, and the transport names in cyan color.
|
||||||
|
//! For other devices, regardless of their type, the transport names are printed
|
||||||
|
//! in red. Whether that device is actually a Bluetooth device or not depends on
|
||||||
|
//! whether it is a Bluetooth device that uses a non-standard Class, or some other
|
||||||
|
//! type of device (there's no way to tell).
|
||||||
|
|
||||||
|
use clap::Parser as _;
|
||||||
|
use itertools::Itertools as _;
|
||||||
|
use owo_colors::{OwoColorize, Style};
|
||||||
|
use rusb::{Device, DeviceDescriptor, Direction, TransferType, UsbContext};
|
||||||
|
use std::{
|
||||||
|
collections::{HashMap, HashSet},
|
||||||
|
time::Duration,
|
||||||
|
};
|
||||||
|
|
||||||
|
const USB_DEVICE_CLASS_DEVICE: u8 = 0x00;
|
||||||
|
const USB_DEVICE_CLASS_WIRELESS_CONTROLLER: u8 = 0xE0;
|
||||||
|
const USB_DEVICE_SUBCLASS_RF_CONTROLLER: u8 = 0x01;
|
||||||
|
const USB_DEVICE_PROTOCOL_BLUETOOTH_PRIMARY_CONTROLLER: u8 = 0x01;
|
||||||
|
|
||||||
|
fn main() -> anyhow::Result<()> {
|
||||||
|
let cli = Cli::parse();
|
||||||
|
|
||||||
|
let mut bt_dev_count = 0;
|
||||||
|
let mut device_serials_by_id: HashMap<(u16, u16), HashSet<String>> = HashMap::new();
|
||||||
|
for device in rusb::devices()?.iter() {
|
||||||
|
let device_desc = device.device_descriptor().unwrap();
|
||||||
|
|
||||||
|
let class_info = ClassInfo::from(&device_desc);
|
||||||
|
let handle = device.open()?;
|
||||||
|
let timeout = Duration::from_secs(1);
|
||||||
|
// some devices don't have languages
|
||||||
|
let lang = handle
|
||||||
|
.read_languages(timeout)
|
||||||
|
.ok()
|
||||||
|
.and_then(|langs| langs.into_iter().next());
|
||||||
|
let serial = lang.and_then(|l| {
|
||||||
|
handle
|
||||||
|
.read_serial_number_string(l, &device_desc, timeout)
|
||||||
|
.ok()
|
||||||
|
});
|
||||||
|
let mfg = lang.and_then(|l| {
|
||||||
|
handle
|
||||||
|
.read_manufacturer_string(l, &device_desc, timeout)
|
||||||
|
.ok()
|
||||||
|
});
|
||||||
|
let product = lang.and_then(|l| handle.read_product_string(l, &device_desc, timeout).ok());
|
||||||
|
|
||||||
|
let is_hci = is_bluetooth_hci(&device, &device_desc)?;
|
||||||
|
let addr_style = if is_hci {
|
||||||
|
bt_dev_count += 1;
|
||||||
|
Style::new().black().on_yellow()
|
||||||
|
} else {
|
||||||
|
Style::new().yellow().on_black()
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut transport_names = Vec::new();
|
||||||
|
let basic_transport_name = format!(
|
||||||
|
"usb:{:04X}:{:04X}",
|
||||||
|
device_desc.vendor_id(),
|
||||||
|
device_desc.product_id()
|
||||||
|
);
|
||||||
|
|
||||||
|
if is_hci {
|
||||||
|
transport_names.push(format!("usb:{}", bt_dev_count - 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
let device_id = (device_desc.vendor_id(), device_desc.product_id());
|
||||||
|
if !device_serials_by_id.contains_key(&device_id) {
|
||||||
|
transport_names.push(basic_transport_name.clone());
|
||||||
|
} else {
|
||||||
|
transport_names.push(format!(
|
||||||
|
"{}#{}",
|
||||||
|
basic_transport_name,
|
||||||
|
device_serials_by_id
|
||||||
|
.get(&device_id)
|
||||||
|
.map(|serials| serials.len())
|
||||||
|
.unwrap_or(0)
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(s) = &serial {
|
||||||
|
if !device_serials_by_id
|
||||||
|
.get(&device_id)
|
||||||
|
.map(|serials| serials.contains(s))
|
||||||
|
.unwrap_or(false)
|
||||||
|
{
|
||||||
|
transport_names.push(format!("{}/{}", basic_transport_name, s))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
println!(
|
||||||
|
"{}",
|
||||||
|
format!(
|
||||||
|
"ID {:04X}:{:04X}",
|
||||||
|
device_desc.vendor_id(),
|
||||||
|
device_desc.product_id()
|
||||||
|
)
|
||||||
|
.style(addr_style)
|
||||||
|
);
|
||||||
|
if !transport_names.is_empty() {
|
||||||
|
let style = if is_hci {
|
||||||
|
Style::new().cyan()
|
||||||
|
} else {
|
||||||
|
Style::new().red()
|
||||||
|
};
|
||||||
|
println!(
|
||||||
|
"{:26}{}",
|
||||||
|
" Bumble Transport Names:".blue(),
|
||||||
|
transport_names.iter().map(|n| n.style(style)).join(" or ")
|
||||||
|
)
|
||||||
|
}
|
||||||
|
println!(
|
||||||
|
"{:26}{:03}/{:03}",
|
||||||
|
" Bus/Device:".green(),
|
||||||
|
device.bus_number(),
|
||||||
|
device.address()
|
||||||
|
);
|
||||||
|
println!(
|
||||||
|
"{:26}{}",
|
||||||
|
" Class:".green(),
|
||||||
|
class_info.formatted_class_name()
|
||||||
|
);
|
||||||
|
println!(
|
||||||
|
"{:26}{}",
|
||||||
|
" Subclass/Protocol:".green(),
|
||||||
|
class_info.formatted_subclass_protocol()
|
||||||
|
);
|
||||||
|
if let Some(s) = serial {
|
||||||
|
println!("{:26}{}", " Serial:".green(), s);
|
||||||
|
device_serials_by_id
|
||||||
|
.entry(device_id)
|
||||||
|
.or_insert(HashSet::new())
|
||||||
|
.insert(s);
|
||||||
|
}
|
||||||
|
if let Some(m) = mfg {
|
||||||
|
println!("{:26}{}", " Manufacturer:".green(), m);
|
||||||
|
}
|
||||||
|
if let Some(p) = product {
|
||||||
|
println!("{:26}{}", " Product:".green(), p);
|
||||||
|
}
|
||||||
|
|
||||||
|
if cli.verbose {
|
||||||
|
print_device_details(&device, &device_desc)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
println!();
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_bluetooth_hci<T: UsbContext>(
|
||||||
|
device: &Device<T>,
|
||||||
|
device_desc: &DeviceDescriptor,
|
||||||
|
) -> rusb::Result<bool> {
|
||||||
|
if device_desc.class_code() == USB_DEVICE_CLASS_WIRELESS_CONTROLLER
|
||||||
|
&& device_desc.sub_class_code() == USB_DEVICE_SUBCLASS_RF_CONTROLLER
|
||||||
|
&& device_desc.protocol_code() == USB_DEVICE_PROTOCOL_BLUETOOTH_PRIMARY_CONTROLLER
|
||||||
|
{
|
||||||
|
Ok(true)
|
||||||
|
} else if device_desc.class_code() == USB_DEVICE_CLASS_DEVICE {
|
||||||
|
for i in 0..device_desc.num_configurations() {
|
||||||
|
for interface in device.config_descriptor(i)?.interfaces() {
|
||||||
|
for d in interface.descriptors() {
|
||||||
|
if d.class_code() == USB_DEVICE_CLASS_WIRELESS_CONTROLLER
|
||||||
|
&& d.sub_class_code() == USB_DEVICE_SUBCLASS_RF_CONTROLLER
|
||||||
|
&& d.protocol_code() == USB_DEVICE_PROTOCOL_BLUETOOTH_PRIMARY_CONTROLLER
|
||||||
|
{
|
||||||
|
return Ok(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(false)
|
||||||
|
} else {
|
||||||
|
Ok(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_device_details<T: UsbContext>(
|
||||||
|
device: &Device<T>,
|
||||||
|
device_desc: &DeviceDescriptor,
|
||||||
|
) -> anyhow::Result<()> {
|
||||||
|
for i in 0..device_desc.num_configurations() {
|
||||||
|
println!(" Configuration {}", i + 1);
|
||||||
|
for interface in device.config_descriptor(i)?.interfaces() {
|
||||||
|
let interface_descriptors: Vec<_> = interface.descriptors().collect();
|
||||||
|
for d in &interface_descriptors {
|
||||||
|
let class_info =
|
||||||
|
ClassInfo::new(d.class_code(), d.sub_class_code(), d.protocol_code());
|
||||||
|
|
||||||
|
println!(
|
||||||
|
" Interface: {}{} ({}, {})",
|
||||||
|
interface.number(),
|
||||||
|
if interface_descriptors.len() > 1 {
|
||||||
|
format!("/{}", d.setting_number())
|
||||||
|
} else {
|
||||||
|
String::new()
|
||||||
|
},
|
||||||
|
class_info.formatted_class_name(),
|
||||||
|
class_info.formatted_subclass_protocol()
|
||||||
|
);
|
||||||
|
|
||||||
|
for e in d.endpoint_descriptors() {
|
||||||
|
println!(
|
||||||
|
" Endpoint {:#04X}: {} {}",
|
||||||
|
e.address(),
|
||||||
|
match e.transfer_type() {
|
||||||
|
TransferType::Control => "CONTROL",
|
||||||
|
TransferType::Isochronous => "ISOCHRONOUS",
|
||||||
|
TransferType::Bulk => "BULK",
|
||||||
|
TransferType::Interrupt => "INTERRUPT",
|
||||||
|
},
|
||||||
|
match e.direction() {
|
||||||
|
Direction::In => "IN",
|
||||||
|
Direction::Out => "OUT",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ClassInfo {
|
||||||
|
class: u8,
|
||||||
|
sub_class: u8,
|
||||||
|
protocol: u8,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClassInfo {
|
||||||
|
fn new(class: u8, sub_class: u8, protocol: u8) -> Self {
|
||||||
|
Self {
|
||||||
|
class,
|
||||||
|
sub_class,
|
||||||
|
protocol,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn class_name(&self) -> Option<&str> {
|
||||||
|
match self.class {
|
||||||
|
0x00 => Some("Device"),
|
||||||
|
0x01 => Some("Audio"),
|
||||||
|
0x02 => Some("Communications and CDC Control"),
|
||||||
|
0x03 => Some("Human Interface Device"),
|
||||||
|
0x05 => Some("Physical"),
|
||||||
|
0x06 => Some("Still Imaging"),
|
||||||
|
0x07 => Some("Printer"),
|
||||||
|
0x08 => Some("Mass Storage"),
|
||||||
|
0x09 => Some("Hub"),
|
||||||
|
0x0A => Some("CDC Data"),
|
||||||
|
0x0B => Some("Smart Card"),
|
||||||
|
0x0D => Some("Content Security"),
|
||||||
|
0x0E => Some("Video"),
|
||||||
|
0x0F => Some("Personal Healthcare"),
|
||||||
|
0x10 => Some("Audio/Video"),
|
||||||
|
0x11 => Some("Billboard"),
|
||||||
|
0x12 => Some("USB Type-C Bridge"),
|
||||||
|
0x3C => Some("I3C"),
|
||||||
|
0xDC => Some("Diagnostic"),
|
||||||
|
USB_DEVICE_CLASS_WIRELESS_CONTROLLER => Some("Wireless Controller"),
|
||||||
|
0xEF => Some("Miscellaneous"),
|
||||||
|
0xFE => Some("Application Specific"),
|
||||||
|
0xFF => Some("Vendor Specific"),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn protocol_name(&self) -> Option<&str> {
|
||||||
|
match self.class {
|
||||||
|
USB_DEVICE_CLASS_WIRELESS_CONTROLLER => match self.sub_class {
|
||||||
|
0x01 => match self.protocol {
|
||||||
|
0x01 => Some("Bluetooth"),
|
||||||
|
0x02 => Some("UWB"),
|
||||||
|
0x03 => Some("Remote NDIS"),
|
||||||
|
0x04 => Some("Bluetooth AMP"),
|
||||||
|
_ => None,
|
||||||
|
},
|
||||||
|
_ => None,
|
||||||
|
},
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn formatted_class_name(&self) -> String {
|
||||||
|
self.class_name()
|
||||||
|
.map(|s| s.to_string())
|
||||||
|
.unwrap_or_else(|| format!("{:#04X}", self.class))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn formatted_subclass_protocol(&self) -> String {
|
||||||
|
format!(
|
||||||
|
"{}/{}{}",
|
||||||
|
self.sub_class,
|
||||||
|
self.protocol,
|
||||||
|
self.protocol_name()
|
||||||
|
.map(|s| format!(" [{}]", s))
|
||||||
|
.unwrap_or_else(String::new)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<&DeviceDescriptor> for ClassInfo {
|
||||||
|
fn from(value: &DeviceDescriptor) -> Self {
|
||||||
|
Self::new(
|
||||||
|
value.class_code(),
|
||||||
|
value.sub_class_code(),
|
||||||
|
value.protocol_code(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(clap::Parser)]
|
||||||
|
#[command(author, version, about, long_about = None)]
|
||||||
|
struct Cli {
|
||||||
|
/// Show additional info for each USB device
|
||||||
|
#[arg(long, default_value_t = false)]
|
||||||
|
verbose: bool,
|
||||||
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
#[pyo3_asyncio::tokio::main]
|
||||||
|
async fn main() -> pyo3::PyResult<()> {
|
||||||
|
pyo3_asyncio::testing::main().await
|
||||||
|
}
|
||||||
|
|
||||||
|
mod wrapper;
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
use bumble::{wrapper, wrapper::transport::Transport};
|
||||||
|
use nix::sys::stat::Mode;
|
||||||
|
use pyo3::prelude::*;
|
||||||
|
|
||||||
|
#[pyo3_asyncio::tokio::test]
|
||||||
|
async fn fifo_transport_can_open() -> PyResult<()> {
|
||||||
|
let dir = tempfile::tempdir().unwrap();
|
||||||
|
let mut fifo = dir.path().to_path_buf();
|
||||||
|
fifo.push("bumble-transport-fifo");
|
||||||
|
nix::unistd::mkfifo(&fifo, Mode::S_IRWXU).unwrap();
|
||||||
|
|
||||||
|
let mut t = Transport::open(format!("file:{}", fifo.to_str().unwrap())).await?;
|
||||||
|
|
||||||
|
t.close().await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[pyo3_asyncio::tokio::test]
|
||||||
|
async fn company_ids() -> PyResult<()> {
|
||||||
|
assert!(wrapper::assigned_numbers::COMPANY_IDS.len() > 2000);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
+446
@@ -0,0 +1,446 @@
|
|||||||
|
//! BLE advertisements.
|
||||||
|
|
||||||
|
use crate::wrapper::assigned_numbers::{COMPANY_IDS, SERVICE_IDS};
|
||||||
|
use crate::wrapper::core::{Uuid128, Uuid16, Uuid32};
|
||||||
|
use itertools::Itertools;
|
||||||
|
use nom::{combinator, multi, number};
|
||||||
|
use std::fmt;
|
||||||
|
use strum::IntoEnumIterator;
|
||||||
|
|
||||||
|
/// The numeric code for a common data type.
|
||||||
|
///
|
||||||
|
/// For known types, see [CommonDataType], or use this type directly for non-assigned codes.
|
||||||
|
#[derive(PartialEq, Eq, Debug, Clone, Copy, Hash)]
|
||||||
|
pub struct CommonDataTypeCode(u8);
|
||||||
|
|
||||||
|
impl From<CommonDataType> for CommonDataTypeCode {
|
||||||
|
fn from(value: CommonDataType) -> Self {
|
||||||
|
let byte = match value {
|
||||||
|
CommonDataType::Flags => 0x01,
|
||||||
|
CommonDataType::IncompleteListOf16BitServiceClassUuids => 0x02,
|
||||||
|
CommonDataType::CompleteListOf16BitServiceClassUuids => 0x03,
|
||||||
|
CommonDataType::IncompleteListOf32BitServiceClassUuids => 0x04,
|
||||||
|
CommonDataType::CompleteListOf32BitServiceClassUuids => 0x05,
|
||||||
|
CommonDataType::IncompleteListOf128BitServiceClassUuids => 0x06,
|
||||||
|
CommonDataType::CompleteListOf128BitServiceClassUuids => 0x07,
|
||||||
|
CommonDataType::ShortenedLocalName => 0x08,
|
||||||
|
CommonDataType::CompleteLocalName => 0x09,
|
||||||
|
CommonDataType::TxPowerLevel => 0x0A,
|
||||||
|
CommonDataType::ClassOfDevice => 0x0D,
|
||||||
|
CommonDataType::SimplePairingHashC192 => 0x0E,
|
||||||
|
CommonDataType::SimplePairingRandomizerR192 => 0x0F,
|
||||||
|
// These two both really have type code 0x10! D:
|
||||||
|
CommonDataType::DeviceId => 0x10,
|
||||||
|
CommonDataType::SecurityManagerTkValue => 0x10,
|
||||||
|
CommonDataType::SecurityManagerOutOfBandFlags => 0x11,
|
||||||
|
CommonDataType::PeripheralConnectionIntervalRange => 0x12,
|
||||||
|
CommonDataType::ListOf16BitServiceSolicitationUuids => 0x14,
|
||||||
|
CommonDataType::ListOf128BitServiceSolicitationUuids => 0x15,
|
||||||
|
CommonDataType::ServiceData16BitUuid => 0x16,
|
||||||
|
CommonDataType::PublicTargetAddress => 0x17,
|
||||||
|
CommonDataType::RandomTargetAddress => 0x18,
|
||||||
|
CommonDataType::Appearance => 0x19,
|
||||||
|
CommonDataType::AdvertisingInterval => 0x1A,
|
||||||
|
CommonDataType::LeBluetoothDeviceAddress => 0x1B,
|
||||||
|
CommonDataType::LeRole => 0x1C,
|
||||||
|
CommonDataType::SimplePairingHashC256 => 0x1D,
|
||||||
|
CommonDataType::SimplePairingRandomizerR256 => 0x1E,
|
||||||
|
CommonDataType::ListOf32BitServiceSolicitationUuids => 0x1F,
|
||||||
|
CommonDataType::ServiceData32BitUuid => 0x20,
|
||||||
|
CommonDataType::ServiceData128BitUuid => 0x21,
|
||||||
|
CommonDataType::LeSecureConnectionsConfirmationValue => 0x22,
|
||||||
|
CommonDataType::LeSecureConnectionsRandomValue => 0x23,
|
||||||
|
CommonDataType::Uri => 0x24,
|
||||||
|
CommonDataType::IndoorPositioning => 0x25,
|
||||||
|
CommonDataType::TransportDiscoveryData => 0x26,
|
||||||
|
CommonDataType::LeSupportedFeatures => 0x27,
|
||||||
|
CommonDataType::ChannelMapUpdateIndication => 0x28,
|
||||||
|
CommonDataType::PbAdv => 0x29,
|
||||||
|
CommonDataType::MeshMessage => 0x2A,
|
||||||
|
CommonDataType::MeshBeacon => 0x2B,
|
||||||
|
CommonDataType::BigInfo => 0x2C,
|
||||||
|
CommonDataType::BroadcastCode => 0x2D,
|
||||||
|
CommonDataType::ResolvableSetIdentifier => 0x2E,
|
||||||
|
CommonDataType::AdvertisingIntervalLong => 0x2F,
|
||||||
|
CommonDataType::ThreeDInformationData => 0x3D,
|
||||||
|
CommonDataType::ManufacturerSpecificData => 0xFF,
|
||||||
|
};
|
||||||
|
|
||||||
|
Self(byte)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<u8> for CommonDataTypeCode {
|
||||||
|
fn from(value: u8) -> Self {
|
||||||
|
Self(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<CommonDataTypeCode> for u8 {
|
||||||
|
fn from(value: CommonDataTypeCode) -> Self {
|
||||||
|
value.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Data types for assigned type codes.
|
||||||
|
///
|
||||||
|
/// See Bluetooth Assigned Numbers § 2.3
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, strum_macros::EnumIter)]
|
||||||
|
#[allow(missing_docs)]
|
||||||
|
pub enum CommonDataType {
|
||||||
|
Flags,
|
||||||
|
IncompleteListOf16BitServiceClassUuids,
|
||||||
|
CompleteListOf16BitServiceClassUuids,
|
||||||
|
IncompleteListOf32BitServiceClassUuids,
|
||||||
|
CompleteListOf32BitServiceClassUuids,
|
||||||
|
IncompleteListOf128BitServiceClassUuids,
|
||||||
|
CompleteListOf128BitServiceClassUuids,
|
||||||
|
ShortenedLocalName,
|
||||||
|
CompleteLocalName,
|
||||||
|
TxPowerLevel,
|
||||||
|
ClassOfDevice,
|
||||||
|
SimplePairingHashC192,
|
||||||
|
SimplePairingRandomizerR192,
|
||||||
|
DeviceId,
|
||||||
|
SecurityManagerTkValue,
|
||||||
|
SecurityManagerOutOfBandFlags,
|
||||||
|
PeripheralConnectionIntervalRange,
|
||||||
|
ListOf16BitServiceSolicitationUuids,
|
||||||
|
ListOf128BitServiceSolicitationUuids,
|
||||||
|
ServiceData16BitUuid,
|
||||||
|
PublicTargetAddress,
|
||||||
|
RandomTargetAddress,
|
||||||
|
Appearance,
|
||||||
|
AdvertisingInterval,
|
||||||
|
LeBluetoothDeviceAddress,
|
||||||
|
LeRole,
|
||||||
|
SimplePairingHashC256,
|
||||||
|
SimplePairingRandomizerR256,
|
||||||
|
ListOf32BitServiceSolicitationUuids,
|
||||||
|
ServiceData32BitUuid,
|
||||||
|
ServiceData128BitUuid,
|
||||||
|
LeSecureConnectionsConfirmationValue,
|
||||||
|
LeSecureConnectionsRandomValue,
|
||||||
|
Uri,
|
||||||
|
IndoorPositioning,
|
||||||
|
TransportDiscoveryData,
|
||||||
|
LeSupportedFeatures,
|
||||||
|
ChannelMapUpdateIndication,
|
||||||
|
PbAdv,
|
||||||
|
MeshMessage,
|
||||||
|
MeshBeacon,
|
||||||
|
BigInfo,
|
||||||
|
BroadcastCode,
|
||||||
|
ResolvableSetIdentifier,
|
||||||
|
AdvertisingIntervalLong,
|
||||||
|
ThreeDInformationData,
|
||||||
|
ManufacturerSpecificData,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CommonDataType {
|
||||||
|
/// Iterate over the zero, one, or more matching types for the provided code.
|
||||||
|
///
|
||||||
|
/// `0x10` maps to both Device Id and Security Manager TK Value, so multiple matching types
|
||||||
|
/// may exist for a single code.
|
||||||
|
pub fn for_type_code(code: CommonDataTypeCode) -> impl Iterator<Item = CommonDataType> {
|
||||||
|
Self::iter().filter(move |t| CommonDataTypeCode::from(*t) == code)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Apply type-specific human-oriented formatting to data, if any is applicable
|
||||||
|
pub fn format_data(&self, data: &[u8]) -> Option<String> {
|
||||||
|
match self {
|
||||||
|
Self::Flags => Some(Flags::matching(data).map(|f| format!("{:?}", f)).join(",")),
|
||||||
|
Self::CompleteListOf16BitServiceClassUuids
|
||||||
|
| Self::IncompleteListOf16BitServiceClassUuids
|
||||||
|
| Self::ListOf16BitServiceSolicitationUuids => {
|
||||||
|
combinator::complete(multi::many0(Uuid16::parse_le))(data)
|
||||||
|
.map(|(_res, uuids)| {
|
||||||
|
uuids
|
||||||
|
.into_iter()
|
||||||
|
.map(|uuid| {
|
||||||
|
SERVICE_IDS
|
||||||
|
.get(&uuid)
|
||||||
|
.map(|name| format!("{:?} ({name})", uuid))
|
||||||
|
.unwrap_or_else(|| format!("{:?}", uuid))
|
||||||
|
})
|
||||||
|
.join(", ")
|
||||||
|
})
|
||||||
|
.ok()
|
||||||
|
}
|
||||||
|
Self::CompleteListOf32BitServiceClassUuids
|
||||||
|
| Self::IncompleteListOf32BitServiceClassUuids
|
||||||
|
| Self::ListOf32BitServiceSolicitationUuids => {
|
||||||
|
combinator::complete(multi::many0(Uuid32::parse))(data)
|
||||||
|
.map(|(_res, uuids)| uuids.into_iter().map(|u| format!("{:?}", u)).join(", "))
|
||||||
|
.ok()
|
||||||
|
}
|
||||||
|
Self::CompleteListOf128BitServiceClassUuids
|
||||||
|
| Self::IncompleteListOf128BitServiceClassUuids
|
||||||
|
| Self::ListOf128BitServiceSolicitationUuids => {
|
||||||
|
combinator::complete(multi::many0(Uuid128::parse_le))(data)
|
||||||
|
.map(|(_res, uuids)| uuids.into_iter().map(|u| format!("{:?}", u)).join(", "))
|
||||||
|
.ok()
|
||||||
|
}
|
||||||
|
Self::ServiceData16BitUuid => Uuid16::parse_le(data)
|
||||||
|
.map(|(rem, uuid)| {
|
||||||
|
format!(
|
||||||
|
"service={:?}, data={}",
|
||||||
|
SERVICE_IDS
|
||||||
|
.get(&uuid)
|
||||||
|
.map(|name| format!("{:?} ({name})", uuid))
|
||||||
|
.unwrap_or_else(|| format!("{:?}", uuid)),
|
||||||
|
hex::encode_upper(rem)
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.ok(),
|
||||||
|
Self::ServiceData32BitUuid => Uuid32::parse(data)
|
||||||
|
.map(|(rem, uuid)| format!("service={:?}, data={}", uuid, hex::encode_upper(rem)))
|
||||||
|
.ok(),
|
||||||
|
Self::ServiceData128BitUuid => Uuid128::parse_le(data)
|
||||||
|
.map(|(rem, uuid)| format!("service={:?}, data={}", uuid, hex::encode_upper(rem)))
|
||||||
|
.ok(),
|
||||||
|
Self::ShortenedLocalName | Self::CompleteLocalName => {
|
||||||
|
std::str::from_utf8(data).ok().map(|s| format!("\"{}\"", s))
|
||||||
|
}
|
||||||
|
Self::TxPowerLevel => {
|
||||||
|
let (_, tx) =
|
||||||
|
combinator::complete(number::complete::i8::<_, nom::error::Error<_>>)(data)
|
||||||
|
.ok()?;
|
||||||
|
|
||||||
|
Some(tx.to_string())
|
||||||
|
}
|
||||||
|
Self::ManufacturerSpecificData => {
|
||||||
|
let (rem, id) = Uuid16::parse_le(data).ok()?;
|
||||||
|
Some(format!(
|
||||||
|
"company={}, data=0x{}",
|
||||||
|
COMPANY_IDS
|
||||||
|
.get(&id)
|
||||||
|
.map(|s| s.to_string())
|
||||||
|
.unwrap_or_else(|| format!("{:?}", id)),
|
||||||
|
hex::encode_upper(rem)
|
||||||
|
))
|
||||||
|
}
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for CommonDataType {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
match self {
|
||||||
|
CommonDataType::Flags => write!(f, "Flags"),
|
||||||
|
CommonDataType::IncompleteListOf16BitServiceClassUuids => {
|
||||||
|
write!(f, "Incomplete List of 16-bit Service Class UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::CompleteListOf16BitServiceClassUuids => {
|
||||||
|
write!(f, "Complete List of 16-bit Service Class UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::IncompleteListOf32BitServiceClassUuids => {
|
||||||
|
write!(f, "Incomplete List of 32-bit Service Class UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::CompleteListOf32BitServiceClassUuids => {
|
||||||
|
write!(f, "Complete List of 32-bit Service Class UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::ListOf16BitServiceSolicitationUuids => {
|
||||||
|
write!(f, "List of 16-bit Service Solicitation UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::ListOf32BitServiceSolicitationUuids => {
|
||||||
|
write!(f, "List of 32-bit Service Solicitation UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::ListOf128BitServiceSolicitationUuids => {
|
||||||
|
write!(f, "List of 128-bit Service Solicitation UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::IncompleteListOf128BitServiceClassUuids => {
|
||||||
|
write!(f, "Incomplete List of 128-bit Service Class UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::CompleteListOf128BitServiceClassUuids => {
|
||||||
|
write!(f, "Complete List of 128-bit Service Class UUIDs")
|
||||||
|
}
|
||||||
|
CommonDataType::ShortenedLocalName => write!(f, "Shortened Local Name"),
|
||||||
|
CommonDataType::CompleteLocalName => write!(f, "Complete Local Name"),
|
||||||
|
CommonDataType::TxPowerLevel => write!(f, "TX Power Level"),
|
||||||
|
CommonDataType::ClassOfDevice => write!(f, "Class of Device"),
|
||||||
|
CommonDataType::SimplePairingHashC192 => {
|
||||||
|
write!(f, "Simple Pairing Hash C-192")
|
||||||
|
}
|
||||||
|
CommonDataType::SimplePairingHashC256 => {
|
||||||
|
write!(f, "Simple Pairing Hash C 256")
|
||||||
|
}
|
||||||
|
CommonDataType::SimplePairingRandomizerR192 => {
|
||||||
|
write!(f, "Simple Pairing Randomizer R-192")
|
||||||
|
}
|
||||||
|
CommonDataType::SimplePairingRandomizerR256 => {
|
||||||
|
write!(f, "Simple Pairing Randomizer R 256")
|
||||||
|
}
|
||||||
|
CommonDataType::DeviceId => write!(f, "Device Id"),
|
||||||
|
CommonDataType::SecurityManagerTkValue => {
|
||||||
|
write!(f, "Security Manager TK Value")
|
||||||
|
}
|
||||||
|
CommonDataType::SecurityManagerOutOfBandFlags => {
|
||||||
|
write!(f, "Security Manager Out of Band Flags")
|
||||||
|
}
|
||||||
|
CommonDataType::PeripheralConnectionIntervalRange => {
|
||||||
|
write!(f, "Peripheral Connection Interval Range")
|
||||||
|
}
|
||||||
|
CommonDataType::ServiceData16BitUuid => {
|
||||||
|
write!(f, "Service Data 16-bit UUID")
|
||||||
|
}
|
||||||
|
CommonDataType::ServiceData32BitUuid => {
|
||||||
|
write!(f, "Service Data 32-bit UUID")
|
||||||
|
}
|
||||||
|
CommonDataType::ServiceData128BitUuid => {
|
||||||
|
write!(f, "Service Data 128-bit UUID")
|
||||||
|
}
|
||||||
|
CommonDataType::PublicTargetAddress => write!(f, "Public Target Address"),
|
||||||
|
CommonDataType::RandomTargetAddress => write!(f, "Random Target Address"),
|
||||||
|
CommonDataType::Appearance => write!(f, "Appearance"),
|
||||||
|
CommonDataType::AdvertisingInterval => write!(f, "Advertising Interval"),
|
||||||
|
CommonDataType::LeBluetoothDeviceAddress => {
|
||||||
|
write!(f, "LE Bluetooth Device Address")
|
||||||
|
}
|
||||||
|
CommonDataType::LeRole => write!(f, "LE Role"),
|
||||||
|
CommonDataType::LeSecureConnectionsConfirmationValue => {
|
||||||
|
write!(f, "LE Secure Connections Confirmation Value")
|
||||||
|
}
|
||||||
|
CommonDataType::LeSecureConnectionsRandomValue => {
|
||||||
|
write!(f, "LE Secure Connections Random Value")
|
||||||
|
}
|
||||||
|
CommonDataType::LeSupportedFeatures => write!(f, "LE Supported Features"),
|
||||||
|
CommonDataType::Uri => write!(f, "URI"),
|
||||||
|
CommonDataType::IndoorPositioning => write!(f, "Indoor Positioning"),
|
||||||
|
CommonDataType::TransportDiscoveryData => {
|
||||||
|
write!(f, "Transport Discovery Data")
|
||||||
|
}
|
||||||
|
CommonDataType::ChannelMapUpdateIndication => {
|
||||||
|
write!(f, "Channel Map Update Indication")
|
||||||
|
}
|
||||||
|
CommonDataType::PbAdv => write!(f, "PB-ADV"),
|
||||||
|
CommonDataType::MeshMessage => write!(f, "Mesh Message"),
|
||||||
|
CommonDataType::MeshBeacon => write!(f, "Mesh Beacon"),
|
||||||
|
CommonDataType::BigInfo => write!(f, "BIGIInfo"),
|
||||||
|
CommonDataType::BroadcastCode => write!(f, "Broadcast Code"),
|
||||||
|
CommonDataType::ResolvableSetIdentifier => {
|
||||||
|
write!(f, "Resolvable Set Identifier")
|
||||||
|
}
|
||||||
|
CommonDataType::AdvertisingIntervalLong => {
|
||||||
|
write!(f, "Advertising Interval Long")
|
||||||
|
}
|
||||||
|
CommonDataType::ThreeDInformationData => write!(f, "3D Information Data"),
|
||||||
|
CommonDataType::ManufacturerSpecificData => {
|
||||||
|
write!(f, "Manufacturer Specific Data")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Accumulates advertisement data to broadcast on a [crate::wrapper::device::Device].
|
||||||
|
#[derive(Debug, Clone, Default)]
|
||||||
|
pub struct AdvertisementDataBuilder {
|
||||||
|
encoded_data: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AdvertisementDataBuilder {
|
||||||
|
/// Returns a new, empty instance.
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
encoded_data: Vec::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Append advertising data to the builder.
|
||||||
|
///
|
||||||
|
/// Returns an error if the data cannot be appended.
|
||||||
|
pub fn append(
|
||||||
|
&mut self,
|
||||||
|
type_code: impl Into<CommonDataTypeCode>,
|
||||||
|
data: &[u8],
|
||||||
|
) -> Result<(), AdvertisementDataBuilderError> {
|
||||||
|
self.encoded_data.push(
|
||||||
|
data.len()
|
||||||
|
.try_into()
|
||||||
|
.ok()
|
||||||
|
.and_then(|len: u8| len.checked_add(1))
|
||||||
|
.ok_or(AdvertisementDataBuilderError::DataTooLong)?,
|
||||||
|
);
|
||||||
|
self.encoded_data.push(type_code.into().0);
|
||||||
|
self.encoded_data.extend_from_slice(data);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn into_bytes(self) -> Vec<u8> {
|
||||||
|
self.encoded_data
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Errors that can occur when building advertisement data with [AdvertisementDataBuilder].
|
||||||
|
#[derive(Debug, PartialEq, Eq, thiserror::Error)]
|
||||||
|
pub enum AdvertisementDataBuilderError {
|
||||||
|
/// The provided adv data is too long to be encoded
|
||||||
|
#[error("Data too long")]
|
||||||
|
DataTooLong,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Eq, strum_macros::EnumIter)]
|
||||||
|
#[allow(missing_docs)]
|
||||||
|
/// Features in the Flags AD
|
||||||
|
pub enum Flags {
|
||||||
|
LeLimited,
|
||||||
|
LeDiscoverable,
|
||||||
|
NoBrEdr,
|
||||||
|
BrEdrController,
|
||||||
|
BrEdrHost,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for Flags {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
write!(f, "{}", self.short_name())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Flags {
|
||||||
|
/// Iterates over the flags that are present in the provided `flags` bytes.
|
||||||
|
pub fn matching(flags: &[u8]) -> impl Iterator<Item = Self> + '_ {
|
||||||
|
// The encoding is not clear from the spec: do we look at the first byte? or the last?
|
||||||
|
// In practice it's only one byte.
|
||||||
|
let first_byte = flags.first().unwrap_or(&0_u8);
|
||||||
|
|
||||||
|
Self::iter().filter(move |f| {
|
||||||
|
let mask = match f {
|
||||||
|
Flags::LeLimited => 0x01_u8,
|
||||||
|
Flags::LeDiscoverable => 0x02,
|
||||||
|
Flags::NoBrEdr => 0x04,
|
||||||
|
Flags::BrEdrController => 0x08,
|
||||||
|
Flags::BrEdrHost => 0x10,
|
||||||
|
};
|
||||||
|
|
||||||
|
mask & first_byte > 0
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An abbreviated form of the flag name.
|
||||||
|
///
|
||||||
|
/// See [Flags::name] for the full name.
|
||||||
|
pub fn short_name(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
Flags::LeLimited => "LE Limited",
|
||||||
|
Flags::LeDiscoverable => "LE General",
|
||||||
|
Flags::NoBrEdr => "No BR/EDR",
|
||||||
|
Flags::BrEdrController => "BR/EDR C",
|
||||||
|
Flags::BrEdrHost => "BR/EDR H",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The human-readable name of the flag.
|
||||||
|
///
|
||||||
|
/// See [Flags::short_name] for a shorter string for use if compactness is important.
|
||||||
|
pub fn name(&self) -> &'static str {
|
||||||
|
match self {
|
||||||
|
Flags::LeLimited => "LE Limited Discoverable Mode",
|
||||||
|
Flags::LeDiscoverable => "LE General Discoverable Mode",
|
||||||
|
Flags::NoBrEdr => "BR/EDR Not Supported",
|
||||||
|
Flags::BrEdrController => "Simultaneous LE and BR/EDR (Controller)",
|
||||||
|
Flags::BrEdrHost => "Simultaneous LE and BR/EDR (Host)",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Rust API for [Bumble](https://github.com/google/bumble).
|
||||||
|
//!
|
||||||
|
//! Bumble is a userspace Bluetooth stack that works with more or less anything that uses HCI. This
|
||||||
|
//! could be physical Bluetooth USB dongles, netsim, HCI proxied over a network from some device
|
||||||
|
//! elsewhere, etc.
|
||||||
|
//!
|
||||||
|
//! It also does not restrict what you can do with Bluetooth the way that OS Bluetooth APIs
|
||||||
|
//! typically do, making it good for prototyping, experimentation, test tools, etc.
|
||||||
|
//!
|
||||||
|
//! Bumble is primarily written in Python. Rust types that wrap the Python API, which is currently
|
||||||
|
//! the bulk of the code, are in the [wrapper] module.
|
||||||
|
|
||||||
|
#![deny(missing_docs, unsafe_code)]
|
||||||
|
|
||||||
|
pub mod wrapper;
|
||||||
|
|
||||||
|
pub mod adv;
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Assigned numbers from the Bluetooth spec.
|
||||||
|
|
||||||
|
use crate::wrapper::core::Uuid16;
|
||||||
|
use lazy_static::lazy_static;
|
||||||
|
use pyo3::{
|
||||||
|
intern,
|
||||||
|
types::{PyDict, PyModule},
|
||||||
|
PyResult, Python,
|
||||||
|
};
|
||||||
|
use std::collections;
|
||||||
|
|
||||||
|
mod services;
|
||||||
|
|
||||||
|
pub use services::SERVICE_IDS;
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
/// Assigned company IDs
|
||||||
|
pub static ref COMPANY_IDS: collections::HashMap<Uuid16, String> = load_company_ids()
|
||||||
|
.expect("Could not load company ids -- are Bumble's Python sources available?");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
fn load_company_ids() -> PyResult<collections::HashMap<Uuid16, String>> {
|
||||||
|
// this takes about 4ms on a fast machine -- slower than constructing in rust, but not slow
|
||||||
|
// enough to worry about
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
PyModule::import(py, intern!(py, "bumble.company_ids"))?
|
||||||
|
.getattr(intern!(py, "COMPANY_IDENTIFIERS"))?
|
||||||
|
.downcast::<PyDict>()?
|
||||||
|
.into_iter()
|
||||||
|
.map(|(k, v)| {
|
||||||
|
Ok((
|
||||||
|
Uuid16::from_be_bytes(k.extract::<u16>()?.to_be_bytes()),
|
||||||
|
v.str()?.to_str()?.to_string(),
|
||||||
|
))
|
||||||
|
})
|
||||||
|
.collect::<PyResult<collections::HashMap<_, _>>>()
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -0,0 +1,82 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Assigned service IDs
|
||||||
|
|
||||||
|
use crate::wrapper::core::Uuid16;
|
||||||
|
use lazy_static::lazy_static;
|
||||||
|
use std::collections;
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
/// Assigned service IDs
|
||||||
|
pub static ref SERVICE_IDS: collections::HashMap<Uuid16, &'static str> = [
|
||||||
|
(0x1800_u16, "Generic Access"),
|
||||||
|
(0x1801, "Generic Attribute"),
|
||||||
|
(0x1802, "Immediate Alert"),
|
||||||
|
(0x1803, "Link Loss"),
|
||||||
|
(0x1804, "TX Power"),
|
||||||
|
(0x1805, "Current Time"),
|
||||||
|
(0x1806, "Reference Time Update"),
|
||||||
|
(0x1807, "Next DST Change"),
|
||||||
|
(0x1808, "Glucose"),
|
||||||
|
(0x1809, "Health Thermometer"),
|
||||||
|
(0x180A, "Device Information"),
|
||||||
|
(0x180D, "Heart Rate"),
|
||||||
|
(0x180E, "Phone Alert Status"),
|
||||||
|
(0x180F, "Battery"),
|
||||||
|
(0x1810, "Blood Pressure"),
|
||||||
|
(0x1811, "Alert Notification"),
|
||||||
|
(0x1812, "Human Interface Device"),
|
||||||
|
(0x1813, "Scan Parameters"),
|
||||||
|
(0x1814, "Running Speed and Cadence"),
|
||||||
|
(0x1815, "Automation IO"),
|
||||||
|
(0x1816, "Cycling Speed and Cadence"),
|
||||||
|
(0x1818, "Cycling Power"),
|
||||||
|
(0x1819, "Location and Navigation"),
|
||||||
|
(0x181A, "Environmental Sensing"),
|
||||||
|
(0x181B, "Body Composition"),
|
||||||
|
(0x181C, "User Data"),
|
||||||
|
(0x181D, "Weight Scale"),
|
||||||
|
(0x181E, "Bond Management"),
|
||||||
|
(0x181F, "Continuous Glucose Monitoring"),
|
||||||
|
(0x1820, "Internet Protocol Support"),
|
||||||
|
(0x1821, "Indoor Positioning"),
|
||||||
|
(0x1822, "Pulse Oximeter"),
|
||||||
|
(0x1823, "HTTP Proxy"),
|
||||||
|
(0x1824, "Transport Discovery"),
|
||||||
|
(0x1825, "Object Transfer"),
|
||||||
|
(0x1826, "Fitness Machine"),
|
||||||
|
(0x1827, "Mesh Provisioning"),
|
||||||
|
(0x1828, "Mesh Proxy"),
|
||||||
|
(0x1829, "Reconnection Configuration"),
|
||||||
|
(0x183A, "Insulin Delivery"),
|
||||||
|
(0x183B, "Binary Sensor"),
|
||||||
|
(0x183C, "Emergency Configuration"),
|
||||||
|
(0x183E, "Physical Activity Monitor"),
|
||||||
|
(0x1843, "Audio Input Control"),
|
||||||
|
(0x1844, "Volume Control"),
|
||||||
|
(0x1845, "Volume Offset Control"),
|
||||||
|
(0x1846, "Coordinated Set Identification Service"),
|
||||||
|
(0x1847, "Device Time"),
|
||||||
|
(0x1848, "Media Control Service"),
|
||||||
|
(0x1849, "Generic Media Control Service"),
|
||||||
|
(0x184A, "Constant Tone Extension"),
|
||||||
|
(0x184B, "Telephone Bearer Service"),
|
||||||
|
(0x184C, "Generic Telephone Bearer Service"),
|
||||||
|
(0x184D, "Microphone Control"),
|
||||||
|
]
|
||||||
|
.into_iter()
|
||||||
|
.map(|(num, name)| (Uuid16::from_le_bytes(num.to_le_bytes()), name))
|
||||||
|
.collect();
|
||||||
|
}
|
||||||
@@ -0,0 +1,196 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Core types
|
||||||
|
|
||||||
|
use crate::adv::CommonDataTypeCode;
|
||||||
|
use lazy_static::lazy_static;
|
||||||
|
use nom::{bytes, combinator};
|
||||||
|
use pyo3::{intern, PyObject, PyResult, Python};
|
||||||
|
use std::fmt;
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
static ref BASE_UUID: [u8; 16] = hex::decode("0000000000001000800000805F9B34FB")
|
||||||
|
.unwrap()
|
||||||
|
.try_into()
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A type code and data pair from an advertisement
|
||||||
|
pub type AdvertisementDataUnit = (CommonDataTypeCode, Vec<u8>);
|
||||||
|
|
||||||
|
/// Contents of an advertisement
|
||||||
|
pub struct AdvertisingData(pub(crate) PyObject);
|
||||||
|
|
||||||
|
impl AdvertisingData {
|
||||||
|
/// Data units in the advertisement contents
|
||||||
|
pub fn data_units(&self) -> PyResult<Vec<AdvertisementDataUnit>> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
let list = self.0.getattr(py, intern!(py, "ad_structures"))?;
|
||||||
|
|
||||||
|
list.as_ref(py)
|
||||||
|
.iter()?
|
||||||
|
.collect::<Result<Vec<_>, _>>()?
|
||||||
|
.into_iter()
|
||||||
|
.map(|tuple| {
|
||||||
|
let type_code = tuple
|
||||||
|
.call_method1(intern!(py, "__getitem__"), (0,))?
|
||||||
|
.extract::<u8>()?
|
||||||
|
.into();
|
||||||
|
let data = tuple
|
||||||
|
.call_method1(intern!(py, "__getitem__"), (1,))?
|
||||||
|
.extract::<Vec<u8>>()?;
|
||||||
|
Ok((type_code, data))
|
||||||
|
})
|
||||||
|
.collect::<Result<Vec<_>, _>>()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 16-bit UUID
|
||||||
|
#[derive(PartialEq, Eq, Hash)]
|
||||||
|
pub struct Uuid16 {
|
||||||
|
/// Big-endian bytes
|
||||||
|
uuid: [u8; 2],
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Uuid16 {
|
||||||
|
/// Construct a UUID from little-endian bytes
|
||||||
|
pub fn from_le_bytes(mut bytes: [u8; 2]) -> Self {
|
||||||
|
bytes.reverse();
|
||||||
|
Self::from_be_bytes(bytes)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Construct a UUID from big-endian bytes
|
||||||
|
pub fn from_be_bytes(bytes: [u8; 2]) -> Self {
|
||||||
|
Self { uuid: bytes }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The UUID in big-endian bytes form
|
||||||
|
pub fn as_be_bytes(&self) -> [u8; 2] {
|
||||||
|
self.uuid
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The UUID in little-endian bytes form
|
||||||
|
pub fn as_le_bytes(&self) -> [u8; 2] {
|
||||||
|
let mut uuid = self.uuid;
|
||||||
|
uuid.reverse();
|
||||||
|
uuid
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn parse_le(input: &[u8]) -> nom::IResult<&[u8], Self> {
|
||||||
|
combinator::map_res(bytes::complete::take(2_usize), |b: &[u8]| {
|
||||||
|
b.try_into().map(|mut uuid: [u8; 2]| {
|
||||||
|
uuid.reverse();
|
||||||
|
Self { uuid }
|
||||||
|
})
|
||||||
|
})(input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for Uuid16 {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
write!(f, "UUID-16:{}", hex::encode_upper(self.uuid))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 32-bit UUID
|
||||||
|
#[derive(PartialEq, Eq, Hash)]
|
||||||
|
pub struct Uuid32 {
|
||||||
|
/// Big-endian bytes
|
||||||
|
uuid: [u8; 4],
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Uuid32 {
|
||||||
|
/// The UUID in big-endian bytes form
|
||||||
|
pub fn as_bytes(&self) -> [u8; 4] {
|
||||||
|
self.uuid
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn parse(input: &[u8]) -> nom::IResult<&[u8], Self> {
|
||||||
|
combinator::map_res(bytes::complete::take(4_usize), |b: &[u8]| {
|
||||||
|
b.try_into().map(|mut uuid: [u8; 4]| {
|
||||||
|
uuid.reverse();
|
||||||
|
Self { uuid }
|
||||||
|
})
|
||||||
|
})(input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for Uuid32 {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
write!(f, "UUID-32:{}", hex::encode_upper(self.uuid))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Uuid16> for Uuid32 {
|
||||||
|
fn from(value: Uuid16) -> Self {
|
||||||
|
let mut uuid = [0; 4];
|
||||||
|
uuid[2..].copy_from_slice(&value.uuid);
|
||||||
|
Self { uuid }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 128-bit UUID
|
||||||
|
#[derive(PartialEq, Eq, Hash)]
|
||||||
|
pub struct Uuid128 {
|
||||||
|
/// Big-endian bytes
|
||||||
|
uuid: [u8; 16],
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Uuid128 {
|
||||||
|
/// The UUID in big-endian bytes form
|
||||||
|
pub fn as_bytes(&self) -> [u8; 16] {
|
||||||
|
self.uuid
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn parse_le(input: &[u8]) -> nom::IResult<&[u8], Self> {
|
||||||
|
combinator::map_res(bytes::complete::take(16_usize), |b: &[u8]| {
|
||||||
|
b.try_into().map(|mut uuid: [u8; 16]| {
|
||||||
|
uuid.reverse();
|
||||||
|
Self { uuid }
|
||||||
|
})
|
||||||
|
})(input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for Uuid128 {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
write!(
|
||||||
|
f,
|
||||||
|
"{}-{}-{}-{}-{}",
|
||||||
|
hex::encode_upper(&self.uuid[..4]),
|
||||||
|
hex::encode_upper(&self.uuid[4..6]),
|
||||||
|
hex::encode_upper(&self.uuid[6..8]),
|
||||||
|
hex::encode_upper(&self.uuid[8..10]),
|
||||||
|
hex::encode_upper(&self.uuid[10..])
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Uuid16> for Uuid128 {
|
||||||
|
fn from(value: Uuid16) -> Self {
|
||||||
|
let mut uuid = *BASE_UUID;
|
||||||
|
uuid[2..4].copy_from_slice(&value.uuid);
|
||||||
|
Self { uuid }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Uuid32> for Uuid128 {
|
||||||
|
fn from(value: Uuid32) -> Self {
|
||||||
|
let mut uuid = *BASE_UUID;
|
||||||
|
uuid[..4].copy_from_slice(&value.uuid);
|
||||||
|
Self { uuid }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,248 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Devices and connections to them
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
adv::AdvertisementDataBuilder,
|
||||||
|
wrapper::{
|
||||||
|
core::AdvertisingData,
|
||||||
|
gatt_client::{ProfileServiceProxy, ServiceProxy},
|
||||||
|
hci::Address,
|
||||||
|
transport::{Sink, Source},
|
||||||
|
ClosureCallback,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
use pyo3::types::PyDict;
|
||||||
|
use pyo3::{intern, types::PyModule, PyObject, PyResult, Python, ToPyObject};
|
||||||
|
use std::path;
|
||||||
|
|
||||||
|
/// A device that can send/receive HCI frames.
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Device(PyObject);
|
||||||
|
|
||||||
|
impl Device {
|
||||||
|
/// Create a Device per the provided file configured to communicate with a controller through an HCI source/sink
|
||||||
|
pub fn from_config_file_with_hci(
|
||||||
|
device_config: &path::Path,
|
||||||
|
source: Source,
|
||||||
|
sink: Sink,
|
||||||
|
) -> PyResult<Self> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
PyModule::import(py, intern!(py, "bumble.device"))?
|
||||||
|
.getattr(intern!(py, "Device"))?
|
||||||
|
.call_method1(
|
||||||
|
intern!(py, "from_config_file_with_hci"),
|
||||||
|
(device_config, source.0, sink.0),
|
||||||
|
)
|
||||||
|
.map(|any| Self(any.into()))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a Device configured to communicate with a controller through an HCI source/sink
|
||||||
|
pub fn with_hci(name: &str, address: &str, source: Source, sink: Sink) -> PyResult<Self> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
PyModule::import(py, intern!(py, "bumble.device"))?
|
||||||
|
.getattr(intern!(py, "Device"))?
|
||||||
|
.call_method1(intern!(py, "with_hci"), (name, address, source.0, sink.0))
|
||||||
|
.map(|any| Self(any.into()))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Turn the device on
|
||||||
|
pub async fn power_on(&self) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method0(py, intern!(py, "power_on"))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Connect to a peer
|
||||||
|
pub async fn connect(&self, peer_addr: &str) -> PyResult<Connection> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method1(py, intern!(py, "connect"), (peer_addr,))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(Connection)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Start scanning
|
||||||
|
pub async fn start_scanning(&self, filter_duplicates: bool) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
let kwargs = PyDict::new(py);
|
||||||
|
kwargs.set_item("filter_duplicates", filter_duplicates)?;
|
||||||
|
self.0
|
||||||
|
.call_method(py, intern!(py, "start_scanning"), (), Some(kwargs))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Register a callback to be called for each advertisement
|
||||||
|
pub fn on_advertisement(
|
||||||
|
&mut self,
|
||||||
|
callback: impl Fn(Python, Advertisement) -> PyResult<()> + Send + 'static,
|
||||||
|
) -> PyResult<()> {
|
||||||
|
let boxed = ClosureCallback::new(move |py, args, _kwargs| {
|
||||||
|
callback(py, Advertisement(args.get_item(0)?.into()))
|
||||||
|
});
|
||||||
|
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method1(py, intern!(py, "add_listener"), ("advertisement", boxed))
|
||||||
|
})
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set the advertisement data to be used when [Device::start_advertising] is called.
|
||||||
|
pub fn set_advertising_data(&mut self, adv_data: AdvertisementDataBuilder) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0.setattr(
|
||||||
|
py,
|
||||||
|
intern!(py, "advertising_data"),
|
||||||
|
adv_data.into_bytes().as_slice(),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Start advertising the data set with [Device.set_advertisement].
|
||||||
|
pub async fn start_advertising(&mut self, auto_restart: bool) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
let kwargs = PyDict::new(py);
|
||||||
|
kwargs.set_item("auto_restart", auto_restart)?;
|
||||||
|
|
||||||
|
self.0
|
||||||
|
.call_method(py, intern!(py, "start_advertising"), (), Some(kwargs))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stop advertising.
|
||||||
|
pub async fn stop_advertising(&mut self) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method0(py, intern!(py, "stop_advertising"))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A connection to a remote device.
|
||||||
|
pub struct Connection(PyObject);
|
||||||
|
|
||||||
|
/// The other end of a connection
|
||||||
|
pub struct Peer(PyObject);
|
||||||
|
|
||||||
|
impl Peer {
|
||||||
|
/// Wrap a [Connection] in a Peer
|
||||||
|
pub fn new(conn: Connection) -> PyResult<Self> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
PyModule::import(py, intern!(py, "bumble.device"))?
|
||||||
|
.getattr(intern!(py, "Peer"))?
|
||||||
|
.call1((conn.0,))
|
||||||
|
.map(|obj| Self(obj.into()))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Populates the peer's cache of services.
|
||||||
|
///
|
||||||
|
/// Returns the discovered services.
|
||||||
|
pub async fn discover_services(&mut self) -> PyResult<Vec<ServiceProxy>> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method0(py, intern!(py, "discover_services"))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.and_then(|list| {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
list.as_ref(py)
|
||||||
|
.iter()?
|
||||||
|
.map(|r| r.map(|h| ServiceProxy(h.to_object(py))))
|
||||||
|
.collect()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a snapshot of the Services currently in the peer's cache
|
||||||
|
pub fn services(&self) -> PyResult<Vec<ServiceProxy>> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.getattr(py, intern!(py, "services"))?
|
||||||
|
.as_ref(py)
|
||||||
|
.iter()?
|
||||||
|
.map(|r| r.map(|h| ServiceProxy(h.to_object(py))))
|
||||||
|
.collect()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Build a [ProfileServiceProxy] for the specified type.
|
||||||
|
/// [Peer::discover_services] or some other means of populating the Peer's service cache must be
|
||||||
|
/// called first, or the required service won't be found.
|
||||||
|
pub fn create_service_proxy<P: ProfileServiceProxy>(&self) -> PyResult<Option<P>> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
let module = py.import(P::PROXY_CLASS_MODULE)?;
|
||||||
|
let class = module.getattr(P::PROXY_CLASS_NAME)?;
|
||||||
|
self.0
|
||||||
|
.call_method1(py, intern!(py, "create_service_proxy"), (class,))
|
||||||
|
.map(|obj| {
|
||||||
|
if obj.is_none(py) {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(P::wrap(obj))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A BLE advertisement
|
||||||
|
pub struct Advertisement(PyObject);
|
||||||
|
|
||||||
|
impl Advertisement {
|
||||||
|
/// Address that sent the advertisement
|
||||||
|
pub fn address(&self) -> PyResult<Address> {
|
||||||
|
Python::with_gil(|py| self.0.getattr(py, intern!(py, "address")).map(Address))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the advertisement is connectable
|
||||||
|
pub fn is_connectable(&self) -> PyResult<bool> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.getattr(py, intern!(py, "is_connectable"))?
|
||||||
|
.extract::<bool>(py)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// RSSI of the advertisement
|
||||||
|
pub fn rssi(&self) -> PyResult<i8> {
|
||||||
|
Python::with_gil(|py| self.0.getattr(py, intern!(py, "rssi"))?.extract::<i8>(py))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Data in the advertisement
|
||||||
|
pub fn data(&self) -> PyResult<AdvertisingData> {
|
||||||
|
Python::with_gil(|py| self.0.getattr(py, intern!(py, "data")).map(AdvertisingData))
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,79 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! GATT client support
|
||||||
|
|
||||||
|
use crate::wrapper::ClosureCallback;
|
||||||
|
use pyo3::types::PyTuple;
|
||||||
|
use pyo3::{intern, PyObject, PyResult, Python};
|
||||||
|
|
||||||
|
/// A GATT service on a remote device
|
||||||
|
pub struct ServiceProxy(pub(crate) PyObject);
|
||||||
|
|
||||||
|
impl ServiceProxy {
|
||||||
|
/// Discover the characteristics in this service.
|
||||||
|
///
|
||||||
|
/// Populates an internal cache of characteristics in this service.
|
||||||
|
pub async fn discover_characteristics(&mut self) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method0(py, intern!(py, "discover_characteristics"))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A GATT characteristic on a remote device
|
||||||
|
pub struct CharacteristicProxy(pub(crate) PyObject);
|
||||||
|
|
||||||
|
impl CharacteristicProxy {
|
||||||
|
/// Subscribe to changes to the characteristic, executing `callback` for each new value
|
||||||
|
pub async fn subscribe(
|
||||||
|
&mut self,
|
||||||
|
callback: impl Fn(Python, &PyTuple) -> PyResult<()> + Send + 'static,
|
||||||
|
) -> PyResult<()> {
|
||||||
|
let boxed = ClosureCallback::new(move |py, args, _kwargs| callback(py, args));
|
||||||
|
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method1(py, intern!(py, "subscribe"), (boxed,))
|
||||||
|
.and_then(|obj| pyo3_asyncio::tokio::into_future(obj.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Read the current value of the characteristic
|
||||||
|
pub async fn read_value(&self) -> PyResult<PyObject> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method0(py, intern!(py, "read_value"))
|
||||||
|
.and_then(|obj| pyo3_asyncio::tokio::into_future(obj.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Equivalent to the Python `ProfileServiceProxy`.
|
||||||
|
pub trait ProfileServiceProxy {
|
||||||
|
/// The module containing the proxy class
|
||||||
|
const PROXY_CLASS_MODULE: &'static str;
|
||||||
|
/// The module class name
|
||||||
|
const PROXY_CLASS_NAME: &'static str;
|
||||||
|
|
||||||
|
/// Wrap a PyObject in the Rust wrapper type
|
||||||
|
fn wrap(obj: PyObject) -> Self;
|
||||||
|
}
|
||||||
@@ -0,0 +1,112 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! HCI
|
||||||
|
|
||||||
|
use itertools::Itertools as _;
|
||||||
|
use pyo3::{exceptions::PyException, intern, types::PyModule, PyErr, PyObject, PyResult, Python};
|
||||||
|
|
||||||
|
/// A Bluetooth address
|
||||||
|
pub struct Address(pub(crate) PyObject);
|
||||||
|
|
||||||
|
impl Address {
|
||||||
|
/// The type of address
|
||||||
|
pub fn address_type(&self) -> PyResult<AddressType> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
let addr_type = self
|
||||||
|
.0
|
||||||
|
.getattr(py, intern!(py, "address_type"))?
|
||||||
|
.extract::<u32>(py)?;
|
||||||
|
|
||||||
|
let module = PyModule::import(py, intern!(py, "bumble.hci"))?;
|
||||||
|
let klass = module.getattr(intern!(py, "Address"))?;
|
||||||
|
|
||||||
|
if addr_type
|
||||||
|
== klass
|
||||||
|
.getattr(intern!(py, "PUBLIC_DEVICE_ADDRESS"))?
|
||||||
|
.extract::<u32>()?
|
||||||
|
{
|
||||||
|
Ok(AddressType::PublicDevice)
|
||||||
|
} else if addr_type
|
||||||
|
== klass
|
||||||
|
.getattr(intern!(py, "RANDOM_DEVICE_ADDRESS"))?
|
||||||
|
.extract::<u32>()?
|
||||||
|
{
|
||||||
|
Ok(AddressType::RandomDevice)
|
||||||
|
} else if addr_type
|
||||||
|
== klass
|
||||||
|
.getattr(intern!(py, "PUBLIC_IDENTITY_ADDRESS"))?
|
||||||
|
.extract::<u32>()?
|
||||||
|
{
|
||||||
|
Ok(AddressType::PublicIdentity)
|
||||||
|
} else if addr_type
|
||||||
|
== klass
|
||||||
|
.getattr(intern!(py, "RANDOM_IDENTITY_ADDRESS"))?
|
||||||
|
.extract::<u32>()?
|
||||||
|
{
|
||||||
|
Ok(AddressType::RandomIdentity)
|
||||||
|
} else {
|
||||||
|
Err(PyErr::new::<PyException, _>("Invalid address type"))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// True if the address is static
|
||||||
|
pub fn is_static(&self) -> PyResult<bool> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.getattr(py, intern!(py, "is_static"))?
|
||||||
|
.extract::<bool>(py)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// True if the address is resolvable
|
||||||
|
pub fn is_resolvable(&self) -> PyResult<bool> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.getattr(py, intern!(py, "is_resolvable"))?
|
||||||
|
.extract::<bool>(py)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Address bytes in _little-endian_ format
|
||||||
|
pub fn as_le_bytes(&self) -> PyResult<Vec<u8>> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method0(py, intern!(py, "to_bytes"))?
|
||||||
|
.extract::<Vec<u8>>(py)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Address bytes as big-endian colon-separated hex
|
||||||
|
pub fn as_hex(&self) -> PyResult<String> {
|
||||||
|
self.as_le_bytes().map(|bytes| {
|
||||||
|
bytes
|
||||||
|
.into_iter()
|
||||||
|
.rev()
|
||||||
|
.map(|byte| hex::encode_upper([byte]))
|
||||||
|
.join(":")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// BT address types
|
||||||
|
#[allow(missing_docs)]
|
||||||
|
#[derive(PartialEq, Eq, Debug)]
|
||||||
|
pub enum AddressType {
|
||||||
|
PublicDevice,
|
||||||
|
RandomDevice,
|
||||||
|
PublicIdentity,
|
||||||
|
RandomIdentity,
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
//! Bumble & Python logging
|
||||||
|
|
||||||
|
use pyo3::types::PyDict;
|
||||||
|
use pyo3::{intern, types::PyModule, PyResult, Python};
|
||||||
|
use std::env;
|
||||||
|
|
||||||
|
/// Returns the uppercased contents of the `BUMBLE_LOGLEVEL` env var, or `default` if it is not present or not UTF-8.
|
||||||
|
///
|
||||||
|
/// The result could be passed to [py_logging_basic_config] to configure Python's logging
|
||||||
|
/// accordingly.
|
||||||
|
pub fn bumble_env_logging_level(default: impl Into<String>) -> String {
|
||||||
|
env::var("BUMBLE_LOGLEVEL")
|
||||||
|
.unwrap_or_else(|_| default.into())
|
||||||
|
.to_ascii_uppercase()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Call `logging.basicConfig` with the provided logging level
|
||||||
|
pub fn py_logging_basic_config(log_level: impl Into<String>) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
let kwargs = PyDict::new(py);
|
||||||
|
kwargs.set_item("level", log_level.into())?;
|
||||||
|
|
||||||
|
PyModule::import(py, intern!(py, "logging"))?
|
||||||
|
.call_method(intern!(py, "basicConfig"), (), Some(kwargs))
|
||||||
|
.map(|_| ())
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -0,0 +1,92 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! Types that wrap the Python API.
|
||||||
|
//!
|
||||||
|
//! Because mutability, aliasing, etc is all hidden behind Python, the normal Rust rules about
|
||||||
|
//! only one mutable reference to one piece of memory, etc, may not hold since using `&mut self`
|
||||||
|
//! instead of `&self` is only guided by inspection of the Python source, not the compiler.
|
||||||
|
//!
|
||||||
|
//! The modules are generally structured to mirror the Python equivalents.
|
||||||
|
|
||||||
|
// Re-exported to make it easy for users to depend on the same `PyObject`, etc
|
||||||
|
pub use pyo3;
|
||||||
|
use pyo3::{
|
||||||
|
prelude::*,
|
||||||
|
types::{PyDict, PyTuple},
|
||||||
|
};
|
||||||
|
pub use pyo3_asyncio;
|
||||||
|
|
||||||
|
pub mod assigned_numbers;
|
||||||
|
pub mod core;
|
||||||
|
pub mod device;
|
||||||
|
pub mod gatt_client;
|
||||||
|
pub mod hci;
|
||||||
|
pub mod logging;
|
||||||
|
pub mod profile;
|
||||||
|
pub mod transport;
|
||||||
|
|
||||||
|
/// Convenience extensions to [PyObject]
|
||||||
|
pub trait PyObjectExt {
|
||||||
|
/// Get a GIL-bound reference
|
||||||
|
fn gil_ref<'py>(&'py self, py: Python<'py>) -> &'py PyAny;
|
||||||
|
|
||||||
|
/// Extract any [FromPyObject] implementation from this value
|
||||||
|
fn extract_with_gil<T>(&self) -> PyResult<T>
|
||||||
|
where
|
||||||
|
T: for<'a> FromPyObject<'a>,
|
||||||
|
{
|
||||||
|
Python::with_gil(|py| self.gil_ref(py).extract::<T>())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PyObjectExt for PyObject {
|
||||||
|
fn gil_ref<'py>(&'py self, py: Python<'py>) -> &'py PyAny {
|
||||||
|
self.as_ref(py)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Wrapper to make Rust closures ([Fn] implementations) callable from Python.
|
||||||
|
///
|
||||||
|
/// The Python callable form returns a Python `None`.
|
||||||
|
#[pyclass(name = "SubscribeCallback")]
|
||||||
|
pub(crate) struct ClosureCallback {
|
||||||
|
// can't use generics in a pyclass, so have to box
|
||||||
|
#[allow(clippy::type_complexity)]
|
||||||
|
callback: Box<dyn Fn(Python, &PyTuple, Option<&PyDict>) -> PyResult<()> + Send + 'static>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClosureCallback {
|
||||||
|
/// Create a new callback around the provided closure
|
||||||
|
pub fn new(
|
||||||
|
callback: impl Fn(Python, &PyTuple, Option<&PyDict>) -> PyResult<()> + Send + 'static,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
callback: Box::new(callback),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[pymethods]
|
||||||
|
impl ClosureCallback {
|
||||||
|
#[pyo3(signature = (*args, **kwargs))]
|
||||||
|
fn __call__(
|
||||||
|
&self,
|
||||||
|
py: Python<'_>,
|
||||||
|
args: &PyTuple,
|
||||||
|
kwargs: Option<&PyDict>,
|
||||||
|
) -> PyResult<Py<PyAny>> {
|
||||||
|
(self.callback)(py, args, kwargs).map(|_| py.None())
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! GATT profiles
|
||||||
|
|
||||||
|
use crate::wrapper::gatt_client::{CharacteristicProxy, ProfileServiceProxy};
|
||||||
|
use pyo3::{intern, PyObject, PyResult, Python};
|
||||||
|
|
||||||
|
/// Exposes the battery GATT service
|
||||||
|
pub struct BatteryServiceProxy(PyObject);
|
||||||
|
|
||||||
|
impl BatteryServiceProxy {
|
||||||
|
/// Get the battery level, if available
|
||||||
|
pub fn battery_level(&self) -> PyResult<Option<CharacteristicProxy>> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.getattr(py, intern!(py, "battery_level"))
|
||||||
|
.map(|level| {
|
||||||
|
if level.is_none(py) {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(CharacteristicProxy(level))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ProfileServiceProxy for BatteryServiceProxy {
|
||||||
|
const PROXY_CLASS_MODULE: &'static str = "bumble.profiles.battery_service";
|
||||||
|
const PROXY_CLASS_NAME: &'static str = "BatteryServiceProxy";
|
||||||
|
|
||||||
|
fn wrap(obj: PyObject) -> Self {
|
||||||
|
Self(obj)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
// Copyright 2023 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed 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.
|
||||||
|
|
||||||
|
//! HCI packet transport
|
||||||
|
|
||||||
|
use pyo3::{intern, types::PyModule, PyObject, PyResult, Python};
|
||||||
|
|
||||||
|
/// A source/sink pair for HCI packet I/O.
|
||||||
|
///
|
||||||
|
/// See <https://google.github.io/bumble/transports/index.html>.
|
||||||
|
pub struct Transport(PyObject);
|
||||||
|
|
||||||
|
impl Transport {
|
||||||
|
/// Open a new Transport for the provided spec, e.g. `"usb:0"` or `"android-netsim"`.
|
||||||
|
pub async fn open(transport_spec: impl Into<String>) -> PyResult<Self> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
PyModule::import(py, intern!(py, "bumble.transport"))?
|
||||||
|
.call_method1(intern!(py, "open_transport"), (transport_spec.into(),))
|
||||||
|
.and_then(pyo3_asyncio::tokio::into_future)
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(Self)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Close the transport.
|
||||||
|
pub async fn close(&mut self) -> PyResult<()> {
|
||||||
|
Python::with_gil(|py| {
|
||||||
|
self.0
|
||||||
|
.call_method0(py, intern!(py, "close"))
|
||||||
|
.and_then(|coroutine| pyo3_asyncio::tokio::into_future(coroutine.as_ref(py)))
|
||||||
|
})?
|
||||||
|
.await
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the source half of the transport.
|
||||||
|
pub fn source(&self) -> PyResult<Source> {
|
||||||
|
Python::with_gil(|py| self.0.getattr(py, intern!(py, "source"))).map(Source)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the sink half of the transport.
|
||||||
|
pub fn sink(&self) -> PyResult<Sink> {
|
||||||
|
Python::with_gil(|py| self.0.getattr(py, intern!(py, "sink"))).map(Sink)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for Transport {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
// can't await in a Drop impl, but we can at least spawn a task to do it
|
||||||
|
let obj = self.0.clone();
|
||||||
|
tokio::spawn(async move { Self(obj).close().await });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The source side of a [Transport].
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Source(pub(crate) PyObject);
|
||||||
|
|
||||||
|
/// The sink side of a [Transport].
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Sink(pub(crate) PyObject);
|
||||||
@@ -46,6 +46,7 @@ from bumble.hci import (
|
|||||||
HCI_LE_Set_Advertising_Parameters_Command,
|
HCI_LE_Set_Advertising_Parameters_Command,
|
||||||
HCI_LE_Set_Default_PHY_Command,
|
HCI_LE_Set_Default_PHY_Command,
|
||||||
HCI_LE_Set_Event_Mask_Command,
|
HCI_LE_Set_Event_Mask_Command,
|
||||||
|
HCI_LE_Set_Extended_Advertising_Enable_Command,
|
||||||
HCI_LE_Set_Extended_Scan_Parameters_Command,
|
HCI_LE_Set_Extended_Scan_Parameters_Command,
|
||||||
HCI_LE_Set_Random_Address_Command,
|
HCI_LE_Set_Random_Address_Command,
|
||||||
HCI_LE_Set_Scan_Enable_Command,
|
HCI_LE_Set_Scan_Enable_Command,
|
||||||
@@ -422,6 +423,25 @@ def test_HCI_LE_Set_Extended_Scan_Parameters_Command():
|
|||||||
basic_check(command)
|
basic_check(command)
|
||||||
|
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------------
|
||||||
|
def test_HCI_LE_Set_Extended_Advertising_Enable_Command():
|
||||||
|
command = HCI_Packet.from_bytes(
|
||||||
|
bytes.fromhex('0139200e010301050008020600090307000a')
|
||||||
|
)
|
||||||
|
assert command.enable == 1
|
||||||
|
assert command.advertising_handles == [1, 2, 3]
|
||||||
|
assert command.durations == [5, 6, 7]
|
||||||
|
assert command.max_extended_advertising_events == [8, 9, 10]
|
||||||
|
|
||||||
|
command = HCI_LE_Set_Extended_Advertising_Enable_Command(
|
||||||
|
enable=1,
|
||||||
|
advertising_handles=[1, 2, 3],
|
||||||
|
durations=[5, 6, 7],
|
||||||
|
max_extended_advertising_events=[8, 9, 10],
|
||||||
|
)
|
||||||
|
basic_check(command)
|
||||||
|
|
||||||
|
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
def test_address():
|
def test_address():
|
||||||
a = Address('C4:F2:17:1A:1D:BB')
|
a = Address('C4:F2:17:1A:1D:BB')
|
||||||
@@ -478,6 +498,7 @@ def run_test_commands():
|
|||||||
test_HCI_LE_Read_Remote_Features_Command()
|
test_HCI_LE_Read_Remote_Features_Command()
|
||||||
test_HCI_LE_Set_Default_PHY_Command()
|
test_HCI_LE_Set_Default_PHY_Command()
|
||||||
test_HCI_LE_Set_Extended_Scan_Parameters_Command()
|
test_HCI_LE_Set_Extended_Scan_Parameters_Command()
|
||||||
|
test_HCI_LE_Set_Extended_Advertising_Enable_Command()
|
||||||
|
|
||||||
|
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
|
|||||||
Reference in New Issue
Block a user