#serialization #object

no-std valuable

Object-safe value inspection, used to pass un-typed structured data across trait-object boundaries

3 unstable releases

new 0.1.1 Jan 17, 2025
0.1.0 Jan 3, 2022
0.0.0 Mar 9, 2021

#239 in Debugging

Download history 745925/week @ 2024-09-27 847547/week @ 2024-10-04 791060/week @ 2024-10-11 850194/week @ 2024-10-18 861171/week @ 2024-10-25 827306/week @ 2024-11-01 831714/week @ 2024-11-08 829842/week @ 2024-11-15 757211/week @ 2024-11-22 754400/week @ 2024-11-29 873919/week @ 2024-12-06 858616/week @ 2024-12-13 428656/week @ 2024-12-20 432153/week @ 2024-12-27 805785/week @ 2025-01-03 849542/week @ 2025-01-10

2,671,472 downloads per month
Used in 10,702 crates (29 directly)

MIT license

130KB
1.5K SLoC

Valuable

Valuable provides object-safe value inspection. Use cases include passing structured data to trait objects and object-safe serialization.

License

This project is licensed under the MIT license.

Contribution

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in Valuable by you, shall be licensed as MIT, without any additional terms or conditions.


lib.rs:

Valuable provides object-safe value inspection. Use cases include passing structured data to trait objects and object-safe serialization.

Getting started

First, derive [Valuable][macro@crate::Valuable] on your types.

use valuable::Valuable;

#[derive(Valuable)]
struct HelloWorld {
    message: Message,
}

#[derive(Valuable)]
enum Message {
    HelloWorld,
    Custom(String),
}

Then, implement a [visitor][Visit] to inspect the data.

use valuable::{NamedValues, Value, Valuable, Visit};

struct Print;

impl Visit for Print {
    fn visit_value(&mut self, value: Value<'_>) {
        match value {
            Value::Structable(v) => {
                println!("struct {}", v.definition().name());
                v.visit(self);
            }
            Value::Enumerable(v) => {
                println!("enum {}::{}", v.definition().name(), v.variant().name());
                v.visit(self);
            }
            Value::Listable(v) => {
                println!("list");
                v.visit(self);
            }
            Value::Mappable(v) => {
                println!("map");
                v.visit(self);
            }
            _ => {
                println!("value {:?}", value);
            }
        }
    }

    fn visit_named_fields(&mut self, named_fields: &NamedValues<'_>) {
        for (field, value) in named_fields.iter() {
            println!("named field {}", field.name());
            value.visit(self);
        }
    }

    fn visit_unnamed_fields(&mut self, values: &[Value<'_>]) {
        for value in values {
            value.visit(self);
        }
    }

    fn visit_entry(&mut self, key: Value<'_>, value: Value<'_>) {
        println!("key / value");
        key.visit(self);
        value.visit(self);
    }
}

Then, use the visitor to visit the value.

let hello_world = HelloWorld { message: Message::HelloWorld };
hello_world.visit(&mut Print);

Related Crates

Dependencies

~100KB