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
use std::{collections::BTreeMap, sync::Arc};
use crate::{
middleware::{CookieJarManager, CookieJarManagerEndpoint},
session::{CookieConfig, Session, SessionStatus},
Endpoint, Middleware, Request,
};
pub struct CookieSession {
config: Arc<CookieConfig>,
}
impl CookieSession {
pub fn new(config: CookieConfig) -> Self {
Self {
config: Arc::new(config),
}
}
}
impl<E: Endpoint> Middleware<E> for CookieSession {
type Output = CookieJarManagerEndpoint<CookieSessionEndpoint<E>>;
fn transform(&self, ep: E) -> Self::Output {
CookieJarManager::new().transform(CookieSessionEndpoint {
inner: ep,
config: self.config.clone(),
})
}
}
pub struct CookieSessionEndpoint<E> {
inner: E,
config: Arc<CookieConfig>,
}
#[async_trait::async_trait]
impl<E: Endpoint> Endpoint for CookieSessionEndpoint<E> {
type Output = E::Output;
async fn call(&self, mut req: Request) -> Self::Output {
let cookie_jar = req.cookie().clone();
let session = self
.config
.get_cookie_value(&cookie_jar)
.and_then(|value| serde_json::from_str::<BTreeMap<String, String>>(&value).ok())
.map(Session::new)
.unwrap_or_else(Session::default);
req.extensions_mut().insert(session.clone());
let resp = self.inner.call(req).await;
match session.status() {
SessionStatus::Changed | SessionStatus::Renewed => {
self.config.set_cookie_value(
&cookie_jar,
&serde_json::to_string(&session.entries()).unwrap_or_default(),
);
}
SessionStatus::Purged => {
self.config.remove_cookie(&cookie_jar);
}
SessionStatus::Unchanged => {}
};
resp
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{
session::test_harness::{index, TestClient},
EndpointExt, Route,
};
#[tokio::test]
async fn cookie_session() {
let app = Route::new()
.at("/:action", index)
.with(CookieSession::new(CookieConfig::default()));
let mut client = TestClient::default();
client.call(&app, 0).await;
client.assert_cookies(vec![]);
client.call(&app, 1).await;
client.assert_cookies(vec![("poem-session", r#"{"a":"10","b":"20"}"#)]);
client.call(&app, 2).await;
client.assert_cookies(vec![("poem-session", r#"{"a":"10","b":"20","c":"30"}"#)]);
client.call(&app, 7).await;
client.call(&app, 6).await;
client.assert_cookies(vec![("poem-session", r#"{"a":"10","b":"20","c":"30"}"#)]);
client.call(&app, 3).await;
client.assert_cookies(vec![("poem-session", r#"{"a":"10","c":"30"}"#)]);
client.call(&app, 4).await;
client.assert_cookies(vec![("poem-session", r#"{}"#)]);
client.call(&app, 5).await;
client.assert_cookies(vec![]);
}
}