musli/storage/mod.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
//! Efficient binary storage encoding for Müsli.
//!
//! The storage encoding is partially upgrade safe:
//!
//! * ✔ Can tolerate missing fields if they are annotated with
//! `#[musli(default)]`.
//! * ✗ Cannot skip over extra unrecognized fields.
//!
//! This means that it's suitable as a storage format, since the data model only
//! evolves in one place. But unsuitable as a wire format since it cannot allow
//! clients to upgrade independent of each other.
//!
//! See [`wire`] or [`descriptive`] for formats which are upgrade stable.
//!
//! [`descriptive`]: crate::descriptive
//! [`wire`]: crate::wire
//!
//! ```
//! use musli::{Encode, Decode};
//!
//! #[derive(Debug, PartialEq, Encode, Decode)]
//! struct Version1 {
//! name: String,
//! }
//!
//! #[derive(Debug, PartialEq, Encode, Decode)]
//! struct Version2 {
//! name: String,
//! #[musli(default)]
//! age: Option<u32>,
//! }
//!
//! let version2 = musli::storage::to_vec(&Version2 {
//! name: String::from("Aristotle"),
//! age: Some(61),
//! })?;
//!
//! assert!(musli::storage::decode::<_, Version1>(version2.as_slice()).is_err());
//!
//! let version1 = musli::storage::to_vec(&Version1 {
//! name: String::from("Aristotle"),
//! })?;
//!
//! let version2: Version2 = musli::storage::decode(version1.as_slice())?;
//!
//! assert_eq!(version2, Version2 {
//! name: String::from("Aristotle"),
//! age: None,
//! });
//! # Ok::<_, musli::storage::Error>(())
//! ```
//!
//! <br>
//!
//! ## Configuring
//!
//! To tweak the behavior of the storage format you can use the [`Encoding`]
//! type:
//!
//! ```
//! use musli::{Encode, Decode};
//! use musli::mode::Binary;
//! use musli::options::{self, Options, Integer};
//! use musli::storage::Encoding;
//!
//! const OPTIONS: Options = options::new().with_integer(Integer::Fixed).build();
//! const CONFIG: Encoding<OPTIONS> = Encoding::new().with_options();
//!
//! #[derive(Debug, PartialEq, Encode, Decode)]
//! struct Person<'a> {
//! name: &'a str,
//! age: u32,
//! }
//!
//! let mut out = Vec::new();
//!
//! let expected = Person {
//! name: "Aristotle",
//! age: 61,
//! };
//!
//! CONFIG.encode(&mut out, &expected)?;
//! let actual = CONFIG.decode(&out[..])?;
//!
//! assert_eq!(expected, actual);
//! # Ok::<_, musli::storage::Error>(())
//! ```
#![cfg(any(
feature = "storage",
feature = "wire",
feature = "descriptive",
feature = "value"
))]
#![cfg_attr(doc_cfg, doc(cfg(feature = "storage")))]
pub(crate) mod de;
pub(crate) mod en;
mod encoding;
mod error;
#[cfg(feature = "test")]
#[cfg_attr(doc_cfg, doc(cfg(feature = "test")))]
#[doc(hidden)]
pub mod test;
/// Convenient result alias for use with `musli::storage`.
pub type Result<T, E = Error> = core::result::Result<T, E>;
#[cfg(feature = "alloc")]
#[cfg_attr(doc_cfg, doc(cfg(feature = "alloc")))]
#[doc(inline)]
pub use self::encoding::to_vec;
#[cfg(feature = "std")]
#[cfg_attr(doc_cfg, doc(cfg(feature = "std")))]
#[doc(inline)]
pub use self::encoding::to_writer;
#[doc(inline)]
pub use self::encoding::{decode, encode, from_slice, to_fixed_bytes, Encoding, DEFAULT, OPTIONS};
#[doc(inline)]
pub use self::error::Error;