Struct value_bag::ValueBag

source ·
pub struct ValueBag<'v> { /* private fields */ }
Expand description

A dynamic structured value.

Capturing values

There are a few ways to capture a value:

  • Using the ValueBag::capture_* and ValueBag::from_* methods.
  • Using the standard From trait.
  • Using the Fill API.

Using the ValueBag::capture_* methods

ValueBag offers a few constructor methods that capture values of different kinds. These methods require a T: 'static to support downcasting.

use value_bag::ValueBag;

let value = ValueBag::capture_debug(&42i32);

assert_eq!(Some(42), value.to_i64());

Capturing a value using these methods will retain type information so that the contents of the bag can be serialized using an appropriate type.

For cases where the 'static bound can’t be satisfied, there’s also a few constructors that exclude it.

use value_bag::ValueBag;

let value = ValueBag::from_debug(&42i32);

assert_eq!(None, value.to_i64());

These ValueBag::from_* methods are lossy though and ValueBag::capture_* should be preferred.

Using the standard From trait

Primitive types can be converted into a ValueBag using the standard From trait.

use value_bag::ValueBag;

let value = ValueBag::from(42i32);

assert_eq!(Some(42), value.to_i64());

Using the Fill API

The fill module provides a way to bridge APIs that may not be directly compatible with other constructor methods.

The Fill trait is automatically implemented for closures, so can usually be used in libraries that can’t implement the trait themselves.

use value_bag::{ValueBag, fill::Slot};

let value = ValueBag::from_fill(&|slot: Slot| {
    #[derive(Debug)]
    struct MyShortLivedValue;

    slot.fill_debug(&MyShortLivedValue)
});

assert_eq!("MyShortLivedValue", format!("{:?}", value));

The trait can also be implemented manually:

use value_bag::{ValueBag, Error, fill::{Slot, Fill}};

struct FillDebug;

impl Fill for FillDebug {
    fn fill(&self, slot: Slot) -> Result<(), Error> {
        slot.fill_debug(&42i32 as &dyn Debug)
    }
}

let value = ValueBag::from_fill(&FillDebug);

assert_eq!(None, value.to_i64());

Inspecting values

Once you have a ValueBag there are also a few ways to inspect it:

  • Using std::fmt
  • Using sval
  • Using serde
  • Using the ValueBag::visit method.
  • Using the ValueBag::to_* methods.
  • Using the ValueBag::downcast_ref method.

Using the ValueBag::visit method

The visit module provides a simple visitor API that can be used to inspect the structure of primitives stored in a ValueBag. More complex datatypes can then be handled using std::fmt, sval, or serde.

#[cfg(not(feature = "std"))] fn main() {}
#[cfg(feature = "std")]
use value_bag::{ValueBag, Error, visit::Visit};

// Implement some simple custom serialization
struct MyVisit(Vec<u8>);
impl<'v> Visit<'v> for MyVisit {
    fn visit_any(&mut self, v: ValueBag) -> Result<(), Error> {
        // Fallback to `Debug` if we didn't visit the value specially
        write!(&mut self.0, "{:?}", v).map_err(|_| Error::msg("failed to write value"))
    }

    fn visit_u64(&mut self, v: u64) -> Result<(), Error> {
        self.0.extend_from_slice(itoa_fmt(v).as_slice());
        Ok(())
    }

    fn visit_i64(&mut self, v: i64) -> Result<(), Error> {
        self.0.extend_from_slice(itoa_fmt(v).as_slice());
        Ok(())
    }

    fn visit_f64(&mut self, v: f64) -> Result<(), Error> {
        self.0.extend_from_slice(ryu_fmt(v).as_slice());
        Ok(())
    }

    fn visit_str(&mut self, v: &str) -> Result<(), Error> {
        self.0.push(b'\"');
        self.0.extend_from_slice(escape(v.as_bytes()));
        self.0.push(b'\"');
        Ok(())
    }

