use crate::{ components::{ password_field::PasswordField, router::{AppRoute, Link}, }, infra::{ api::HostService, common_component::{CommonComponent, CommonComponentParts}, }, }; use anyhow::{bail, Result}; use lldap_auth::{ opaque::client::registration as opaque_registration, password_reset::ServerPasswordResetResponse, registration, }; use validator_derive::Validate; use yew::prelude::*; use yew_form::Form; use yew_form_derive::Model; use yew_router::{prelude::History, scope_ext::RouterScopeExt}; /// The fields of the form, with the constraints. #[derive(Model, Validate, PartialEq, Eq, Clone, Default)] pub struct FormModel { #[validate(length(min = 8, message = "Invalid password. Min length: 8"))] password: String, #[validate(must_match(other = "password", message = "Passwords must match"))] confirm_password: String, } pub struct ResetPasswordStep2Form { common: CommonComponentParts, form: Form, username: Option, opaque_data: Option, } #[derive(Clone, PartialEq, Eq, Properties)] pub struct Props { pub token: String, } pub enum Msg { ValidateTokenResponse(Result), FormUpdate, Submit, RegistrationStartResponse(Result>), RegistrationFinishResponse(Result<()>), } impl CommonComponent for ResetPasswordStep2Form { fn handle_msg( &mut self, ctx: &Context, msg: ::Message, ) -> Result { use anyhow::Context; match msg { Msg::ValidateTokenResponse(response) => { self.username = Some(response?.user_id); Ok(true) } Msg::FormUpdate => Ok(true), Msg::Submit => { if !self.form.validate() { bail!("Check the form for errors"); } let mut rng = rand::rngs::OsRng; let new_password = self.form.model().password; let registration_start_request = opaque_registration::start_registration(&new_password, &mut rng) .context("Could not initiate password change")?; let req = registration::ClientRegistrationStartRequest { username: self.username.clone().unwrap(), registration_start_request: registration_start_request.message, }; self.opaque_data = Some(registration_start_request.state); self.common.call_backend( ctx, HostService::register_start(req), Msg::RegistrationStartResponse, ); Ok(true) } Msg::RegistrationStartResponse(res) => { let res = res.context("Could not initiate password change")?; let registration = self.opaque_data.take().expect("Missing registration data"); let mut rng = rand::rngs::OsRng; let registration_finish = opaque_registration::finish_registration( registration, res.registration_response, &mut rng, ) .context("Error during password change")?; let req = registration::ClientRegistrationFinishRequest { server_data: res.server_data, registration_upload: registration_finish.message, }; self.common.call_backend( ctx, HostService::register_finish(req), Msg::RegistrationFinishResponse, ); Ok(false) } Msg::RegistrationFinishResponse(response) => { if response.is_ok() { ctx.link().history().unwrap().push(AppRoute::Login); } response?; Ok(true) } } } fn mut_common(&mut self) -> &mut CommonComponentParts { &mut self.common } } impl Component for ResetPasswordStep2Form { type Message = Msg; type Properties = Props; fn create(ctx: &Context) -> Self { let mut component = ResetPasswordStep2Form { common: CommonComponentParts::::create(), form: yew_form::Form::::new(FormModel::default()), opaque_data: None, username: None, }; let token = ctx.props().token.clone(); component.common.call_backend( ctx, HostService::reset_password_step2(token), Msg::ValidateTokenResponse, ); component } fn update(&mut self, ctx: &Context, msg: Self::Message) -> bool { CommonComponentParts::::update(self, ctx, msg) } fn view(&self, ctx: &Context) -> Html { let link = &ctx.link(); match (&self.username, &self.common.error) { (None, None) => { return html! { {"Validating token"} } } (None, Some(e)) => { return html! { <>
{e.to_string() }
{"Back"} } } _ => (), }; type Field = yew_form::Field; html! { <>

{"Reset your password"}

form={&self.form} field_name="password" class="form-control" class_invalid="is-invalid has-error" class_valid="has-success" oninput={link.callback(|_| Msg::FormUpdate)} />
{&self.form.field_message("password")}
{&self.form.field_message("confirm_password")}
{ if let Some(e) = &self.common.error { html! {
{e.to_string() }
} } else { html! {} } } } } }