forked from seanmonstar/reqwest
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcookie.rs
137 lines (113 loc) · 3.85 KB
/
cookie.rs
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
//! HTTP Cookies
use std::convert::TryInto;
use crate::header;
use std::fmt;
use std::time::SystemTime;
/// A single HTTP cookie.
pub struct Cookie<'a>(cookie_crate::Cookie<'a>);
impl<'a> Cookie<'a> {
fn parse(value: &'a crate::header::HeaderValue) -> Result<Cookie<'a>, CookieParseError> {
std::str::from_utf8(value.as_bytes())
.map_err(cookie_crate::ParseError::from)
.and_then(cookie_crate::Cookie::parse)
.map_err(CookieParseError)
.map(Cookie)
}
/// The name of the cookie.
pub fn name(&self) -> &str {
self.0.name()
}
/// The value of the cookie.
pub fn value(&self) -> &str {
self.0.value()
}
/// Returns true if the 'HttpOnly' directive is enabled.
pub fn http_only(&self) -> bool {
self.0.http_only().unwrap_or(false)
}
/// Returns true if the 'Secure' directive is enabled.
pub fn secure(&self) -> bool {
self.0.secure().unwrap_or(false)
}
/// Returns true if 'SameSite' directive is 'Lax'.
pub fn same_site_lax(&self) -> bool {
self.0.same_site() == Some(cookie_crate::SameSite::Lax)
}
/// Returns true if 'SameSite' directive is 'Strict'.
pub fn same_site_strict(&self) -> bool {
self.0.same_site() == Some(cookie_crate::SameSite::Strict)
}
/// Returns the path directive of the cookie, if set.
pub fn path(&self) -> Option<&str> {
self.0.path()
}
/// Returns the domain directive of the cookie, if set.
pub fn domain(&self) -> Option<&str> {
self.0.domain()
}
/// Get the Max-Age information.
pub fn max_age(&self) -> Option<std::time::Duration> {
self.0.max_age().map(|d| {
d.try_into()
.expect("time::Duration into std::time::Duration")
})
}
/// The cookie expiration time.
pub fn expires(&self) -> Option<SystemTime> {
self.0.expires().map(SystemTime::from)
}
}
impl<'a> fmt::Debug for Cookie<'a> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
self.0.fmt(f)
}
}
pub(crate) fn extract_response_cookie_headers<'a>(
headers: &'a hyper::HeaderMap,
) -> impl Iterator<Item = &'a str> + 'a {
headers
.get_all(header::SET_COOKIE)
.iter()
.filter_map(|value| std::str::from_utf8(value.as_bytes()).ok())
}
pub(crate) fn extract_response_cookies<'a>(
headers: &'a hyper::HeaderMap,
) -> impl Iterator<Item = Result<Cookie<'a>, CookieParseError>> + 'a {
headers
.get_all(header::SET_COOKIE)
.iter()
.map(|value| Cookie::parse(value))
}
// FIXME: this type is superfluous?
/// A persistent cookie store that provides session support.
pub struct CookieStore(pub(crate) Box<dyn CookieStorage>);
use std::any::Any;
/// Actions for a persistent cookie store providing session supprt.
pub trait CookieStorage: fmt::Debug + Send + Sync {
/// Store a set of Set-Cookie header values recevied from `url`
fn store_cookie_headers(&mut self, cookie_headers: Vec<&str>, url: &url::Url);
/// Get any Cookie values in the store for `url`
fn get_cookie_headers(&self, url: &url::Url) -> Vec<String>;
/// To facilitate downcasting
fn as_any(&self) -> &dyn Any;
/// To facilitate downcasting
fn as_any_mut(&mut self) -> &mut dyn Any;
}
impl<'a> fmt::Debug for CookieStore {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
self.0.fmt(f)
}
}
/// Error representing a parse failure of a 'Set-Cookie' header.
pub(crate) struct CookieParseError(cookie_crate::ParseError);
impl<'a> fmt::Debug for CookieParseError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
self.0.fmt(f)
}
}
impl<'a> fmt::Display for CookieParseError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
self.0.fmt(f)
}
}
impl std::error::Error for CookieParseError {}