use crate::cookies::set_cookie; use anyhow::{anyhow, Context, Result}; use graphql_client::GraphQLQuery; use lldap_auth::{login, registration, JWTClaims}; 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 .context("Could not reach server") .and_then(|data| handler(meta.status, data)); callback.emit(message) }) } struct RequestBody(T); impl<'a, R> From<&'a R> for RequestBody> where R: serde::ser::Serialize, { fn from(request: &'a R) -> Self { Self(Json(request)) } } impl From for RequestBody { fn from(request: yew::format::Nothing) -> Self { Self(request) } } fn call_server( url: &str, request: RB, callback: Callback>, error_message: &'static str, parse_response: F, ) -> Result where F: Fn(String) -> Result + 'static, CallbackResult: 'static, RB: Into>, Req: Into, { let request = { // If the request type is empty (if the size is 0), it's a get. if std::mem::size_of::() == 0 { Request::get(url) } else { Request::post(url) } } .header("Content-Type", "application/json") .body(request.into().0)?; let handler = create_handler(callback, move |status: http::StatusCode, data: String| { if status.is_success() { parse_response(data) } else { Err(anyhow!("{}[{}]: {}", error_message, status, data)) } }); FetchService::fetch_with_options(request, get_default_options(), handler) } fn call_server_json_with_error_message( url: &str, request: RB, callback: Callback>, error_message: &'static str, ) -> Result where CallbackResult: serde::de::DeserializeOwned + 'static, RB: Into>, Req: Into, { call_server(url, request, callback, error_message, |data: String| { serde_json::from_str(&data).context("Could not parse response") }) } fn call_server_empty_response_with_error_message( url: &str, request: RB, callback: Callback>, error_message: &'static str, ) -> Result where RB: Into>, Req: Into, { call_server( url, request, callback, error_message, |_data: String| Ok(()), ) } impl HostService { pub fn graphql_query( variables: QueryType::Variables, callback: Callback>, error_message: &'static str, ) -> Result where QueryType: GraphQLQuery + 'static, { let unwrap_graphql_response = |graphql_client::Response { data, errors }| { data.ok_or_else(|| { anyhow!( "Errors: [{}]", errors .unwrap_or_default() .iter() .map(ToString::to_string) .collect::>() .join(", ") ) }) }; let parse_graphql_response = move |data: String| { serde_json::from_str(&data) .context("Could not parse response") .and_then(unwrap_graphql_response) }; let request_body = QueryType::build_query(variables); call_server( "/api/graphql", &request_body, callback, error_message, parse_graphql_response, ) } pub fn login_start( request: login::ClientLoginStartRequest, callback: Callback>>, ) -> Result { call_server_json_with_error_message( "/auth/opaque/login/start", &request, callback, "Could not start authentication: ", ) } pub fn login_finish( request: login::ClientLoginFinishRequest, callback: Callback>, ) -> Result { let set_cookies = |jwt_claims: JWTClaims| { let is_admin = jwt_claims.groups.contains("lldap_admin"); set_cookie("user_id", &jwt_claims.user, &jwt_claims.exp) .map(|_| set_cookie("is_admin", &is_admin.to_string(), &jwt_claims.exp)) .map(|_| (jwt_claims.user.clone(), is_admin)) .context("Error clearing cookie") }; let parse_token = move |data: String| { get_claims_from_jwt(&data) .context("Could not parse response") .and_then(set_cookies) }; call_server( "/auth/opaque/login/finish", &request, callback, "Could not finish authentication", parse_token, ) } pub fn register_start( request: registration::ClientRegistrationStartRequest, callback: Callback>>, ) -> Result { call_server_json_with_error_message( "/auth/opaque/register/start", &request, callback, "Could not start registration: ", ) } pub fn register_finish( request: registration::ClientRegistrationFinishRequest, callback: Callback>, ) -> Result { call_server_empty_response_with_error_message( "/auth/opaque/register/finish", &request, callback, "Could not finish registration", ) } pub fn logout(callback: Callback>) -> Result { call_server_empty_response_with_error_message( "/auth/logout", yew::format::Nothing, callback, "Could not logout", ) } }