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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
//! Contains the `ElementStatus` enum and its associated traits. It indicates
//! the status resulting from operations on the specific individual elements.

use crate::distance_matrix::error::Error;
use phf::phf_map;
use serde::{Deserialize, Deserializer, Serialize, Serializer};

// -----------------------------------------------------------------------------

/// The `status` fields within the response object contain the status of the
/// request, and may contain useful debugging information. The Distance Matrix
/// API returns a top-level status field, with information about the request in
/// general, as well as a status field for each element field, with information
/// about that particular origin-destination pairing.

#[derive(Clone, Debug, Default, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[repr(u8)]
pub enum ElementStatus {
    /// Indicates the requested route is too long and cannot be processed.
    MaxRouteLengthExceeded = 0,
    /// Indicates that the origin and/or destination of this pairing could not
    /// be geocoded.
    NotFound = 1,
    /// Indicates the response contains a valid result.
    #[default] Ok = 2,
    /// Indicates no route could be found between the origin and destination.
    ZeroResults = 3,
} // struct

// -----------------------------------------------------------------------------

impl<'de> Deserialize<'de> for ElementStatus {
    /// Manual implementation of `Deserialize` for `serde`. This will take
    /// advantage of the `phf`-powered `TryFrom` implementation for this type.
    fn deserialize<D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
        let string = String::deserialize(deserializer)?;
        match ElementStatus::try_from(string.as_str()) {
            Ok(variant) => Ok(variant),
            Err(error) => Err(serde::de::Error::custom(error.to_string()))
        } // match
    } // fn
} // impl

// -----------------------------------------------------------------------------

impl Serialize for ElementStatus {
    /// Manual implementation of `Serialize` for `serde`.
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where S: Serializer {
        serializer.serialize_str(std::convert::Into::<&str>::into(self))
    } // fn
} // impl

// -----------------------------------------------------------------------------

impl std::convert::From<&ElementStatus> for &str {
    /// Converts a `ElementStatus` enum to a `String` that contains a [element
    /// status](https://developers.google.com/maps/documentation/distance-matrix/intro#element-level-status-codes)
    /// code.
    fn from(element_status: &ElementStatus) -> Self {
        match element_status {
            ElementStatus::MaxRouteLengthExceeded => "MAX_ROUTE_LENGTH_EXCEEDED",
            ElementStatus::NotFound => "NOT_FOUND",
            ElementStatus::Ok => "OK",
            ElementStatus::ZeroResults => "ZERO_RESULTS",
        } // match
    } // fn
} // impl

// -----------------------------------------------------------------------------

impl std::fmt::Display for ElementStatus {
    /// Converts a `ElementStatus` enum to a `String` that contains a [element
    /// status](https://developers.google.com/maps/documentation/distance-matrix/intro#element-level-status-codes)
    /// code.
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        write!(f, "{}", std::convert::Into::<&str>::into(self))
    } // fmt
} // impl

// -----------------------------------------------------------------------------

impl std::convert::From<&ElementStatus> for String {
    /// Converts a `ElementStatus` enum to a `String` that contains a [element
    /// status](https://developers.google.com/maps/documentation/distance-matrix/intro#element-level-status-codes)
    /// code.
    fn from(element_status: &ElementStatus) -> Self {
        std::convert::Into::<&str>::into(element_status).to_string()
    } // fn
} // impl

// -----------------------------------------------------------------------------

static ELEMENT_STATUSES_BY_CODE: phf::Map<&'static str, ElementStatus> = phf_map! {
    "MAX_ROUTE_LENGTH_EXCEEDED" => ElementStatus::MaxRouteLengthExceeded,
    "NOT_FOUND" => ElementStatus::NotFound,
    "OK" => ElementStatus::Ok,
    "ZERO_RESULTS" => ElementStatus::ZeroResults,
};

// -----------------------------------------------------------------------------

impl std::convert::TryFrom<&str> for ElementStatus {
    // Error definitions are contained in the
    // `google_maps\src\distance_matrix\error.rs` module.
    type Error = crate::distance_matrix::error::Error;
    /// Gets a `ElementStatus` enum from a `String` that contains a valid
    /// [element
    /// status](https://developers.google.com/maps/documentation/distance-matrix/intro#element-level-status-codes)
    /// code.
    fn try_from(element_status_code: &str) -> Result<Self, Self::Error> {
        ELEMENT_STATUSES_BY_CODE
            .get(element_status_code)
            .cloned()
            .ok_or_else(|| Error::InvalidElementStatusCode(element_status_code.to_string()))
    } // fn
} // impl

// -----------------------------------------------------------------------------

impl std::str::FromStr for ElementStatus {
    // Error definitions are contained in the
    // `google_maps\src\distance_matrix\error.rs` module.
    type Err = crate::distance_matrix::error::Error;
    /// Gets a `ElementStatus` enum from a `String` that contains a valid
    /// [element
    /// status](https://developers.google.com/maps/documentation/distance-matrix/intro#element-level-status-codes)
    /// code.
    fn from_str(element_status_code: &str) -> Result<Self, Self::Err> {
        ELEMENT_STATUSES_BY_CODE
            .get(element_status_code)
            .cloned()
            .ok_or_else(|| Error::InvalidElementStatusCode(element_status_code.to_string()))
    } // fn
} // impl

// -----------------------------------------------------------------------------

impl ElementStatus {
    /// Formats a `ElementStatus` enum into a string that is presentable to the
    /// end user.
    pub fn display(&self) -> &str {
        match self {
            ElementStatus::MaxRouteLengthExceeded => "Maximum Route Length Exceeded",
            ElementStatus::NotFound => "Not Found",
            ElementStatus::Ok => "OK",
            ElementStatus::ZeroResults => "Zero Results",
        } // match
    } // fn
} // impl