| Crates.io | capnp_conv_macros |
| lib.rs | capnp_conv_macros |
| version | 0.3.2 |
| created_at | 2023-10-22 17:31:54.929863+00 |
| updated_at | 2024-11-11 13:33:22.454105+00 |
| description | Provides attribute macros that implement capnp read/write traits |
| homepage | |
| repository | https://github.com/aikalant/capnp_conv |
| max_upload_size | |
| id | 1010684 |
| size | 61,497 |
The capnp_conv macro facilitates conversions between Cap'n Proto structs/unions/unions/groups to their rust counterparts by implementing all the necessary Capn' Proto Rust generated builder/reader calls.
Inspired by the (seemingly abandoned?) existing PR by @realcr.
The following capnp schema file directly translates to the below rust file:
struct SomeStruct { }
enum CapnpEnum {
val1 @0;
val2 @1;
}
struct CapnpStruct {
voidVal @0 :Void;
i32Val @1 :Int32;
textVal @2 :Text;
dataVal @3 :Data;
structVal @4 :SomeStruct;
enumVal @5 :CapnpEnum;
listVal @6 :List(SomeStruct);
}
#[capnp_conv(some_struct)]
pub struct SomeStruct {}
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
void_val: (),
i32_val: i32,
text_val: String,
#[capnp_conv(type = "data")]
data_val: Vec<u8>,
struct_val: SomeStruct,
#[capnp_conv(type = "enum")]
enum_val: CapnpEnum,
list_val: Vec<SomeStruct>,
}
The capnp_conv proc macro implements the capnp_conv::Readable, capnp_conv::Writable, and TryFrom<Reader> traits, which handle all of the reading/writing:
fn read(reader: capnp_struct::Reader) -> Result<RustStruct, capnp::Error> {
RustStruct::read(reader)?
}
fn write(rust_struct: RustStruct, builder: capnp_struct::Builder) {
rust_struct.write(builder)
}
Capnp group, enum, union, and data types require the field attribute with a type specifier.
Capnp groups are represented by separate rust structs.
struct CapnpStruct {
groupVal :group {
val @0 :Void;
}
}
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
#[capnp_conv(type = "group")]
group_val: RustStructGroup,
}
#[capnp_conv(capnp_struct::group_val)]
pub struct RustStructGroup {
val: ()
}
The macro has two options when it comes to enums: enum and enum_remote. Because code generated capnp files already contain enum definitions, it is possible to use them directly with enum. This eliminates the need to write an extra enum definition, but there are some times when it is useful to define a separate enum, for example, if it is necessary to derive traits or use other macros on the enum. For these cases, use enum_remote with a separately defined rust enum. When used on a rust enum, the capnp_conv macro generates the from/into trait implementations for its capnp counterpart.
enum CapnpEnum {
val1 @0;
val2 @1;
}
struct CapnpStruct {
enumVal @0 :CapnpEnum;
enumValRemote @1 :CapnpEnum;
}
#[capnp_conv(CapnpEnum)]
pub enum RustEnum {
Val1,
Val2,
}
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
#[capnp_conv(type = "enum")]
enum_val: CapnpEnum,
#[capnp_conv(type = "enum_remote")]
enum_val_remote: RustEnum,
}
Unions can be represented in two different ways. One is by using rust enums, but structs can also represent capnp unions with fields wrapped in Option<T>s and containing the #[capnp_conv(union_variant)] attribute macro. This eliminates the need to create a separate item for unnamed unions, but it can be more cumbersome to manipulate.
Rust enum variants have the same requirements for using the #[capnp_conv(type = xxx)] field attributes.
struct CapnpStruct {
namedUnion :union {
val1 @0 :Void;
val2 @1 :Void;
}
union {
val1 @1 :Void;
val2 @2 :Void;
}
}
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
#[capnp_conv(type = "union")]
named_union: RustStructUnion,
#[capnp_conv(type = "unnamed_union")]
unnamed_union: RustStructUnamedUnion,
}
#[capnp_conv(capnp_struct::named_union)]
pub enum RustStructUnion {
Val1(()),
Val2(()).
}
#[capnp_conv(capnp_struct)]
pub enum RustStructUnamedUnion {
Val1(()),
Val2(()).
}
// or
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
#[capnp_conv(type = "union")]
named_union: RustStructUnion,
#[capnp_conv(union_variant)]
val1: Option<()>,
#[capnp_conv(union_variant)]
val2: Option<()>,
}
#[capnp_conv(capnp_struct::named_union)]
pub struct RustStructUnion {
#[capnp_conv(union_variant)]
val1: Option<()>,
#[capnp_conv(union_variant)]
val2: Option<()>,
}
The capnp Data type is functionally identical to List(UInt8), both of which are represented with Vec<u8> in rust. However, capnpc generates two distinct structs that handle reading and writing of the two types. There is no way to specify which one a Vec<u8> is intended to represent, which necessitates the use of the field attribute in the case of Data types.
struct CapnpStruct {
named_union :union {
list @0 :List(UInt8);
data @1 :Data;
}
}
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
list: Vec<u8>,
#[capnp_conv(type = "data")]
data: Vec<u8>,
}
capnp_conv includes several other features that can be enabled through setting options in field attributes
Rust field names must normally match the field names of their capnp counterparts (converted to the appropriate rust case). Using the name attribute, it is possible to disentangle them.
struct CapnpStruct {
capVal @0 :Void;
capnpUnion :union {
capVal1 @1 : Void;
capVal2 @2 : Void;
}
}
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
#[capnp_conv(name = "capVal")] //or #[capnp_conv(name = "cap_val")]
arbitrary_name: (),
#[capnp_conv(name = "capnp_union")]
#[capnp_conv(type = "union")]
rust_union: RustUnnion,
}
#[capnp_conv(capnp_struct::capnp_union)]
pub enum RustUnion {
#[capnp_conv(name = "capVal1")]
ArbitraryName1(())
#[capnp_conv(name = "cap_val2")]
ArbitraryName2(())
}
Wrapping a field with Option<T> indicates that the field is optional.
When a struct is written to a builder, optional fields will be skipped if they are None.
When a struct is read from a reader, optional fields that are pointer types (List, Struct, Text, Data) will be set to None if the field has not been set in the capnp message. Primitive types and enums will always be read and set to Some.
struct CapnpStruct {
capVal @0 :Void;
}
#[capnp_conv(capnp_struct)]
pub struct RustStruct {
cap_val: Option<()>,
}
Adding #[capnp_conv(skip_write)] to a field's attributes will result in the field not being written. Similar to as if it were optional with None.
Adding #[capnp_conv(skip_read)] to a field's attributes will result in the field never being read. During a read, it be set to the field type's default value (note the field type must implement the Default trait), or None if the field is optional. This can be futher configured by setting the field's default attribute (see below).
#[capnp_conv(skip)] skips both reading and writing.
Unions cannot contain skipped fields.
Setting the #[capnp_conv(default = "path_to_func_to_call")] attribute to a field that is configured as skip_read or skip will set the field to the output of calling the function specified rather than the default value during a read.
Unions cannot contain default overrides.
Generics are supported for both structs and enums.
struct CapnpStruct(T,Y) {
tVal @0 :T;
list @1 :List(Y);
unionVal :union {
tVal @2 :T;
list @3 :List(Y);
}
}
#[capnp_conv(capnp_struct)]
pub struct RustStruct<T,Y> {
t_val: T,
list: Vec<Y>,
#[capnp_conv(type = "union")]
union_val: RustStructUnionVal<T,Y>,
}
#[capnp_conv(capnp_struct::union_val)]
pub enum RustStructUnionVal<T,Y> {
TVal(T),
List(Vec<Y>),
}
Limitations:
One feature of capnp schemas that is not easily reproduced in rust is nested struct definitions. This is not typically an issue as they can be implemented as flattened rust structs, but when combined with the fact that nested capnp structs/unions/groups have access to all the generic types of all of their ancestors, this can be problematic for rust models.
For example:
struct ParentStruct(T,Y,R) {
tVal @0 :T;
yVal @1 :Y;
rVal @2 :R;
unionVal :union {
voidVal @3 :Void,
tVal @4 :T,
}
struct ChildStruct {
tVal @0 :T
}
}
When defining parent_struct::child_struct in rust, the struct must have T, Y, and R even though it only uses T. Rust does not allow structs to have generics that they do not use, but we can use PhantomData
pub struct ChildStruct<T,Y,R> {
t_val: T,
phantom: PhantomData<*const (Y,R)>,
}
When reading/writing, PhantomData types will be automatically skipped, even without the field attribute.
Unfortunately there is no "PhantomVariant" for rust enums, so for unions, instead make the second field of any variant a PhantomData:
pub enum UnionVal(T,Y,R) {
VoidVal((), PhantomData<*const (Y,R)>),
TVal(T),
}
Short term:
Vec read and writeanypointer typetype = "data" attribute must be of type Vec<u8>Type1<Type2<T>>)Long term:
clear_enum_fields function to struct represented capnp unions that sets all union fields to None.