musli_core/lib.rs
1//! [<img alt="github" src="https://img.shields.io/badge/github-udoprog/musli-8da0cb?style=for-the-badge&logo=github" height="20">](https://github.com/udoprog/musli)
2//! [<img alt="crates.io" src="https://img.shields.io/crates/v/musli-core.svg?style=for-the-badge&color=fc8d62&logo=rust" height="20">](https://crates.io/crates/musli-core)
3//! [<img alt="docs.rs" src="https://img.shields.io/badge/docs.rs-musli--core-66c2a5?style=for-the-badge&logoColor=white&logo=" height="20">](https://docs.rs/musli-core)
4//!
5//! Core traits for [Müsli].
6//!
7//! [Müsli]: https://docs.rs/musli
8
9#![deny(missing_docs)]
10#![no_std]
11#![cfg_attr(doc_cfg, feature(doc_cfg))]
12
13#[cfg(feature = "alloc")]
14extern crate alloc as rust_alloc;
15
16#[cfg(feature = "std")]
17extern crate std;
18
19pub mod alloc;
20
21mod context;
22#[doc(inline)]
23pub use self::context::Context;
24
25pub mod de;
26#[doc(inline)]
27pub use self::de::{Decode, Decoder};
28
29pub mod en;
30#[doc(inline)]
31pub use self::en::{Encode, Encoder};
32
33pub mod hint;
34pub mod mode;
35pub mod no_std;
36
37mod expecting;
38mod impls;
39mod internal;
40mod never;
41
42/// This is an attribute macro that must be used when implementing a
43/// [`Encoder`].
44///
45/// It is required to use because a [`Encoder`] implementation might introduce
46/// new associated types in the future, and this [not yet supported] on a
47/// language level in Rust. So this attribute macro polyfills any missing types
48/// automatically.
49///
50/// Note that using derives directly from `musli_core` requires you to use the
51/// `#[musli_core::encoder(crate = musli_core)]` attribute.
52///
53/// [not yet supported]: https://rust-lang.github.io/rfcs/2532-associated-type-defaults.html
54///
55/// # Examples
56///
57/// ```
58/// use std::fmt;
59///
60/// use musli_core::Context;
61/// use musli_core::en::{Encoder, Encode};
62///
63/// struct MyEncoder<'a, C: ?Sized> {
64/// value: &'a mut Option<u32>,
65/// cx: &'a C,
66/// }
67///
68/// #[musli_core::encoder(crate = musli_core)]
69/// impl<C: ?Sized + Context> Encoder for MyEncoder<'_, C> {
70/// type Cx = C;
71/// type Ok = ();
72///
73/// fn cx(&self) -> &C {
74/// self.cx
75/// }
76///
77/// fn expecting(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
78/// write!(f, "32-bit unsigned integers")
79/// }
80///
81/// fn encode<T>(self, value: T) -> Result<Self::Ok, C::Error>
82/// where
83/// T: Encode<Self::Mode>,
84/// {
85/// value.encode(self.cx, self)
86/// }
87///
88/// fn encode_u32(self, value: u32) -> Result<(), Self::Error> {
89/// *self.value = Some(value);
90/// Ok(())
91/// }
92/// }
93/// ```
94#[doc(inline)]
95pub use musli_macros::encoder;
96
97/// This is an attribute macro that must be used when implementing a
98/// [`Decoder`].
99///
100/// It is required to use because a [`Decoder`] implementation might introduce
101/// new associated types in the future, and this is [not yet supported] on a
102/// language level in Rust. So this attribute macro polyfills any missing types
103/// automatically.
104///
105/// Note that using derives directly from `musli_core` requires you to use the
106/// `#[musli_core::decoder(crate = musli_core)]` attribute.
107///
108/// [not yet supported]: https://rust-lang.github.io/rfcs/2532-associated-type-defaults.html
109///
110/// # Examples
111///
112/// ```
113/// use std::fmt;
114///
115/// use musli_core::Context;
116/// use musli_core::de::{Decoder, Decode};
117///
118/// struct MyDecoder<'a, C: ?Sized> {
119/// cx: &'a C,
120/// }
121///
122/// #[musli_core::decoder(crate = musli_core)]
123/// impl<'de, C: ?Sized + Context> Decoder<'de> for MyDecoder<'_, C> {
124/// type Cx = C;
125///
126/// fn cx(&self) -> &C {
127/// self.cx
128/// }
129///
130/// fn expecting(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
131/// write!(f, "32-bit unsigned integers")
132/// }
133///
134/// fn decode_u32(self) -> Result<u32, Self::Error> {
135/// Ok(42)
136/// }
137/// }
138/// ```
139#[doc(inline)]
140pub use musli_macros::decoder;
141
142/// This is an attribute macro that must be used when implementing a
143/// [`Visitor`].
144///
145/// It is required to use because a [`Visitor`] implementation might introduce
146/// new associated types in the future, and this is [not yet supported] on a
147/// language level in Rust. So this attribute macro polyfills any missing types
148/// automatically.
149///
150/// Note that using derives directly from `musli_core` requires you to use the
151/// `#[musli_core::visitor(crate = musli_core)]` attribute.
152///
153/// [not yet supported]: https://rust-lang.github.io/rfcs/2532-associated-type-defaults.html
154/// [`Visitor`]: crate::de::Visitor
155///
156/// # Examples
157///
158/// ```
159/// use std::fmt;
160///
161/// use musli_core::Context;
162/// use musli_core::de::Visitor;
163///
164/// struct AnyVisitor;
165///
166/// #[musli_core::visitor(crate = musli_core)]
167/// impl<'de, C: ?Sized + Context> Visitor<'de, C> for AnyVisitor {
168/// type Ok = ();
169///
170/// #[inline]
171/// fn expecting(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
172/// write!(
173/// f,
174/// "value that can be decoded into dynamic container"
175/// )
176/// }
177/// }
178/// ```
179#[doc(inline)]
180pub use musli_macros::visitor;
181
182/// Internal implementation details of musli.
183///
184/// Using these directly is not supported.
185#[doc(hidden)]
186pub mod __priv {
187 use crate::context::Context;
188 use crate::de::{Decoder, EntryDecoder};
189
190 pub use ::core::fmt;
191 pub use ::core::option::Option;
192 pub use ::core::result::Result;
193
194 pub use crate::never::Never;
195
196 #[inline(always)]
197 pub fn default<T>() -> T
198 where
199 T: ::core::default::Default,
200 {
201 ::core::default::Default::default()
202 }
203
204 /// Note that this returns `true` if skipping was unsupported.
205 #[inline(always)]
206 pub fn skip<'de, D>(decoder: D) -> Result<bool, D::Error>
207 where
208 D: Decoder<'de>,
209 {
210 Ok(decoder.try_skip()?.is_unsupported())
211 }
212
213 /// Note that this returns `true` if skipping was unsupported.
214 #[inline(always)]
215 pub fn skip_field<'de, D>(decoder: D) -> Result<bool, <D::Cx as Context>::Error>
216 where
217 D: EntryDecoder<'de>,
218 {
219 skip(decoder.decode_value()?)
220 }
221
222 pub use Option::{None, Some};
223 pub use Result::{Err, Ok};
224}