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
use anyhow::Result;
use stdweb::{js, unstable::TryInto};
use thiserror::Error;
#[derive(Debug, Error)]
pub enum CookieError {
#[error("no cookie found for the given name")]
NotFound,
}
pub struct CookieService;
impl CookieService {
pub fn new() -> Self {
CookieService
}
pub fn set(&self, name: &str, value: &str) {
self.set_expiring(name, value, 365)
}
pub fn get(&self, name: &str) -> Result<String> {
let cookie_strings = js! { return document.cookie.split(';') };
let cookies: Vec<String> = cookie_strings.try_into()?;
cookies
.iter()
.filter_map(|x| {
let name_value: Vec<_> = x.splitn(2, '=').collect();
match name_value.get(0) {
None => None,
Some(c) => {
if c.trim_start() == name {
name_value.get(1).map(|x| (*x).to_owned())
} else {
None
}
}
}
})
.collect::<Vec<String>>()
.pop()
.ok_or_else(|| CookieError::NotFound.into())
}
pub fn remove(&self, name: &str) {
self.set_expiring(name, "", -1);
}
fn set_expiring(&self, name: &str, value: &str, days: i32) {
js! {
document.cookie = @{name} + "=" + (@{value} || "") +
";max-age=" + (@{days} * 24 * 60 * 60) + ";path=/";
}
}
}