🗝
summary refs log tree commit diff
path: root/src/server/account.rs
blob: b2c294ffb4f3f6f0bf67705dbe471ddc4c715f98 (plain) (blame)
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
use std::time::SystemTime;

use axum::{
    body::Body,
    extract::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", 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(jar: CookieJar, State(store): State<Store>) -> Result<Response, Response> {
    let name = account_auth(&jar, &store).await.prompt_login()?;
    let account = store.get_account(&name).await.prompt_logout()?;
    let body = account.scopes.join(" ");

    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()
}