    fn visit_bool(&mut self, v: bool) -> Result<(), Error> {
        self.0.extend_from_slice(if v { b"true" } else { b"false" });
        Ok(())
    }
}

let value = ValueBag::from(42i64);

let mut visitor = MyVisit(vec![]);
value.visit(&mut visitor)?;

Using std::fmt

Any ValueBag can be formatted using the std::fmt machinery as either Debug or Display.

use value_bag::ValueBag;

let value = ValueBag::from(true);

assert_eq!("true", format!("{:?}", value));

Using sval

When the sval2 feature is enabled, any ValueBag can be serialized using sval. This makes it possible to visit any typed structure captured in the ValueBag, including complex datatypes like maps and sequences.

sval doesn’t need to allocate so can be used in no-std environments.

First, enable the sval2 feature in your Cargo.toml:

[dependencies.value-bag]
features = ["sval2"]

Then stream the contents of the ValueBag using sval.

use value_bag::ValueBag;

let value = ValueBag::from(42i64);
let json = sval_json::stream_to_string(value)?;

Using serde

When the serde1 feature is enabled, any ValueBag can be serialized using serde. This makes it possible to visit any typed structure captured in the ValueBag, including complex datatypes like maps and sequences.

serde needs a few temporary allocations, so also brings in the std feature.

First, enable the serde1 feature in your Cargo.toml:

[dependencies.value-bag]
features = ["serde1"]

Then stream the contents of the ValueBag using serde.

use value_bag::ValueBag;

let value = ValueBag::from(42i64);
let json = serde_json::to_string(&value)?;

Also see serde.rs for more examples of writing your own serializers.

Using the ValueBag::to_* methods

ValueBag provides a set of methods for attempting to pull a concrete value out. These are useful for ad-hoc analysis but aren’t intended for exhaustively serializing the contents of a ValueBag.

use value_bag::ValueBag;

let value = ValueBag::capture_display(&42u64);

assert_eq!(Some(42u64), value.to_u64());

Using the ValueBag::downcast_ref method

When a ValueBag is created using one of the capture_* constructors, it can be downcast back to its original value. This can also be useful for ad-hoc analysis where there’s a common possible non-primitive type that could be captured.

use value_bag::ValueBag;

let timestamp = now();
let value = ValueBag::capture_debug(&timestamp);

assert!(value.downcast_ref::<SystemTime>().is_some());

Implementations§

source§

impl<'v> ValueBag<'v>

source

pub fn to_str(&self) -> Option<Cow<'v, str>>

Try get a str from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones. If the serialization implementation produces a short lived string it will be allocated.

source§

impl<'v> ValueBag<'v>

source

pub fn try_capture<T>(value: &'v T) -> Option<Self>where T: ?Sized + 'static,

Try capture a raw value.

This method will return Some if the value is a simple primitive that can be captured without losing its structure. In other cases this method will return None.

source

pub fn to_u64(&self) -> Option<u64>

Try get a u64 from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones.

source

pub fn to_i64(&self) -> Option<i64>

Try get a i64 from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones.

source

pub fn to_u128(&self) -> Option<u128>

Try get a u128 from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones.

source

pub fn to_i128(&self) -> Option<i128>

Try get a i128 from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones.

source

pub fn to_f64(&self) -> Option<f64>

Try get a f64 from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones.

source

pub fn to_bool(&self) -> Option<bool>

Try get a bool from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones.

source

pub fn to_char(&self) -> Option<char>

Try get a char from this value.

This method is cheap for primitive types, but may call arbitrary serialization implementations for complex ones.

source

pub fn to_borrowed_str(&self) -> Option<&'v str>

Try get a str from this value.

This method is cheap for primitive types. It won’t allocate an owned String if the value is a complex type.

source

pub fn is<T: 'static>(&self) -> bool

Check whether this value can be downcast to T.

source

pub fn downcast_ref<T: 'static>(&self) -> Option<&T>

Try downcast this value to T.

source§

impl<'v> ValueBag<'v>

source

pub fn capture_error<T>(value: &'v T) -> Selfwhere T: Error + 'static,

Get a value from an error.

source

pub fn from_dyn_error(value: &'v (dyn Error + 'static)) -> Self

Get a value from an erased value.

source

pub fn to_borrowed_error(&self) -> Option<&'v (dyn Error + 'static)>

