dbus-codegen-rust

This program takes D-Bus XML Introspection data and generates Rust code for calling and implementing the interfaces in the introspection data.

Example

From a D-Bus interface like this:

<node>
    <interface name="org.example.test">
        <method name="Foo">
            <arg type="i" name="bar" direction="in"/>
            <arg type="s" name="baz" direction="out"/>
        </method>
        <signal name="Laundry">
            <arg type="b" name="eaten"/>
        </signal>
    </interface>
 </node>

This code will generate a few things.

Common for client and server sides

rust pub trait OrgExampleTest { type Err; fn foo(&self, bar: i32) -> Result<String, Self::Err>; }

For properties, get_xx and set_xx methods will be generated. There is currently no get_all method.

```rust

[derive(Debug, Default)]

pub struct OrgExampleTestLaundry { pub eaten: bool, }

impl dbus::SignalArgs for OrgExampleTestLaundry { /* code here */ } ```

Client side

rust use OrgExampleTest; let myString = try!(myConnPath.foo(myInteger));

rust use SignalArgs; myConnection.add_match(OrgExampleTestLaundry::match_str(None, None)); for msg in c.incoming(1000) { if let Some(laundrySignal) = OrgExampleTestLaundry::from_message(&msg) { println!("Laundry was eaten: {:?}", laundrySignal.eaten); } }

Server side

rust myInterface = orgexampletest_server(&myFactory, ());

This interface can then be added to a tree::ObjectPath, as shown in the main page.

In addition, you also need to implement the interface's methods, like this:

rust impl OrgExampleTest for MyStruct { type Err = tree::MethodErr; fn foo(&self, bar: i32) -> Result<String, Self::Err> { /* Your code here */ } }

I've been experimenting with different ways of how to make the generated server function reach the implementing struct, this is controlled by the command line parameter methodaccess.

  1. If methodaccess is MethodInfo, then you need to implement the interface for the MethodInfo struct, like this:

rust impl<M: tree::MethodType<D>, D> OrgExampleTest for tree::MethodInfo<M, D> { type Err = tree::MethodErr; fn foo(&self, bar: i32) -> Result<String, Self::Err> { /* Your code here */ } }

  1. If methodaccess is RefClosure, then you need to supply a closure that returns a reference to the implementing struct. This is a good option if the struct is stored in tree (this means implementing tree::DataType).

rust myInterface = orgexampletest_server(&myFactory, (), |m| m.path.get_data());

  1. If methodaccess is AsRefClosure, then you need to supply a closure that returns an object which can reference to the implementing struct. The object is dropped after the method is called. This works well with Arc/Rc, like this:

rust let myRc = Rc::new(myStruct); myInterface = orgexampletest_server(&myFactory, (), move |_| myRc.clone());

There is also a methodtype parameter that controls whether the server function will work well with MTFn, MTFnMut or MTSync trees, or all three (called Generic). Or not generate a server function at all (None).

Usage

Once you have installed dbus-codegen-rust, use the following command to import your XML:

dbus-codegen-rust < mydefinition.xml

This will print the generated Rust code to stdout, so you can pipe it into another file if you want:

dbus-codegen-rust < mydefinition.xml > mod.rs

Dbus-codegen-rust can also fetch the xml definition for you. Here's an example that generates client definitions for PolicyKit:

dbus-codegen-rust -s -d org.freedesktop.PolicyKit1 -p "/org/freedesktop/PolicyKit1/Authority" -m None > policykit.rs

See available options:

dbus-codegen-rust --help