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
use cookie::{Cookie, Key, PrivateJar};
use serde::de::DeserializeOwned;
use serde::ser::Serialize;
use serde_json;
use std::fmt;
use app::AppState;
use context::Context;
use error::Error;
pub struct SessionStorage {
secret_key: Key,
}
impl fmt::Debug for SessionStorage {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_struct("SessionStorage").finish()
}
}
impl SessionStorage {
pub fn builder() -> Builder {
Builder { secret_key: None }
}
pub(crate) fn secret_key(&self) -> &Key {
&self.secret_key
}
}
pub struct Builder {
secret_key: Option<Key>,
}
impl fmt::Debug for Builder {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_struct("Builder").finish()
}
}
impl Builder {
pub fn secret_key<K>(&mut self, master_key: K) -> &mut Self
where
K: AsRef<[u8]>,
{
self.secret_key = Some(Key::from_master(master_key.as_ref()));
self
}
pub fn finish(&mut self) -> SessionStorage {
SessionStorage {
secret_key: self.secret_key.take().unwrap_or_else(|| Key::generate()),
}
}
}
#[derive(Debug)]
pub struct Session<'a> {
context: &'a Context,
}
impl<'a> Session<'a> {
#[allow(missing_docs)]
pub fn get<T>(&self, key: &str) -> Result<Option<T>, Error>
where
T: DeserializeOwned,
{
match self.with_private(|jar| jar.get(key))? {
Some(cookie) => serde_json::from_str(cookie.value())
.map(Some)
.map_err(Error::bad_request),
None => Ok(None),
}
}
#[allow(missing_docs)]
pub fn set<T>(&self, key: &str, value: T) -> Result<(), Error>
where
T: Serialize,
{
let value = serde_json::to_string(&value).map_err(Error::internal_server_error)?;
let cookie = Cookie::new(key.to_owned(), value);
self.with_private(|mut jar| jar.add(cookie))?;
Ok(())
}
#[allow(missing_docs)]
pub fn remove(&self, key: &str) -> Result<(), Error> {
self.with_private(|mut jar| jar.remove(Cookie::named(key.to_owned())))
}
fn with_private<R>(&self, f: impl FnOnce(PrivateJar) -> R) -> Result<R, Error> {
AppState::with(|state| Ok(self.context.cookies()?.with_private(state.session().secret_key(), f)))
}
}
#[allow(missing_docs)]
pub trait ContextSessionExt {
fn session(&self) -> Session;
}
impl ContextSessionExt for Context {
fn session(&self) -> Session {
Session { context: self }
}
}