Revision control

Copy as Markdown

Other Tools

//! Contains structs and traits closely related to D-Bus messages.
use std::{fmt, ptr};
use super::{ffi, Error, libc, init_dbus};
use crate::strings::{BusName, Path, Interface, Member, ErrorName};
use std::ffi::CStr;
use super::arg::{Append, AppendAll, IterAppend, ReadAll, Get, Iter, Arg, RefArg, TypeMismatchError};
#[derive(Copy, Clone, Debug, Eq, Ord, PartialEq, PartialOrd, Hash)]
/// One of the four different message types.
pub enum MessageType {
/// This is a method call D-Bus message
MethodCall = 1,
/// This is a method return Ok D-Bus message, used when the method call message was successfully processed
MethodReturn = 2,
/// This is a method return with error D-Bus message, used when the method call message could not be handled
Error = 3,
/// This is a signal, usually sent to whoever wants to listen
Signal = 4,
}
impl<'a> TryFrom<&'a str> for MessageType {
type Error = ();
fn try_from(value: &'a str) -> Result<Self, <crate::message::MessageType as TryFrom<&'a str>>::Error> {
match value {
"error" => Ok(MessageType::Error),
"method_call" => Ok(MessageType::MethodCall),
"method_return" => Ok(MessageType::MethodReturn),
"signal" => Ok(MessageType::Signal),
_ => Err(())
}
}
}
mod signalargs;
pub use self::signalargs::SignalArgs;
mod matchrule;
pub use self::matchrule::MatchRule;
use std::convert::TryFrom;
mod parser;
pub use self::parser::Error as MatchRuleParserError;
/// A D-Bus message. A message contains headers - usually destination address, path, interface and member,
/// and a list of arguments.
pub struct Message {
msg: *mut ffi::DBusMessage,
}
unsafe impl Send for Message {}
impl Message {
/// Creates a new method call message.
pub fn new_method_call<'d, 'p, 'i, 'm, D, P, I, M>(destination: D, path: P, iface: I, method: M) -> Result<Message, String>
where D: Into<BusName<'d>>, P: Into<Path<'p>>, I: Into<Interface<'i>>, M: Into<Member<'m>> {
init_dbus();
let (d, p, i, m) = (destination.into(), path.into(), iface.into(), method.into());
let ptr = unsafe {
ffi::dbus_message_new_method_call(d.as_ptr(), p.as_ptr(), i.as_ptr(), m.as_ptr())
};
if ptr.is_null() { Err("D-Bus error: dbus_message_new_method_call failed".into()) }
else { Ok(Message { msg: ptr}) }
}
/// Creates a new method call message.
pub fn method_call(destination: &BusName, path: &Path, iface: &Interface, name: &Member) -> Message {
init_dbus();
let ptr = unsafe {
ffi::dbus_message_new_method_call(destination.as_ptr(), path.as_ptr(),
iface.as_ptr(), name.as_ptr())
};
if ptr.is_null() { panic!("D-Bus error: dbus_message_new_method_call failed") }
Message { msg: ptr}
}
/// Creates a new message that is a replica of this message, but without a serial.
///
/// May fail if out of memory or file descriptors.
pub fn duplicate(&self) -> Result<Self, String> {
let ptr = unsafe {
ffi::dbus_message_copy(self.msg)
};
if ptr.is_null() {
Err("D-Bus error: dbus_message_copy failed".into())
} else {
Ok(Message { msg: ptr })
}
}
/// Creates a new method call message.
pub fn call_with_args<'d, 'p, 'i, 'm, A, D, P, I, M>(destination: D, path: P, iface: I, method: M, args: A) -> Message
where D: Into<BusName<'d>>, P: Into<Path<'p>>, I: Into<Interface<'i>>, M: Into<Member<'m>>, A: AppendAll {
let mut msg = Message::method_call(&destination.into(), &path.into(), &iface.into(), &method.into());
msg.append_all(args);
msg
}
/// Creates a new signal message.
pub fn new_signal<P, I, M>(path: P, iface: I, name: M) -> Result<Message, String>
where P: Into<String>, I: Into<String>, M: Into<String> {
init_dbus();
let p = Path::new(path)?;
let i = Interface::new(iface)?;
let m = Member::new(name)?;
let ptr = unsafe {
ffi::dbus_message_new_signal(p.as_ptr(), i.as_ptr(), m.as_ptr())
};
if ptr.is_null() { Err("D-Bus error: dbus_message_new_signal failed".into()) }
else { Ok(Message { msg: ptr}) }
}
/// Creates a new signal message.
pub fn signal(path: &Path, iface: &Interface, name: &Member) -> Message {
init_dbus();
let ptr = unsafe {
ffi::dbus_message_new_signal(path.as_ptr(), iface.as_ptr(), name.as_ptr())
};
if ptr.is_null() { panic!("D-Bus error: dbus_message_new_signal failed") }
Message { msg: ptr}
}
/// Creates a method reply for this method call.
pub fn new_method_return(m: &Message) -> Option<Message> {
let ptr = unsafe { ffi::dbus_message_new_method_return(m.msg) };
if ptr.is_null() { None } else { Some(Message { msg: ptr} ) }
}
/// Creates a method return (reply) for this method call.
pub fn method_return(&self) -> Message {
let ptr = unsafe { ffi::dbus_message_new_method_return(self.msg) };
if ptr.is_null() { panic!("D-Bus error: dbus_message_new_method_return failed") }
Message {msg: ptr}
}
/// Creates a reply for a method call message.
///
/// Panics if called for a message which is not a method call.
pub fn return_with_args<A: AppendAll>(&self, args: A) -> Message {
let mut m = self.method_return();
m.append_all(args);
m
}
/// Creates a new error reply
pub fn error(&self, error_name: &ErrorName, error_message: &CStr) -> Message {
let ptr = unsafe { ffi::dbus_message_new_error(self.msg, error_name.as_ptr(), error_message.as_ptr()) };
if ptr.is_null() { panic!("D-Bus error: dbus_message_new_error failed") }
Message { msg: ptr}
}
/// Get the MessageItems that make up the message.
///
/// Note: use `iter_init` or `get1`/`get2`/etc instead for faster access to the arguments.
/// This method is provided for backwards compatibility.
pub fn get_items(&self) -> Vec<crate::arg::messageitem::MessageItem> {
let mut i = self.iter_init();
let mut v = vec!();
while let Some(z) = crate::arg::messageitem::MessageItem::get(&mut i) { v.push(z); i.next(); }
v
}
/// Get the D-Bus serial of a message, if one was specified.
pub fn get_serial(&self) -> Option<u32> {
let x = unsafe { ffi::dbus_message_get_serial(self.msg) };
if x == 0 { None } else { Some(x) }
}
/// Get the serial of the message this message is a reply to, if present.
pub fn get_reply_serial(&self) -> Option<u32> {
let s = unsafe { ffi::dbus_message_get_reply_serial(self.msg) };
if s == 0 { None } else { Some(s) }
}
/// Returns true if the message does not expect a reply.
pub fn get_no_reply(&self) -> bool { unsafe { ffi::dbus_message_get_no_reply(self.msg) != 0 } }
/// Set whether or not the message expects a reply.
///
/// Set to true if you send a method call and do not want a reply.
pub fn set_no_reply(&mut self, v: bool) {
unsafe { ffi::dbus_message_set_no_reply(self.msg, if v { 1 } else { 0 }) }
}
/// Returns true if the message can cause a service to be auto-started.
pub fn get_auto_start(&self) -> bool { unsafe { ffi::dbus_message_get_auto_start(self.msg) != 0 } }
/// Sets whether or not the message can cause a service to be auto-started.
///
/// Defaults to true.
pub fn set_auto_start(&mut self, v: bool) {
unsafe { ffi::dbus_message_set_auto_start(self.msg, if v { 1 } else { 0 }) }
}
/// Add one or more MessageItems to this Message.
///
/// Note: using `append1`, `append2` or `append3` might be faster, especially for large arrays.
/// This method is provided for backwards compatibility.
pub fn append_items(&mut self, v: &[crate::arg::messageitem::MessageItem]) {
let mut ia = IterAppend::new(self);
for a in v { a.append_by_ref(&mut ia); }
}
/// Appends one argument to this message.
/// Use in builder style: e g `m.method_return().append1(7i32)`
pub fn append1<A: Append>(mut self, a: A) -> Self {
{
let mut m = IterAppend::new(&mut self);
m.append(a);
}
self
}
/// Appends two arguments to this message.
/// Use in builder style: e g `m.method_return().append2(7i32, 6u8)`
pub fn append2<A1: Append, A2: Append>(mut self, a1: A1, a2: A2) -> Self {
{
let mut m = IterAppend::new(&mut self);
m.append(a1); m.append(a2);
}
self
}
/// Appends three arguments to this message.
/// Use in builder style: e g `m.method_return().append3(7i32, 6u8, true)`
pub fn append3<A1: Append, A2: Append, A3: Append>(mut self, a1: A1, a2: A2, a3: A3) -> Self {
{
let mut m = IterAppend::new(&mut self);
m.append(a1); m.append(a2); m.append(a3);
}
self
}
/// Appends RefArgs to this message.
/// Use in builder style: e g `m.method_return().append_ref(&[7i32, 6u8, true])`
pub fn append_ref<A: RefArg>(mut self, r: &[A]) -> Self {
{
let mut m = IterAppend::new(&mut self);
for rr in r {
rr.append(&mut m);
}
}
self
}
/// Appends arguments to a message.
pub fn append_all<A: AppendAll>(&mut self, a: A) {
let mut m = IterAppend::new(self);
a.append(&mut m);
}
/// Gets the first argument from the message, if that argument is of type G1.
/// Returns None if there are not enough arguments, or if types don't match.
pub fn get1<'a, G1: Get<'a>>(&'a self) -> Option<G1> {
let mut i = Iter::new(&self);
i.get()
}
/// Gets the first two arguments from the message, if those arguments are of type G1 and G2.
/// Returns None if there are not enough arguments, or if types don't match.
pub fn get2<'a, G1: Get<'a>, G2: Get<'a>>(&'a self) -> (Option<G1>, Option<G2>) {
let mut i = Iter::new(&self);
let g1 = i.get();
if !i.next() { return (g1, None); }
(g1, i.get())
}
/// Gets the first three arguments from the message, if those arguments are of type G1, G2 and G3.
/// Returns None if there are not enough arguments, or if types don't match.
pub fn get3<'a, G1: Get<'a>, G2: Get<'a>, G3: Get<'a>>(&'a self) -> (Option<G1>, Option<G2>, Option<G3>) {
let mut i = Iter::new(&self);
let g1 = i.get();
if !i.next() { return (g1, None, None) }
let g2 = i.get();
if !i.next() { return (g1, g2, None) }
(g1, g2, i.get())
}
/// Gets the first four arguments from the message, if those arguments are of type G1, G2, G3 and G4.
/// Returns None if there are not enough arguments, or if types don't match.
pub fn get4<'a, G1: Get<'a>, G2: Get<'a>, G3: Get<'a>, G4: Get<'a>>(&'a self) -> (Option<G1>, Option<G2>, Option<G3>, Option<G4>) {
let mut i = Iter::new(&self);
let g1 = i.get();
if !i.next() { return (g1, None, None, None) }
let g2 = i.get();
if !i.next() { return (g1, g2, None, None) }
let g3 = i.get();
if !i.next() { return (g1, g2, g3, None) }
(g1, g2, g3, i.get())
}
/// Gets the first five arguments from the message, if those arguments are of type G1, G2, G3 and G4.
/// Returns None if there are not enough arguments, or if types don't match.
/// Note: If you need more than five arguments, use `iter_init` instead.
pub fn get5<'a, G1: Get<'a>, G2: Get<'a>, G3: Get<'a>, G4: Get<'a>, G5: Get<'a>>(&'a self) -> (Option<G1>, Option<G2>, Option<G3>, Option<G4>, Option<G5>) {
let mut i = Iter::new(&self);
let g1 = i.get();
if !i.next() { return (g1, None, None, None, None) }
let g2 = i.get();
if !i.next() { return (g1, g2, None, None, None) }
let g3 = i.get();
if !i.next() { return (g1, g2, g3, None, None) }
let g4 = i.get();
if !i.next() { return (g1, g2, g3, g4, None) }
(g1, g2, g3, g4, i.get())
}
/// Gets the first argument from the message, if that argument is of type G1.
///
/// Returns a TypeMismatchError if there are not enough arguments, or if types don't match.
pub fn read1<'a, G1: Arg + Get<'a>>(&'a self) -> Result<G1, TypeMismatchError> {
let mut i = Iter::new(&self);
i.read()
}
/// Gets the first two arguments from the message, if those arguments are of type G1 and G2.
///
/// Returns a TypeMismatchError if there are not enough arguments, or if types don't match.
pub fn read2<'a, G1: Arg + Get<'a>, G2: Arg + Get<'a>>(&'a self) -> Result<(G1, G2), TypeMismatchError> {
let mut i = Iter::new(&self);
Ok((i.read()?, i.read()?))
}
/// Gets the first three arguments from the message, if those arguments are of type G1, G2 and G3.
///
/// Returns a TypeMismatchError if there are not enough arguments, or if types don't match.
pub fn read3<'a, G1: Arg + Get<'a>, G2: Arg + Get<'a>, G3: Arg + Get<'a>>(&'a self) ->
Result<(G1, G2, G3), TypeMismatchError> {
let mut i = Iter::new(&self);
Ok((i.read()?, i.read()?, i.read()?))
}
/// Gets the first four arguments from the message, if those arguments are of type G1, G2, G3 and G4.
///
/// Returns a TypeMismatchError if there are not enough arguments, or if types don't match.
pub fn read4<'a, G1: Arg + Get<'a>, G2: Arg + Get<'a>, G3: Arg + Get<'a>, G4: Arg + Get<'a>>(&'a self) ->
Result<(G1, G2, G3, G4), TypeMismatchError> {
let mut i = Iter::new(&self);
Ok((i.read()?, i.read()?, i.read()?, i.read()?))
}
/// Gets the first five arguments from the message, if those arguments are of type G1, G2, G3, G4 and G5.
///
/// Returns a TypeMismatchError if there are not enough arguments, or if types don't match.
/// Note: If you need more than five arguments, use `iter_init` instead.
pub fn read5<'a, G1: Arg + Get<'a>, G2: Arg + Get<'a>, G3: Arg + Get<'a>, G4: Arg + Get<'a>, G5: Arg + Get<'a>>(&'a self) ->
Result<(G1, G2, G3, G4, G5), TypeMismatchError> {
let mut i = Iter::new(&self);
Ok((i.read()?, i.read()?, i.read()?, i.read()?, i.read()?))
}
/// Gets arguments from a message.
///
/// If this was an error reply or if types mismatch, an error is returned.
pub fn read_all<R: ReadAll>(&self) -> Result<R, Error> {
self.set_error_from_msg()?;
Ok(R::read(&mut self.iter_init())?)
}
/// Returns a struct for retreiving the arguments from a message. Supersedes get_items().
pub fn iter_init(&self) -> Iter { Iter::new(&self) }
/// Gets the MessageType of the Message.
pub fn msg_type(&self) -> MessageType {
match unsafe { ffi::dbus_message_get_type(self.msg) } {
1 => MessageType::MethodCall,
2 => MessageType::MethodReturn,
3 => MessageType::Error,
4 => MessageType::Signal,
x => panic!("Invalid message type {}", x),
}
}
fn msg_internal_str<'a>(&'a self, c: *const libc::c_char) -> Option<&'a str> {
if c.is_null() { return None };
let cc = unsafe { CStr::from_ptr(c) };
std::str::from_utf8(cc.to_bytes_with_nul()).ok()
}
/// Gets the name of the connection that originated this message.
pub fn sender(&self) -> Option<BusName> {
self.msg_internal_str(unsafe { ffi::dbus_message_get_sender(self.msg) })
.map(|s| unsafe { BusName::from_slice_unchecked(s) })
}
/// Gets the object path this Message is being sent to.
pub fn path(&self) -> Option<Path> {
self.msg_internal_str(unsafe { ffi::dbus_message_get_path(self.msg) })
.map(|s| unsafe { Path::from_slice_unchecked(s) })
}
/// Gets the destination this Message is being sent to.
pub fn destination(&self) -> Option<BusName> {
self.msg_internal_str(unsafe { ffi::dbus_message_get_destination(self.msg) })
.map(|s| unsafe { BusName::from_slice_unchecked(s) })
}
/// Sets the destination of this Message
///
/// If dest is none, that means broadcast to all relevant destinations.
pub fn set_destination(&mut self, dest: Option<BusName>) {
let c_dest = dest.as_ref().map(|d| d.as_cstr().as_ptr()).unwrap_or(ptr::null());
assert!(unsafe { ffi::dbus_message_set_destination(self.msg, c_dest) } != 0);
}
/// Gets the interface this Message is being sent to.
pub fn interface(&self) -> Option<Interface> {
self.msg_internal_str(unsafe { ffi::dbus_message_get_interface(self.msg) })
.map(|s| unsafe { Interface::from_slice_unchecked(s) })
}
/// Gets the interface member being called.
pub fn member(&self) -> Option<Member> {
self.msg_internal_str(unsafe { ffi::dbus_message_get_member(self.msg) })
.map(|s| unsafe { Member::from_slice_unchecked(s) })
}
/// When the remote end returns an error, the message itself is
/// correct but its contents is an error. This method will
/// transform such an error to a D-Bus Error or otherwise return
/// the original message.
pub fn as_result(&mut self) -> Result<&mut Message, Error> {
self.set_error_from_msg().map(|_| self)
}
pub (crate) fn set_error_from_msg(&self) -> Result<(), Error> {
let mut e = Error::empty();
if unsafe { ffi::dbus_set_error_from_message(e.get_mut(), self.msg) } != 0 { Err(e) }
else { Ok(()) }
}
pub (crate) fn ptr(&self) -> *mut ffi::DBusMessage { self.msg }
pub (crate) fn from_ptr(ptr: *mut ffi::DBusMessage, add_ref: bool) -> Message {
if add_ref {
unsafe { ffi::dbus_message_ref(ptr) };
}
Message { msg: ptr }
}
/// Sets serial number manually - mostly for internal use
///
/// When sending a message, a serial will be automatically assigned, so you don't need to call
/// this method. However, it can be very useful in test code that is supposed to handle a method call.
/// This way, you can create a method call and handle it without sending it to a real D-Bus instance.
pub fn set_serial(&mut self, val: u32) {
unsafe { ffi::dbus_message_set_serial(self.msg, val) };
}
/// Marshals a message - mostly for internal use
///
/// The function f will be called one or more times with bytes to be written somewhere.
/// You should call set_serial to manually set a serial number before calling this function
pub fn marshal<E, F: FnMut(&[u8]) -> Result<(), E>>(&self, mut f: F) -> Result<(), E> {
let mut len = 0;
let mut data = ptr::null_mut();
if unsafe { ffi::dbus_message_marshal(self.msg, &mut data, &mut len) } == 0 {
panic!("out of memory");
}
let s = unsafe { std::slice::from_raw_parts(data as *mut u8 as *const u8, len as usize) };
let r = f(s);
unsafe { ffi::dbus_free(data as *mut _) };
r
}
/// Demarshals a message - mostly for internal use
pub fn demarshal(data: &[u8]) -> Result<Self, Error> {
let mut e = Error::empty();
let p = unsafe { ffi::dbus_message_demarshal(data.as_ptr() as *const _, data.len() as _, e.get_mut()) };
if p == ptr::null_mut() {
Err(e)
} else {
Ok(Self::from_ptr(p, false))
}
}
/// Returns the size of the message - mostly for internal use
///
/// Returns Err(()) on protocol errors. Make sure you have at least 16 bytes in the buffer
/// before calling this method.
pub fn demarshal_bytes_needed(data: &[u8]) -> Result<usize, ()> {
const MIN_HEADER: usize = 16;
if data.len() < MIN_HEADER { return Ok(MIN_HEADER); }
let x = unsafe { ffi::dbus_message_demarshal_bytes_needed(data.as_ptr() as *const _, data.len() as _) };
if x < MIN_HEADER as _ { Err(()) } else { Ok(x as usize) }
}
}
impl Drop for Message {
fn drop(&mut self) {
unsafe {
ffi::dbus_message_unref(self.msg);
}
}
}
impl fmt::Debug for Message {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
let mut x = f.debug_struct("Message");
x.field("Type", &self.msg_type());
// The &&** derefs to a &&str, which implements &dyn Debug
if let Some(ref path) = self.path() { x.field("Path", &&**path); }
if let Some(ref iface) = self.interface() { x.field("Interface", &&**iface); }
if let Some(ref member) = self.member() { x.field("Member", &&**member); }
if let Some(ref sender) = self.sender() { x.field("Sender", &&**sender); }
if let Some(ref dest) = self.destination() { x.field("Destination", &&**dest); }
if let Some(ref serial) = self.get_serial() { x.field("Serial", serial); }
if let Some(ref rs) = self.get_reply_serial() { x.field("ReplySerial", rs); }
let mut args = vec!();
let mut iter = self.iter_init();
while let Some(a) = iter.get_refarg() {
args.push(a);
iter.next();
}
let args2: &[_] = &args;
x.field("Args", &args2);
x.finish()
}
}
#[cfg(test)]
mod test {
use crate::{Message};
use crate::strings::BusName;
#[test]
fn set_valid_destination() {
let mut m = Message::new_method_call("org.test.rust", "/", "org.test.rust", "Test").unwrap();
let d = Some(BusName::new(":1.14").unwrap());
m.set_destination(d);
assert!(!m.get_no_reply());
m.set_no_reply(true);
assert!(m.get_no_reply());
}
#[test]
fn marshal() {
let mut m = Message::new_method_call("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus", "Hello").unwrap();
m.set_serial(1);
let r = m.marshal(|d| {
let m2 = Message::demarshal(d).unwrap();
assert_eq!(&*m2.path().unwrap(), "/org/freedesktop/DBus");
Err(45)
});
assert_eq!(45, r.unwrap_err());
}
}