use std::{
any::{Any, TypeId},
collections::HashMap,
fmt::Write,
future::Future,
pin::Pin,
};
use async_trait::async_trait;
use zbus::MessageFlags;
use zbus_names::{InterfaceName, MemberName};
use zvariant::{DynamicType, OwnedValue, Value};
use crate::{fdo, Connection, Message, ObjectServer, Result, SignalContext};
use tracing::trace;
pub enum DispatchResult<'a> {
NotFound,
RequiresMut,
Async(Pin<Box<dyn Future<Output = Result<()>> + Send + 'a>>),
}
impl<'a> DispatchResult<'a> {
pub fn new_async<F, T, E>(conn: &'a Connection, msg: &'a Message, f: F) -> Self
where
F: Future<Output = ::std::result::Result<T, E>> + Send + 'a,
T: serde::Serialize + DynamicType + Send + Sync,
E: zbus::DBusError + Send,
{
DispatchResult::Async(Box::pin(async move {
let hdr = msg.header()?;
let ret = f.await;
if !hdr
.primary()
.flags()
.contains(MessageFlags::NoReplyExpected)
{
match ret {
Ok(r) => conn.reply(msg, &r).await,
Err(e) => conn.reply_dbus_error(&hdr, e).await,
}
.map(|_seq| ())
} else {
trace!("No reply expected for {:?} by the caller.", msg);
Ok(())
}
}))
}
}
#[async_trait]
pub trait Interface: Any + Send + Sync {
fn name() -> InterfaceName<'static>
where
Self: Sized;
async fn get(&self, property_name: &str) -> Option<fdo::Result<OwnedValue>>;
async fn get_all(&self) -> HashMap<String, OwnedValue>;
fn set<'call>(
&'call self,
property_name: &'call str,
value: &'call Value<'_>,
ctxt: &'call SignalContext<'_>,
) -> DispatchResult<'call> {
let _ = (property_name, value, ctxt);
DispatchResult::RequiresMut
}
async fn set_mut(
&mut self,
property_name: &str,
value: &Value<'_>,
ctxt: &SignalContext<'_>,
) -> Option<fdo::Result<()>>;
fn call<'call>(
&'call self,
server: &'call ObjectServer,
connection: &'call Connection,
msg: &'call Message,
name: MemberName<'call>,
) -> DispatchResult<'call>;
fn call_mut<'call>(
&'call mut self,
server: &'call ObjectServer,
connection: &'call Connection,
msg: &'call Message,
name: MemberName<'call>,
) -> DispatchResult<'call>;
fn introspect_to_writer(&self, writer: &mut dyn Write, level: usize);
}
impl dyn Interface {
pub(crate) fn downcast_ref<T: Any>(&self) -> Option<&T> {
if <dyn Interface as Any>::type_id(self) == TypeId::of::<T>() {
Some(unsafe { &*(self as *const dyn Interface as *const T) })
} else {
None
}
}
pub(crate) fn downcast_mut<T: Any>(&mut self) -> Option<&mut T> {
if <dyn Interface as Any>::type_id(self) == TypeId::of::<T>() {
Some(unsafe { &mut *(self as *mut dyn Interface as *mut T) })
} else {
None
}
}
}