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
use core::fmt;

#[cfg(not(feature = "serde-unsafe"))]
use serde::de::Unexpected;
use serde::de::{Deserializer, Error, Visitor};

use crate::FastStr;

// https://github.com/serde-rs/serde/blob/629802f2abfd1a54a6072992888fea7ca5bc209f/serde/src/private/de.rs#L56-L125
fn fast_str<'de: 'a, 'a, D>(deserializer: D) -> Result<FastStr, D::Error>
where
    D: Deserializer<'de>,
{
    struct FastStrVisitor;

    impl<'a> Visitor<'a> for FastStrVisitor {
        type Value = FastStr;

        fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
            formatter.write_str("a string")
        }

        fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
        where
            E: Error,
        {
            Ok(FastStr::new(v))
        }

        fn visit_borrowed_str<E>(self, v: &'a str) -> Result<Self::Value, E>
        where
            E: Error,
        {
            Ok(FastStr::new(v))
        }

        fn visit_string<E>(self, v: String) -> Result<Self::Value, E>
        where
            E: Error,
        {
            Ok(FastStr::from(v))
        }

        fn visit_bytes<E>(self, v: &[u8]) -> Result<Self::Value, E>
        where
            E: Error,
        {
            #[cfg(feature = "serde-unsafe")]
            {
                Ok(unsafe { FastStr::new_u8_slice_unchecked(v) })
            }
            #[cfg(not(feature = "serde-unsafe"))]
            match FastStr::new_u8_slice(v) {
                Ok(s) => Ok(s),
                Err(_) => Err(Error::invalid_value(Unexpected::Bytes(v), &self)),
            }
        }

        fn visit_borrowed_bytes<E>(self, v: &'a [u8]) -> Result<Self::Value, E>
        where
            E: Error,
        {
            #[cfg(feature = "serde-unsafe")]
            {
                Ok(unsafe { FastStr::new_u8_slice_unchecked(v) })
            }
            #[cfg(not(feature = "serde-unsafe"))]
            match FastStr::new_u8_slice(v) {
                Ok(s) => Ok(s),
                Err(_) => Err(Error::invalid_value(Unexpected::Bytes(v), &self)),
            }
        }

        fn visit_byte_buf<E>(self, v: Vec<u8>) -> Result<Self::Value, E>
        where
            E: Error,
        {
            #[cfg(not(feature = "serde-unsafe"))]
            simdutf8::basic::from_utf8(&v)
                .map_err(|_| Error::invalid_value(Unexpected::Bytes(&v), &self))?;
            // Safety: we have checked that v is valid utf-8
            Ok(unsafe { FastStr::from_vec_u8_unchecked(v) })
        }
    }

    deserializer.deserialize_string(FastStrVisitor)
}

impl serde::Serialize for FastStr {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: serde::Serializer,
    {
        self.as_str().serialize(serializer)
    }
}

impl<'de> serde::Deserialize<'de> for FastStr {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: serde::Deserializer<'de>,
    {
        fast_str(deserializer)
    }
}