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
//! Utility functions to handle URL manipulation.

pub use url;

use std::{borrow::Borrow, string::ToString};
use url::form_urlencoded;

/// Creates an endpoint with a query
pub fn construct_ep<E, Q>(ep: E, query: Option<Q>) -> String
where
    E: Into<String>,
    Q: AsRef<str>,
{
    let mut ep = ep.into();
    if let Some(query) = query {
        append_query(&mut ep, query);
    }
    ep
}

/// Appends a query to an endpoint
pub fn append_query<Q>(ep: &mut String, query: Q)
where
    Q: AsRef<str>,
{
    ep.push('?');
    ep.push_str(query.as_ref());
}

/// Encodes `key` and `val` as urlencoded values.
pub fn encoded_pair<K, V>(key: K, val: V) -> String
where
    K: AsRef<str>,
    V: ToString,
{
    form_urlencoded::Serializer::new(String::new())
        .append_pair(key.as_ref(), &val.to_string())
        .finish()
}

/// Encodes multiple values for the same key
pub fn encoded_vec_pairs<K, I>(pairs: impl IntoIterator<Item = (K, I)>) -> String
where
    K: AsRef<str>,
    I: IntoIterator,
    I::Item: AsRef<str>,
{
    let mut serializer = form_urlencoded::Serializer::new(String::new());
    pairs.into_iter().for_each(|(key, vals)| {
        let key = key.as_ref();
        vals.into_iter().for_each(|val| {
            serializer.append_pair(key, val.as_ref());
        });
    });

    serializer.finish()
}

/// Encodes an iterator of key:value pairs as urlencoded values.
pub fn encoded_pairs<I, K, V>(iter: I) -> String
where
    I: IntoIterator,
    I::Item: Borrow<(K, V)>,
    K: AsRef<str>,
    V: AsRef<str>,
{
    iter.into_iter()
        .fold(
            form_urlencoded::Serializer::new(String::new()),
            |mut acc, v| {
                let (k, v) = v.borrow();
                let k = k.as_ref();
                let v = v.as_ref();
                if v.is_empty() {
                    acc.append_key_only(k);
                } else {
                    acc.append_pair(k, v);
                }
                acc
            },
        )
        .finish()
}

#[cfg(test)]
mod tests {
    use super::{append_query, construct_ep, encoded_pair, encoded_pairs, encoded_vec_pairs};

    #[test]
    fn appends_query() {
        let mut ep = "http://somewebsite.xxx".to_owned();
        let query = "lang=en";
        let want = "http://somewebsite.xxx?lang=en";
        append_query(&mut ep, query);
        assert_eq!(ep, want);
    }

    #[test]
    fn constructs_endpoint() {
        let ep = "http://somewebsite.xxx";
        let query = "lang=en,id=55555";
        let want = "http://somewebsite.xxx?lang=en,id=55555";
        assert_eq!(construct_ep(ep, None::<&str>), ep);
        assert_eq!(construct_ep(ep, Some(query)), want);
    }

    #[test]
    fn encodes_pair() {
        let key = "lang";
        let val = "en&";
        let want = "lang=en%26";
        assert_eq!(encoded_pair(key, val), want);
    }

    #[test]
    fn encodes_pairs() {
        let pairs = [("lang", "en&"), ("id", "1337"), ("country", "xxx")];
        let want = "lang=en%26&id=1337&country=xxx";
        assert_eq!(encoded_pairs(pairs), want);
    }

    #[test]
    fn encodes_vec_pairs() {
        let pairs = [
            ("lang", vec!["en", "pl&"]),
            ("id", vec!["1337"]),
            ("country", vec!["xxx", "yyy", "zzz"]),
        ];
        let want = "lang=en&lang=pl%26&id=1337&country=xxx&country=yyy&country=zzz";
        assert_eq!(encoded_vec_pairs(pairs), want);
    }
}