# Copyright 2021-2024 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 # # https://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. # ----------------------------------------------------------------------------- # Imports # ----------------------------------------------------------------------------- import asyncio import logging import sys import os import secrets import websockets import json from bumble.core import AdvertisingData from bumble.device import Device, AdvertisingParameters, AdvertisingEventProperties from bumble.hci import ( CodecID, CodingFormat, OwnAddressType, ) from bumble.profiles.ascs import AudioStreamControlService from bumble.profiles.bap import ( UnicastServerAdvertisingData, CodecSpecificCapabilities, ContextType, AudioLocation, SupportedSamplingFrequency, SupportedFrameDuration, ) from bumble.profiles.pacs import PacRecord, PublishedAudioCapabilitiesService from bumble.profiles.cap import CommonAudioServiceService from bumble.profiles.csip import CoordinatedSetIdentificationService, SirkType from bumble.profiles.vcp import VolumeControlService from bumble.transport import open_transport_or_link from typing import Optional def dumps_volume_state(volume_setting: int, muted: int, change_counter: int) -> str: return json.dumps( { 'volume_setting': volume_setting, 'muted': muted, 'change_counter': change_counter, } ) # ----------------------------------------------------------------------------- async def main() -> None: if len(sys.argv) < 3: print('Usage: run_vcp_renderer.py ' '') return print('<<< connecting to HCI...') async with await open_transport_or_link(sys.argv[2]) as hci_transport: print('<<< connected') device = Device.from_config_file_with_hci( sys.argv[1], hci_transport.source, hci_transport.sink ) await device.power_on() # Add "placeholder" services to enable Android LEA features. csis = CoordinatedSetIdentificationService( set_identity_resolving_key=secrets.token_bytes(16), set_identity_resolving_key_type=SirkType.PLAINTEXT, ) device.add_service(CommonAudioServiceService(csis)) device.add_service( PublishedAudioCapabilitiesService( supported_source_context=ContextType.PROHIBITED, available_source_context=ContextType.PROHIBITED, supported_sink_context=ContextType.MEDIA, available_sink_context=ContextType.MEDIA, sink_audio_locations=( AudioLocation.FRONT_LEFT | AudioLocation.FRONT_RIGHT ), sink_pac=[ # Codec Capability Setting 48_4 PacRecord( coding_format=CodingFormat(CodecID.LC3), codec_specific_capabilities=CodecSpecificCapabilities( supported_sampling_frequencies=( SupportedSamplingFrequency.FREQ_48000 ), supported_frame_durations=( SupportedFrameDuration.DURATION_10000_US_SUPPORTED ), supported_audio_channel_count=[1], min_octets_per_codec_frame=120, max_octets_per_codec_frame=120, supported_max_codec_frames_per_sdu=1, ), ), ], ) ) device.add_service(AudioStreamControlService(device, sink_ase_id=[1, 2])) vcs = VolumeControlService() device.add_service(vcs) ws: Optional[websockets.WebSocketServerProtocol] = None def on_volume_state(volume_setting: int, muted: int, change_counter: int): if ws: asyncio.create_task( ws.send(dumps_volume_state(volume_setting, muted, change_counter)) ) vcs.on('volume_state', on_volume_state) advertising_data = ( bytes( AdvertisingData( [ ( AdvertisingData.COMPLETE_LOCAL_NAME, bytes('Bumble LE Audio', 'utf-8'), ), ( AdvertisingData.FLAGS, bytes( [ AdvertisingData.LE_GENERAL_DISCOVERABLE_MODE_FLAG | AdvertisingData.BR_EDR_HOST_FLAG | AdvertisingData.BR_EDR_CONTROLLER_FLAG ] ), ), ( AdvertisingData.INCOMPLETE_LIST_OF_16_BIT_SERVICE_CLASS_UUIDS, bytes(PublishedAudioCapabilitiesService.UUID), ), ] ) ) + csis.get_advertising_data() + bytes(UnicastServerAdvertisingData()) ) await device.create_advertising_set( advertising_parameters=AdvertisingParameters( advertising_event_properties=AdvertisingEventProperties(), own_address_type=OwnAddressType.PUBLIC, ), advertising_data=advertising_data, ) async def serve(websocket: websockets.WebSocketServerProtocol, _path): nonlocal ws await websocket.send( dumps_volume_state(vcs.volume_setting, vcs.muted, vcs.change_counter) ) ws = websocket async for message in websocket: volume_state = json.loads(message) vcs.volume_state_bytes = bytes( [ volume_state['volume_setting'], volume_state['muted'], volume_state['change_counter'], ] ) await device.notify_subscribers( vcs.volume_state, vcs.volume_state_bytes ) ws = None await websockets.serve(serve, 'localhost', 8989) await hci_transport.source.terminated # ----------------------------------------------------------------------------- logging.basicConfig(level=os.environ.get('BUMBLE_LOGLEVEL', 'DEBUG').upper()) asyncio.run(main())