This program takes D-Bus XML Introspection data and generates Rust code for calling and implementing the interfaces in the introspection data.
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.
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
pub struct OrgExampleTestLaundry { pub eaten: bool, }
impl dbus::SignalArgs for OrgExampleTestLaundry { /* code here */ } ```
ConnPath
, which makes methods easy to call for a client, like this:rust
use OrgExampleTest;
let myString = try!(myConnPath.foo(myInteger));
match_str
and from_message
, like this: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);
}
}
tree::Interface
, like this: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
.
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 */
}
}
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());
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
).
SignalArgs::to_emit_message
or ConnPath::emit
to get a message which can be sent over the connection.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