Try get an error from this value.

source§

impl<'v> ValueBag<'v>

source

pub fn from_fill<T>(value: &'v T) -> Selfwhere T: Fill,

Get a value from a fillable slot.

source§

impl<'v> ValueBag<'v>

source

pub fn capture_debug<T>(value: &'v T) -> Selfwhere T: Debug + 'static,

Get a value from a debuggable type.

This method will attempt to capture the given value as a well-known primitive before resorting to using its Debug implementation.

source

pub fn capture_display<T>(value: &'v T) -> Selfwhere T: Display + 'static,

Get a value from a displayable type.

This method will attempt to capture the given value as a well-known primitive before resorting to using its Display implementation.

source

pub fn from_debug<T>(value: &'v T) -> Selfwhere T: Debug,

Get a value from a debuggable type without capturing support.

source

pub fn from_display<T>(value: &'v T) -> Selfwhere T: Display,

Get a value from a displayable type without capturing support.

source

pub fn from_dyn_debug(value: &'v dyn Debug) -> Self

Get a value from a debuggable type without capturing support.

source

pub fn from_dyn_display(value: &'v dyn Display) -> Self

Get a value from a displayable type without capturing support.

source§

impl<'v> ValueBag<'v>

source

pub fn capture_serde1<T>(value: &'v T) -> Selfwhere T: Serialize + 'static,

Get a value from a structured type.

This method will attempt to capture the given value as a well-known primitive before resorting to using its Value implementation.

source

pub fn from_serde1<T>(value: &'v T) -> Selfwhere T: Serialize,

Get a value from a structured type without capturing support.

source§

impl<'v> ValueBag<'v>

source

pub fn capture_sval2<T>(value: &'v T) -> Selfwhere T: Value + 'static,

Get a value from a structured type.

This method will attempt to capture the given value as a well-known primitive before resorting to using its Value implementation.

source

pub fn from_sval2<T>(value: &'v T) -> Selfwhere T: Value,

Get a value from a structured type without capturing support.

source

pub fn from_dyn_sval2(value: &'v dyn Value) -> Self

Get a value from a structured type without capturing support.

source§

impl<'v> ValueBag<'v>

source

pub fn visit(&self, visitor: impl Visit<'v>) -> Result<(), Error>

Visit this value using a simple visitor.

The visitor isn’t strictly required to inspect the contents of a value bag. It’s useful for simple cases where a full framework like serde or sval isn’t necessary.

source§

impl<'v> ValueBag<'v>

source

pub fn to_test_token(&self) -> TestToken

Convert the value bag into a token for testing.

This isn’t a general-purpose API for working with values outside of testing.

source§

impl<'v> ValueBag<'v>

source

pub fn by_ref<'u>(&'u self) -> ValueBag<'u>

Get a ValueBag from a reference to a ValueBag.

Trait Implementations§

source§

impl<'v> Clone for ValueBag<'v>

source§

fn clone(&self) -> ValueBag<'v>

Returns a copy of the value. Read more
1.0.0 · source§

fn clone_from(&mut self, source: &Self)

Performs copy-assignment from source. Read more
source§

impl<'v> Debug for ValueBag<'v>

source§

fn fmt(&self, f: &mut Formatter<'_>) -> Result

Formats the value using the given formatter. Read more
source§

impl<'v> Display for ValueBag<'v>

source§

fn fmt(&self, f: &mut Formatter<'_>) -> Result

Formats the value using the given formatter. Read more
source§

impl<'a, 'v> From<&'a ()> for ValueBag<'v>

source§

fn from(value: &'a ()) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a bool> for ValueBag<'v>

source§

fn from(value: &'a bool) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a char> for ValueBag<'v>

source§

fn from(value: &'a char) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a f32> for ValueBag<'v>

source§

fn from(value: &'a f32) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a f64> for ValueBag<'v>

source§

fn from(value: &'a f64) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a i16> for ValueBag<'v>

source§

fn from(value: &'a i16) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a i32> for ValueBag<'v>

source§

fn from(value: &'a i32) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a i64> for ValueBag<'v>

source§

fn from(value: &'a i64) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a i8> for ValueBag<'v>

source§

fn from(value: &'a i8) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a isize> for ValueBag<'v>

source§

fn from(value: &'a isize) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a u16> for ValueBag<'v>

source§

fn from(value: &'a u16) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a u32> for ValueBag<'v>

source§

fn from(value: &'a u32) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a u64> for ValueBag<'v>

source§

fn from(value: &'a u64) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a u8> for ValueBag<'v>

source§

fn from(value: &'a u8) -> Self

Converts to this type from the input type.
source§

impl<'a, 'v> From<&'a usize> for ValueBag<'v>

source§

fn from(value: &'a usize) -> Self

Converts to this type from the input type.
source§

impl<'v> From<&'v String> for ValueBag<'v>

source§

fn from(v: &'v String) -> ValueBag<'v>

Converts to this type from the input type.
source§

impl<'v> From<&'v i128> for ValueBag<'v>

source§

fn from(value: &'v i128) -> Self

Converts to this type from the input type.
source§

impl<'v> From<&'v str> for ValueBag<'v>

source§

fn from(value: &'v str) -> Self

Converts to this type from the input type.
source§

impl<'v> From<&'v u128> for ValueBag<'v>

source§

fn from(value: &'v u128) -> Self

Converts to this type from the input type.
source§

impl<'v> From<()> for ValueBag<'v>

source§

fn from(value: ()) -> Self

Converts to this type from the input type.
source§

impl<'v> From<bool> for ValueBag<'v>

source§

fn from(value: bool) -> Self

Converts to this type from the input type.
source§

impl<'v> From<char> for ValueBag<'v>

source§

fn from(value: char) -> Self

Converts to this type from the input type.
source§

impl<'v> From<f32> for ValueBag<'v>

source§

fn from(value: f32) -> Self

Converts to this type from the input type.
source§

impl<'v> From<f64> for ValueBag<'v>

source§

fn from(value: f64) -> Self

Converts to this type from the input type.
source§

impl<'v> From<i16> for ValueBag<'v>

source§

fn from(value: i16) -> Self

Converts to this type from the input type.
source§

impl<'v> From<i32> for ValueBag<'v>

source§

fn from(value: i32) -> Self

Converts to this type from the input type.
source§

impl<'v> From<i64> for ValueBag<'v>

source§

fn from(value: i64) -> Self

Converts to this type from the input type.
source§

impl<'v> From<i8> for ValueBag<'v>

source§

fn from(value: i8) -> Self

Converts to this type from the input type.
source§

impl<'v> From<isize> for ValueBag<'v>

source§

fn from(value: isize) -> Self

Converts to this type from the input type.
source§

impl<'v> From<u16> for ValueBag<'v>

source§

fn from(value: u16) -> Self

Converts to this type from the input type.
source§

impl<'v> From<u32> for ValueBag<'v>

source§

fn from(value: u32) -> Self

Converts to this type from the input type.
source§

impl<'v> From<u64> for ValueBag<'v>

source§

fn from(value: u64) -> Self

Converts to this type from the input type.
source§

impl<'v> From<u8> for ValueBag<'v>

source§

fn from(value: u8) -> Self

Converts to this type from the input type.
source§

impl<'v> From<usize> for ValueBag<'v>

source§

fn from(value: usize) -> Self

Converts to this type from the input type.
source§

impl<'v> Serialize for ValueBag<'v>

source§

fn serialize<S>(&self, s: S) -> Result<S::Ok, S::Error>where S: Serializer,

Serialize this value into the given Serde serializer. Read more
source§

impl<'v> Value for ValueBag<'v>

source§

fn stream<'sval, S: Stream<'sval> + ?Sized>(&'sval self, s: &mut S) -> Result

Stream this value through a [Stream].
§

fn to_bool(&self) -> Option<bool>

Try convert this value into a boolean.
§

fn to_f32(&self) -> Option<f32>

Try convert this value into a 32bit binary floating point number.
§

fn to_f64(&self) -> Option<f64>

Try convert this value into a 64bit binary floating point number.
§

fn to_i8(&self) -> Option<i8>

Try convert this value into a signed 8bit integer.
§

fn to_i16(&self) -> Option<i16>

Try convert this value into a signed 16bit integer.
§

fn to_i32(&self) -> Option<i32>

Try convert this value into a signed 32bit integer.
§

fn to_i64(&self) -> Option<i64>

Try convert this value into a signed 64bit integer.
§

fn to_i128(&self) -> Option<i128>

Try convert this value into a signed 128bit integer.
§

fn to_u8(&self) -> Option<u8>

Try convert this value into an unsigned 8bit integer.
§

fn to_u16(&self) -> Option<u16>

Try convert this value into an unsigned 16bit integer.
§

fn to_u32(&self) -> Option<u32>

Try convert this value into an unsigned 32bit integer.
§

fn to_u64(&self) -> Option<u64>

Try convert this value into an unsigned 64bit integer.
§

fn to_u128(&self) -> Option<u128>

Try convert this value into an unsigned 128bit integer.
§

fn to_text(&self) -> Option<&str>

Try convert this value into a text string.
§

fn to_binary(&self) -> Option<&[u8]>

Try convert this value into a bitstring.
source§

impl<'sval> ValueRef<'sval> for ValueBag<'sval>

source§

fn stream_ref<S: Stream<'sval> + ?Sized>(&self, s: &mut S) -> Result

Stream this value through a [Stream].

Auto Trait Implementations§

§

impl<'v> !RefUnwindSafe for ValueBag<'v>

§

impl<'v> !Send for ValueBag<'v>

§

impl<'v> !Sync for ValueBag<'v>

§

impl<'v> Unpin for ValueBag<'v>

§

impl<'v> !UnwindSafe for ValueBag<'v>

Blanket Implementations§

source§

impl<T> Any for Twhere T: 'static + ?Sized,

source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
source§

impl<T> Borrow<T> for Twhere T: ?Sized,

source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
source§

impl<T> BorrowMut<T> for Twhere T: ?Sized,

source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
source§

impl<T> From<T> for T

source§

fn from(t: T) -> T

Returns the argument unchanged.

source§

impl<T, U> Into<U> for Twhere U: From<T>,

source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

source§

impl<T> Serialize for Twhere T: Serialize + ?Sized,

source§

fn erased_serialize(&self, serializer: &mut dyn Serializer) -> Result<Ok, Error>

source§

impl<T> ToOwned for Twhere T: Clone,

§

type Owned = T

The resulting type after obtaining ownership.
source§

fn to_owned(&self) -> T

Creates owned data from borrowed data, usually by cloning. Read more
source§

fn clone_into(&self, target: &mut T)

Uses borrowed data to replace owned data, usually by cloning. Read more
source§

impl<T> ToString for Twhere T: Display + ?Sized,

source§

default fn to_string(&self) -> String

Converts the given value to a String. Read more
source§

impl<T, U> TryFrom<U> for Twhere U: Into<T>,

§

type Error = Infallible

The type returned in the event of a conversion error.
source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
source§

impl<T, U> TryInto<U> for Twhere U: TryFrom<T>,

§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
§

impl<T> Value for Twhere T: Value,