#serde #const-generics #serialization #serialize-deserialize

no-std serde_default_utils

A set of simple helper functions to cut corners with serde_default

4 releases (2 breaking)

0.3.1 Nov 18, 2024
0.3.0 Oct 7, 2024
0.2.2 Feb 1, 2024
0.2.1 Jan 16, 2024
0.1.0 Jan 10, 2024

#417 in Encoding

Download history 250/week @ 2024-09-18 228/week @ 2024-09-25 346/week @ 2024-10-02 250/week @ 2024-10-09 332/week @ 2024-10-16 168/week @ 2024-10-23 131/week @ 2024-10-30 153/week @ 2024-11-06 191/week @ 2024-11-13 93/week @ 2024-11-20 202/week @ 2024-11-27 203/week @ 2024-12-04 122/week @ 2024-12-11 91/week @ 2024-12-18 160/week @ 2024-12-25 144/week @ 2025-01-01

526 downloads per month
Used in zika

MIT/Apache

17KB
118 lines

serde_default_utils

v

Overview

This is a simple set of functions to make your life easier while working with defaults in serde. Based on const generic parameters. Heavily inspired by discussions on issues about serde defaults, but mostly this one

Kudos

  • JohnTheCoolingFan posted this solution, I just made it available as crate and a macro that helps to generate another const generic function for any const generic type.
  • bytedream made a more powerful version of it, and although I still see const generic approach as more readable, I have to admit that for strings it's superior, hence - included under the feature

Example

    use serde_default_utils::*;
    use serde::{Deserialize, Serialize};

        const JSON: &str =
            r#"{
                "yes_or_no":false,
                "max":60,
                "delta":-77,
                "delimeter":"☀",
                "motto":"I matter"
            }"#;
        const INLINE_JSON: &str =
            r#"{
                "inline_motto":"I matter even more",
                "slice":["I", "matter", "even", "more"],
                "slice_u64":[9000, 8000, 10000, 7000]
            }"#;
    const EMPTY_JSON: &str = r#"{}"#;
    const MAX: u32 = 7;

    #[derive(Serialize, Deserialize, Default)]
    struct Config {
        #[serde(default = "default_bool::<true>")]
        yes_or_no: bool,
        #[serde(default = "default_i16::<-3>")]
        delta: i16,
        // you can even use consts right here
        #[serde(default = "default_u32::<MAX>")]
        max: u32,
        #[serde(default = "default_char::<'☀'>")]
        delimeter: char,
    }

    #[cfg(feature = "inline")]
    #[serde_inline_default]
    #[derive(Serialize, Deserialize, Default)]
    struct InlineConfig {
        #[serde_inline_default("You matter even more".to_string())]
        inline_motto: String,
        #[serde_inline_default(vec!["You".into(), "matter".into(), "even".into(), "more".into()])]
        slice: Vec<String>,
        #[serde_inline_default(vec![98712, 12346, 129389, 102937])]
        slice_u64: Vec<u64>,
    }

    fn main() {
        // existing json fields are not changed
        let config: Config = serde_json::from_str(JSON).unwrap();
        let s = serde_json::to_string(&config).unwrap();
        assert_eq!(r#"{"yes_or_no":false,"delta":-77,"max":60,"delimeter":"☀"}"#, &s);
        // if the field is not present - it is substituted with defaults
        let config: Config = serde_json::from_str(EMPTY_JSON).unwrap();
        let s = serde_json::to_string(&config).unwrap();
        assert_eq!(r#"{"yes_or_no":true,"delta":-3,"max":7,"delimeter":"☀"}"#, &s);
        // the default impl is just calling underlying type defaults unless you have a custom impl Default
        let config = Config::default();
        let s = serde_json::to_string(&config).unwrap();
        assert_eq!(r#"{"yes_or_no":false,"delta":0,"max":0,"delimeter":"\u0000"}"#, &s);

        // Inline
        #[cfg(feature = "inline")]
        {
        // existing json fields are not changed
        let config: InlineConfig = serde_json::from_str(INLINE_JSON).unwrap();
        let s = serde_json::to_string(&config).unwrap();
        assert_eq!(r#"{"inline_motto":"I matter even more","slice":["I","matter","even","more"],"slice_u64":[9000,8000,10000,7000]}"#, &s);
        // if the field is not present - it is substituted with defaults
        let config: InlineConfig = serde_json::from_str(EMPTY_JSON).unwrap();
        let s = serde_json::to_string(&config).unwrap();
        assert_eq!(r#"{"inline_motto":"You matter even more","slice":["You","matter","even","more"],"slice_u64":[98712,12346,129389,102937]}"#, &s);
        // the default impl is just calling underlying type defaults unless you have a custom impl Default
        let config = InlineConfig::default();
        let s = serde_json::to_string(&config).unwrap();
        assert_eq!(r#"{"inline_motto":"","slice":[],"slice_u64":[]}"#, &s);
        }
    }

License: MIT OR Apache-2.0

Dependencies

~100KB