annotation-rs

Compile-time annotation parser for rust

Features

Annotation

annotation-rs provides a derive macro Annotation to create annotation structure by struct, StructStruct, TupleStruct and NoFieldStruct are all supported. ```rust use annotation_rs::Annotation;

[derive(Annotation)]

struct NoField;

[derive(Annotation)]

struct Tuple(i32, String);

[derive(Annotation)]

struct Struct { int: i32, float: f64, bool: bool, } ```

Types

```rust use annotation_rs::{AnnotationEnumValue, Annotation};

[derive(Annotation)]

struct Bar;

[derive(AnnotationEnumValue)]

enum SomeEnum { A, B }

[derive(Annotation)]

struct Foo { pub string: String, pub bool: bool, pub int: i32, // or other integer types like u32 ... pub float: f32, // or other float types like f64 pub object: Bar, // any defined object #[field(enumvalue=true)] pub enumfield: SomeEnum, // have to add enum_value option pub list: Vec, // nested type of vec can`t be Object, Vec or HashMap pub map: std::collections::HashMap, pub optional: Option // optional field } ```

Options

Enum

Use derive AnnotationEnumValue on Enum to create a Enum value type. ```rust use annotation_rs::AnnotationEnumValue;

[derive(AnnotationEnumValue)]

enum SomeEnum { A, B } And then, the enum can be used as a field type. * `variant_value` attribute\ Customize a string corresponding value to variant(default is the snake case of variant name in Rust). rust use annotation_rs::AnnotationEnumValue;

[derive(AnnotationEnumValue)]

enum SomeEnum { #[variant_value("aaa")] // default is 'a' A, B } ```

Parse annotations with synandquote

annotation_rs::AnnotationStructures<T> can be used in parse_macro_input! rust let annotations = syn::parse_macro_inpit!(input as annotation_rs::AnnotationStructures<Foo>); If you want to parse annotation from syn::Meta, use annotation_rs::AnnotationStructure::from_meta().\ And annotation structure with value can be convert to token automatically. But the visibility of each field must be public. ```rust use proc_macro::TokenStream;

[derive(Annotation)]

struct Foo { #[field(default = 1024)] pub int32: i32 }

fn derivefn(input: TokenStream) -> TokenStream { let annotations = syn::parsemacroinput!(input as annotationrs::AnnotationStructures); let attrs = annotations.attrs;

TokenStream::from(quote::quote! {
    fn get_attrs() -> Vec<Foo> {
        vec![#(#attrs),*]
    }
})

} ```

Generate derive macro

If you want to use builtin reader generator, enable annotation_reader feature. Macro generate_reader is used to generate a derive macro. ```rust use annotationrs::generatereader;

generated_reader!( MyDerive, [StructAttribute1, StructAttribute2], [FieldAttribute1, FieldAttribute2] );

`` The macro will generate a public derive, it can be use to read annotations ofstruct,enumorunion, and record the metadata by generateimpl` block.

Read annotations

Use the generated derive macro on a struct, and you can use the macro has_annotation and get_annotationto process annotations of the struct. The feature require nightly rustc because proc_macro_hygiene is required. ```rust

![feature(procmacrohygiene)]

use annotationrs::{getannotation, has_annotation};

[derive(MyDerive)]

[StructAttribute1("some parameters")]

struct Foo { #[FieldAttribute1("some parameters")] field: i32 }

fn somefn() { assert!(hasannotation!(Foo, StructAttribute1)); assert!(hasannotation!(Foo::field, FieldAttribute1)); let structattr1: Option = getannotation!(Foo, StructAttribute1); let fieldattr1: Option = get_annotation!(Foo::field, StructAttribute1); } ```