use crate::cookies::set_cookie; use anyhow::{anyhow, Result}; use lldap_model::*; use yew::callback::Callback; use yew::format::Json; use yew::services::fetch::{Credentials, FetchOptions, FetchService, FetchTask, Request, Response}; #[derive(Default)] pub struct HostService {} fn get_default_options() -> FetchOptions { FetchOptions { credentials: Some(Credentials::SameOrigin), ..FetchOptions::default() } } fn get_claims_from_jwt(jwt: &str) -> Result { use jwt::*; let token = Token::::parse_unverified(jwt)?; Ok(token.claims().clone()) } fn create_handler( callback: Callback>, handler: F, ) -> Callback>> where F: Fn(http::StatusCode, Resp) -> Result + 'static, CallbackResult: 'static, { Callback::once(move |response: Response>| { let (meta, maybe_data) = response.into_parts(); let message = maybe_data .map_err(|e| anyhow!("Could not reach server: {}", e)) .and_then(|data| handler(meta.status, data)); callback.emit(message) }) } impl HostService { pub fn list_users( request: ListUsersRequest, callback: Callback>>, ) -> Result { let url = "/api/users"; let request = Request::post(url) .header("Content-Type", "application/json") .body(Json(&request))?; let handler = create_handler(callback, |status, data: String| { if status.is_success() { serde_json::from_str(&data).map_err(|e| anyhow!("Could not parse response: {}", e)) } else { Err(anyhow!("[{}]: {}", status, data)) } }); FetchService::fetch_with_options(request, get_default_options(), handler) } pub fn login_start( request: login::ClientLoginStartRequest, callback: Callback>, ) -> Result { let url = "/auth/opaque/login/start"; let request = Request::post(url) .header("Content-Type", "application/json") .body(Json(&request))?; let handler = create_handler(callback, |status, data: String| { if status.is_success() { serde_json::from_str(&data).map_err(|e| anyhow!("Could not parse response: {}", e)) } else { Err(anyhow!( "Could not start authentication: [{}]: {}", status, data )) } }); FetchService::fetch_with_options(request, get_default_options(), handler) } pub fn login_finish( request: login::ClientLoginFinishRequest, callback: Callback>, ) -> Result { let url = "/auth/opaque/login/finish"; let request = Request::post(url) .header("Content-Type", "application/json") .body(Json(&request))?; let handler = create_handler(callback, |status, data: String| { if status.is_success() { get_claims_from_jwt(&data) .map_err(|e| anyhow!("Could not parse response: {}", e)) .and_then(|jwt_claims| { set_cookie("user_id", &jwt_claims.user, &jwt_claims.exp) .map(|_| jwt_claims.user.clone()) .map_err(|e| anyhow!("Error clearing cookie: {}", e)) }) } else { Err(anyhow!( "Could not finish authentication: [{}]: {}", status, data )) } }); FetchService::fetch_with_options(request, get_default_options(), handler) } pub fn logout(callback: Callback>) -> Result { let url = "/auth/logout"; let request = Request::post(url).body(yew::format::Nothing)?; let handler = create_handler(callback, |status, data: String| { if status.is_success() { Ok(()) } else { Err(anyhow!("Could not logout: [{}]: {}", status, data)) } }); FetchService::fetch_with_options(request, get_default_options(), handler) } pub fn create_user( request: CreateUserRequest, callback: Callback>, ) -> Result { let url = "/api/users/create"; let request = Request::post(url) .header("Content-Type", "application/json") .body(Json(&request))?; let handler = create_handler(callback, |status, data: String| { if status.is_success() { Ok(()) } else { Err(anyhow!("Could not create a user: [{}]: {}", status, data)) } }); FetchService::fetch_with_options(request, get_default_options(), handler) } }