The [EmptyType
] trait and [Container
] trait work together
to create structures with optional members and provides an api
convert between the two.
```rust use empty_type::{EmptyType, Container};
struct Data { key: String }
struct OptionalData {
key: Option
impl EmptyType for Data { type Container = OptionalData; }
impl Container for OptionalData { type Value = Data;
fn try_open(&mut self) -> Result<Self::Value, Box<dyn std::error::Error>> {
Ok(Data {
key: self.key.open()
})
}
}
// This allows conversion between the two types
fn main() { let mut empty = Data::new_empty(); empty.key = Some(String::new()); // should be the default value let resolved = empty.resolve(); } ```
The behavior above is tedious and complicated. The proc_macro
[EmptyType
] creates
the optional data structures for you with the feature derive
enabled
```rust use empty_type::EmptyType;
struct Data { key: String }
fn main() { let mut empty = Data::new_empty(); empty.key = Some(String::new()); // should be the default value let resolved = empty.resolve(); } ```
Serde support is provided by the feature flag serde
and a helper function [deserialize_empty
]
is provided to deserialize empty values
```rust
struct Data { value: String }
const JSON: &str = r#" { "value": "data" } "#;
fn usewithdeserializer() { let mut de = serdejson::Deserializer::fromstr(JSON); let emptyvalue = deserializeempty::(&mut de).unwrap();
let full_value = empty_value.resolve();
assert_eq!(full_value.value.as_str(), "data");
}
fn useimplicitdeserialization() { let value: emptytype::Empty = serdejson::fromstr(JSON).unwrap(); let fullvalue = value.resolve();
assert_eq!(full_value.value.as_str(), "data");
}
```
Container is automatically implemented for [Option<T>
] and bool
. This allows
container unwraps to propagate up through containers.
A special container types [Fallible
] and [Optional
] provide small variations to
the way that types are opened.
Optional is a wrapper around source types that are initially option. Optional
roughly represents Some(Option<T>)
. Opening this container always results in an [Option
]
This is distinct from Option<Option<T>>
as it's impossible for the wrapping option
to be None
with the semantics described.
```rust
use emptytype::Optional;
struct Data {
optionaldata: Option
struct MaybeData { optional_data: Optional
Fallible is similar to Optional except it requires that the underlying type implement [Default
].
The semantics of fallible are to always return the default value of the underlying [Container
].
Another important distinction is that Fallible will swallow serde Deserialize errors. Any error in deserialization will result in the default type being emitted.
_