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
use std::{fmt, str::FromStr};

/// HTTP Methods.
///
/// Also contains a special method (ANY) for routes that run on all methods, which will never be the method of a request.
/// From <https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods>.
#[derive(Debug, Hash, PartialEq, Eq, Clone, Copy)]
pub enum Method {
    /// HTTP GET Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/GET)
    ///
    /// Used for retrieving data
    GET,

    /// HTTP POST Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/POST)
    ///
    /// Used for submitting data
    POST,

    /// HTTP PUT Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/PUT)
    ///
    /// Used for updating data
    PUT,

    /// HTTP DELETE Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/DELETE)
    ///
    /// Used for deleting data
    DELETE,

    /// HTTP OPTIONS Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/OPTIONS)
    ///
    /// Used for requesting information about the server
    OPTIONS,

    /// HTTP HEAD Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/HEAD)
    ///
    /// For getting the response from a GET request without the body
    HEAD,

    /// HTTP PATCH Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/PATCH)
    ///
    /// Used for applying a partial update to a resource
    PATCH,

    /// HTTP TRACE Method.
    /// [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/TRACE)
    ///
    /// Used for tracing the route of a request
    TRACE,

    /// For routes that run on all methods
    ///
    /// Will not be use in a request
    ANY,
}

impl FromStr for Method {
    type Err = ();

    /// Convert a string to a method.
    /// If the string is not a valid method or is ANY, an error will be returned.
    /// ## Examples
    /// ```rust
    /// # use std::str::FromStr;
    /// # use afire::{Method};
    /// assert!(Method::from_str("GET").unwrap() == Method::GET);
    /// assert!(Method::from_str("POST").unwrap() == Method::POST);
    /// assert!(Method::from_str("PUT").unwrap() == Method::PUT);
    /// assert!(Method::from_str("DELETE").unwrap() == Method::DELETE);
    /// assert!(Method::from_str("OPTIONS").unwrap() == Method::OPTIONS);
    /// assert!(Method::from_str("HEAD").unwrap() == Method::HEAD);
    /// assert!(Method::from_str("PATCH").unwrap() == Method::PATCH);
    /// assert!(Method::from_str("TRACE").unwrap() == Method::TRACE);
    /// assert!(Method::from_str("ANY") == Err(()));
    /// assert!(Method::from_str("foo") == Err(()));
    /// ```
    fn from_str(s: &str) -> Result<Self, Self::Err> {
        Ok(match s.to_uppercase().as_str() {
            "GET" => Method::GET,
            "POST" => Method::POST,
            "PUT" => Method::PUT,
            "DELETE" => Method::DELETE,
            "OPTIONS" => Method::OPTIONS,
            "HEAD" => Method::HEAD,
            "PATCH" => Method::PATCH,
            "TRACE" => Method::TRACE,
            _ => return Err(()),
        })
    }
}

impl fmt::Display for Method {
    /// Returns the string representation of the method.
    ///
    /// ```rust
    /// # use afire::{Method};
    /// assert_eq!("GET", Method::GET.to_string());
    /// ```
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        f.write_str(match *self {
            Method::GET => "GET",
            Method::POST => "POST",
            Method::PUT => "PUT",
            Method::DELETE => "DELETE",
            Method::OPTIONS => "OPTIONS",
            Method::HEAD => "HEAD",
            Method::PATCH => "PATCH",
            Method::TRACE => "TRACE",
            Method::ANY => "ANY",
        })
    }
}