// Copyright 2024 System76 // SPDX-License-Identifier: MPL-2.0 use crate::{Active, Event}; use futures::join; use std::{ collections::HashMap, hash::{Hash, Hasher}, time::Duration, }; use zbus::zvariant::OwnedObjectPath; const DEFAILT_DEVICE_ICON: &str = "bluetooth-symbolic"; // Copied from https://github.com/bluez/bluez/blob/39467578207889fd015775cbe81a3db9dd26abea/src/dbus-common.c#L53 fn device_type_to_icon(device_type: &str) -> &'static str { match device_type { "computer" => "laptop-symbolic", "phone" => "smartphone-symbolic", "network-wireless" => "network-wireless-symbolic", "audio-headset" => "audio-headset-symbolic", "audio-headphones" => "audio-headphones-symbolic", "camera-video" => "camera-video-symbolic", "audio-card" => "audio-card-symbolic", "input-gaming" => "input-gaming-symbolic", "input-keyboard" => "input-keyboard-symbolic", "input-tablet" => "input-tablet-symbolic", "input-mouse" => "input-mouse-symbolic", "printer" => "printer-network-symbolic", "camera-photo" => "camera-photo-symbolic", _ => DEFAILT_DEVICE_ICON, } } #[derive(Default, Debug, Clone)] pub struct Device { alias: Option, pub address: String, pub adapter: OwnedObjectPath, pub enabled: Active, pub paired: bool, pub icon: &'static str, pub battery: Option, } impl Device { pub async fn from_device(proxy: &bluez_zbus::BluetoothDevice<'_>) -> zbus::Result { let (address, adapter, alias) = join!( proxy.device.address(), proxy.device.adapter(), proxy.device.name() ); let address = address?; if address.is_empty() { return Err(zbus::Error::Failure("Device has no MAC address".to_owned())); } let adapter = adapter?; if adapter.is_empty() { return Err(zbus::Error::Failure("Device has no adapter".to_owned())); } let alias = alias.ok(); let device_type: String = proxy.icon().await; let paired = proxy.device.paired().await.unwrap_or(false); let enabled = if proxy.device.connected().await.unwrap_or(false) { Active::Enabled } else { Active::Disabled }; let battery = match &proxy.battery { Some(battery) => match battery.percentage().await { Ok(percentage) => Some(percentage.to_string()), Err(why) => { eprintln!("couldn't fetch battery percentage: {why}"); None } }, None => None, }; let icon = device_type_to_icon(device_type.as_str()); Ok(Self { alias, address, adapter, enabled, paired, icon, battery, }) } #[must_use] pub fn is_connected(&self) -> bool { self.enabled == Active::Enabled } /// Update the state of the device without overriding intermediary states. /// /// # Panics /// /// Panics if the device used for update doesn't have the same MAC address pub fn update(&mut self, updates: Vec) { for udpate in updates { match udpate { DeviceUpdate::Alias(alias) => self.alias = alias, DeviceUpdate::Enabled(enabled) => { self.enabled = match (self.enabled, enabled) { (Active::Enabling, Active::Enabled) => Active::Enabled, (Active::Disabling, Active::Disabled) => Active::Disabled, (Active::Enabled | Active::Disabled, status) => status, (status, _) => status, } } DeviceUpdate::Paired(paired) => { self.paired = paired; } DeviceUpdate::Icon(icon) => self.icon = icon, DeviceUpdate::Battery(battery) => self.battery = battery, } } if self.enabled == Active::Disabled { self.battery = None; } } #[must_use] pub fn has_alias(&self) -> bool { self.alias.is_some() } #[must_use] pub fn is_known_device_type(&self) -> bool { self.icon != DEFAILT_DEVICE_ICON } #[must_use] pub fn alias_or_addr(&self) -> &str { self.alias.as_ref().unwrap_or(&self.address) } } impl Hash for Device { fn hash(&self, state: &mut H) { self.address.hash(state); } } impl PartialEq for Device { fn eq(&self, other: &Self) -> bool { self.address == other.address } } impl Eq for Device {} #[derive(Debug, Clone)] pub enum DeviceUpdate { Alias(Option), Enabled(Active), Paired(bool), Icon(&'static str), Battery(Option), } impl DeviceUpdate { pub fn from_update(update: HashMap<&'_ str, zbus::zvariant::Value<'_>>) -> Vec { update .into_iter() .filter_map(|(key, value)| { match (key, value) { ("Alias", zbus::zvariant::Value::Str(value)) => { Some(DeviceUpdate::Alias(Some(value.into()))) } ("Connected", zbus::zvariant::Value::Bool(value)) => { Some(DeviceUpdate::Enabled(if value { Active::Enabled } else { Active::Disabled })) } ("Paired", zbus::zvariant::Value::Bool(value)) => { Some(DeviceUpdate::Paired(value)) } ("Icon", zbus::zvariant::Value::Str(value)) => { Some(DeviceUpdate::Icon(device_type_to_icon(&value))) } ("Percentage", zbus::zvariant::Value::U8(percentage)) => { Some(DeviceUpdate::Battery(Some(percentage.to_string()))) } // Battery (message, value) => { tracing::debug!(message, ?value, "device update"); None } } }) .collect() } } pub async fn disconnect_device( connection: zbus::Connection, device_path: OwnedObjectPath, ) -> Event { let proxy = match bluez_zbus::get_device(&connection, device_path.clone()).await { Err(why) => { tracing::error!("Unable to get the device: {why}"); return Event::DeviceFailed(device_path); } Ok(proxy) => proxy, }; for attempt in 1..5 { let result = async { if !proxy.device.connected().await? { return Ok(()); } proxy.device.disconnect().await } .await; if let Err(why) = result { tracing::warn!("Unable to disconnect to device: {why}"); tokio::time::sleep(Duration::from_millis(1000 * attempt)).await; } else { return Event::Ok; } } Event::DeviceFailed(device_path) } pub async fn connect_device(connection: zbus::Connection, device_path: OwnedObjectPath) -> Event { let proxy = match bluez_zbus::get_device(&connection, device_path.clone()).await { Err(why) => { tracing::error!("Unable to get the device: {why}"); return Event::DeviceFailed(device_path); } Ok(proxy) => proxy, }; for attempt in 1..5 { let result = async { if proxy.device.connected().await? { Ok(()) } else if !proxy.device.paired().await.unwrap_or(false) { proxy.device.pair().await?; proxy.device.connect().await } else { proxy.device.connect().await } } .await; if let Err(why) = result { tracing::warn!("Unable to connect to device: {why}"); tokio::time::sleep(Duration::from_millis(1000 * attempt)).await; } else { return Event::Ok; } } Event::DeviceFailed(device_path) } pub async fn forget_device(connection: zbus::Connection, device_path: OwnedObjectPath) -> Event { let mut result: zbus::Result<()> = Ok(()); let proxy = match bluez_zbus::get_device(&connection, device_path.clone()).await { Err(why) => { tracing::error!("Unable to get the device: {why}"); return Event::DeviceFailed(device_path); } Ok(proxy) => proxy, }; let adapter_path = match proxy.device.adapter().await { Err(why) => { tracing::error!("Unable to get the adapter: {why}"); return Event::DeviceFailed(device_path); } Ok(adapter_path) => adapter_path, }; let adapter = match bluez_zbus::get_adapter(&connection, adapter_path).await { Err(why) => { tracing::error!("Unable to get the adapter: {why}"); return Event::DeviceFailed(device_path); } Ok(adapter) => adapter, }; for attempt in 1..5 { result = async { if proxy.device.connected().await? { proxy.device.disconnect().await?; } adapter.remove_device(&proxy.path()).await } .await; if let Err(why) = &result { tracing::warn!("Unable to connect to device: {why}"); tokio::time::sleep(Duration::from_millis(1000 * attempt)).await; } else { return Event::Ok; } } if result.is_err() { Event::DeviceFailed(device_path) } else { Event::Ok } } pub async fn get_devices(connection: zbus::Connection, adapter_path: OwnedObjectPath) -> Event { // TODO error handling let result: zbus::Result> = async { futures::future::join_all( bluez_zbus::get_devices(&connection, Some(&adapter_path)) .await? .into_iter() .map( |(path, device)| async move { Ok((path, Device::from_device(&device).await?)) }, ), ) .await .into_iter() .collect::, _>>() } .await; match result { Ok(devices) => Event::SetDevices(devices), Err(why) => { tracing::error!("zbus connection failed. {why}"); Event::DBusError(why) } } } #[cfg(test)] mod tests { use super::*; #[test] fn test_update_device_with_intermediary_state() { let mut device = Device { alias: None, adapter: OwnedObjectPath::try_from("/dev/bluez/hci0").unwrap(), address: "AA:BB:CC:DD:EE:FF".to_owned(), enabled: Active::Disabled, paired: false, icon: "bluetooth-symbolic", battery: None, }; device.update(vec![ DeviceUpdate::Enabled(Active::Enabled), DeviceUpdate::Alias(Some("Foo".to_owned())), ]); assert_eq!(device.enabled, Active::Enabled); assert_eq!(device.alias, Some("Foo".to_owned())); device.enabled = Active::Disabling; device.update(vec![ DeviceUpdate::Enabled(Active::Enabled), DeviceUpdate::Alias(Some("Foo".to_owned())), ]); assert_eq!(device.enabled, Active::Disabling); device.enabled = Active::Enabling; device.update(vec![ DeviceUpdate::Enabled(Active::Enabled), DeviceUpdate::Alias(Some("Foo".to_owned())), ]); assert_eq!(device.enabled, Active::Enabled); } }