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
|
use std::time::SystemTime;
use axum::{
body::Body,
extract::{Path, State},
http::{HeaderMap, StatusCode, Uri},
response::{IntoResponse, Response},
routing::get,
Router,
};
use axum_extra::extract::CookieJar;
use maud::html;
use tap::Pipe;
use crate::server::{store::Store, Handoffs, Nevermind};
use super::{account_auth, render_html, ApiState};
pub fn bind(app: Router<ApiState>) -> Router<ApiState> {
app.route("/", get(get_panel))
.route("/scopes/:token", get(get_scopes))
.route("/handoff", get(get_handoff))
}
#[axum::debug_handler(state = ApiState)]
async fn get_panel(jar: CookieJar, State(store): State<Store>) -> Result<Response, Response> {
Ok(account_auth(&jar, &store)
.await
.prompt_login()?
.pipe(render_normal_panel))
}
fn render_normal_panel(name: String) -> Response {
render_html(
html!(title { "dissociate" }),
html! {
p { "currently logged in as " (name) }
a href="/logout" { button { "log out" } }
},
)
}
#[axum::debug_handler(state = ApiState)]
async fn get_scopes(
Path(token): Path<String>,
State(store): State<Store>,
) -> Result<Response, Response> {
let error_response = || {
Response::builder()
.status(StatusCode::BAD_REQUEST)
.body(Body::empty())
.unwrap()
};
let mut parts = Vec::new();
let (name, expires) = store.check_token(&token).await.ok_or_else(error_response)?;
parts.push(
expires
.duration_since(SystemTime::UNIX_EPOCH)
.unwrap()
.as_secs()
.to_string(),
);
{
let account = store.get_account(&name).await.ok_or_else(error_response)?;
parts.extend(account.scopes.iter().cloned());
};
let body = parts.join("\n");
Ok(Response::builder()
.status(StatusCode::OK)
.header("Content-Type", "text/plain")
.body(body.into())
.unwrap())
}
#[axum::debug_handler(state = ApiState)]
async fn get_handoff(
jar: CookieJar,
State(Handoffs(handoffs)): State<Handoffs>,
State(store): State<Store>,
headers: HeaderMap,
) -> Response {
let Some(origin_header) = headers.get("Origin") else {
return (StatusCode::BAD_REQUEST, "Missing Origin header").into_response();
};
let Some(origin) = origin_header
.to_str()
.ok()
.and_then(|origin| origin.parse::<Uri>().ok())
.and_then(|origin| origin.host().map(ToString::to_string))
else {
return (StatusCode::BAD_REQUEST, "Malformed Origin header").into_response();
};
if !handoffs.contains(&origin) {
return (StatusCode::FORBIDDEN, "Origin not registered for handoff").into_response();
}
let builder = Response::builder()
.header("Access-Control-Allow-Credentials", "true")
.header("Access-Control-Allow-Origin", origin_header);
if let Some(token) = jar.get("dissociate-token") {
if let Some((_, expires)) = store.check_token(token.value()).await {
let expires_in = expires
.duration_since(SystemTime::now())
.unwrap_or_default()
.as_secs();
return builder
.status(StatusCode::OK)
.body(Body::from(format!(
r#"{{"token":"{}","expiresIn":{}}}"#,
token.value().to_string(),
expires_in,
)))
.unwrap();
}
}
builder
.status(StatusCode::UNAUTHORIZED)
.body(Body::empty())
.unwrap()
}
|