mirror of
https://github.com/nitnelave/lldap.git
synced 2023-04-12 14:25:13 +00:00
b2cfc0ed03
This is a massive change to all the components, since the interface changed. There are opportunities to greatly simplify some components by turning them into functional_components, but this work has tried to stay as mechanical as possible.
314 lines
12 KiB
Rust
314 lines
12 KiB
Rust
use crate::{
|
|
components::router::{AppRoute, Link},
|
|
infra::{
|
|
api::HostService,
|
|
common_component::{CommonComponent, CommonComponentParts},
|
|
},
|
|
};
|
|
use anyhow::{anyhow, bail, Result};
|
|
use gloo_console::error;
|
|
use lldap_auth::*;
|
|
use validator_derive::Validate;
|
|
use yew::prelude::*;
|
|
use yew_form::Form;
|
|
use yew_form_derive::Model;
|
|
use yew_router::{prelude::History, scope_ext::RouterScopeExt};
|
|
|
|
#[derive(PartialEq, Eq, Default)]
|
|
enum OpaqueData {
|
|
#[default]
|
|
None,
|
|
Login(opaque::client::login::ClientLogin),
|
|
Registration(opaque::client::registration::ClientRegistration),
|
|
}
|
|
|
|
impl OpaqueData {
|
|
fn take(&mut self) -> Self {
|
|
std::mem::take(self)
|
|
}
|
|
}
|
|
|
|
/// The fields of the form, with the constraints.
|
|
#[derive(Model, Validate, PartialEq, Eq, Clone, Default)]
|
|
pub struct FormModel {
|
|
#[validate(custom(
|
|
function = "empty_or_long",
|
|
message = "Password should be longer than 8 characters"
|
|
))]
|
|
old_password: String,
|
|
#[validate(length(min = 8, message = "Invalid password. Min length: 8"))]
|
|
password: String,
|
|
#[validate(must_match(other = "password", message = "Passwords must match"))]
|
|
confirm_password: String,
|
|
}
|
|
|
|
fn empty_or_long(value: &str) -> Result<(), validator::ValidationError> {
|
|
if value.is_empty() || value.len() >= 8 {
|
|
Ok(())
|
|
} else {
|
|
Err(validator::ValidationError::new(""))
|
|
}
|
|
}
|
|
|
|
pub struct ChangePasswordForm {
|
|
common: CommonComponentParts<Self>,
|
|
form: Form<FormModel>,
|
|
opaque_data: OpaqueData,
|
|
}
|
|
|
|
#[derive(Clone, PartialEq, Eq, Properties)]
|
|
pub struct Props {
|
|
pub username: String,
|
|
pub is_admin: bool,
|
|
}
|
|
|
|
pub enum Msg {
|
|
FormUpdate,
|
|
Submit,
|
|
AuthenticationStartResponse(Result<Box<login::ServerLoginStartResponse>>),
|
|
SubmitNewPassword,
|
|
RegistrationStartResponse(Result<Box<registration::ServerRegistrationStartResponse>>),
|
|
RegistrationFinishResponse(Result<()>),
|
|
}
|
|
|
|
impl CommonComponent<ChangePasswordForm> for ChangePasswordForm {
|
|
fn handle_msg(
|
|
&mut self,
|
|
ctx: &Context<Self>,
|
|
msg: <Self as Component>::Message,
|
|
) -> Result<bool> {
|
|
use anyhow::Context;
|
|
match msg {
|
|
Msg::FormUpdate => Ok(true),
|
|
Msg::Submit => {
|
|
if !self.form.validate() {
|
|
bail!("Check the form for errors");
|
|
}
|
|
if ctx.props().is_admin {
|
|
self.handle_msg(ctx, Msg::SubmitNewPassword)
|
|
} else {
|
|
let old_password = self.form.model().old_password;
|
|
if old_password.is_empty() {
|
|
bail!("Current password should not be empty");
|
|
}
|
|
let mut rng = rand::rngs::OsRng;
|
|
let login_start_request =
|
|
opaque::client::login::start_login(&old_password, &mut rng)
|
|
.context("Could not initialize login")?;
|
|
self.opaque_data = OpaqueData::Login(login_start_request.state);
|
|
let req = login::ClientLoginStartRequest {
|
|
username: ctx.props().username.clone(),
|
|
login_start_request: login_start_request.message,
|
|
};
|
|
self.common.call_backend(
|
|
ctx,
|
|
HostService::login_start(req),
|
|
Msg::AuthenticationStartResponse,
|
|
);
|
|
Ok(true)
|
|
}
|
|
}
|
|
Msg::AuthenticationStartResponse(res) => {
|
|
let res = res.context("Could not initiate login")?;
|
|
match self.opaque_data.take() {
|
|
OpaqueData::Login(l) => {
|
|
opaque::client::login::finish_login(l, res.credential_response).map_err(
|
|
|e| {
|
|
// Common error, we want to print a full error to the console but only a
|
|
// simple one to the user.
|
|
error!(&format!("Invalid username or password: {}", e));
|
|
anyhow!("Invalid username or password")
|
|
},
|
|
)?;
|
|
}
|
|
_ => panic!("Unexpected data in opaque_data field"),
|
|
};
|
|
self.handle_msg(ctx, Msg::SubmitNewPassword)
|
|
}
|
|
Msg::SubmitNewPassword => {
|
|
let mut rng = rand::rngs::OsRng;
|
|
let new_password = self.form.model().password;
|
|
let registration_start_request =
|
|
opaque::client::registration::start_registration(&new_password, &mut rng)
|
|
.context("Could not initiate password change")?;
|
|
let req = registration::ClientRegistrationStartRequest {
|
|
username: ctx.props().username.clone(),
|
|
registration_start_request: registration_start_request.message,
|
|
};
|
|
self.opaque_data = OpaqueData::Registration(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")?;
|
|
match self.opaque_data.take() {
|
|
OpaqueData::Registration(registration) => {
|
|
let mut rng = rand::rngs::OsRng;
|
|
let registration_finish =
|
|
opaque::client::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,
|
|
);
|
|
}
|
|
_ => panic!("Unexpected data in opaque_data field"),
|
|
};
|
|
Ok(false)
|
|
}
|
|
Msg::RegistrationFinishResponse(response) => {
|
|
if response.is_ok() {
|
|
ctx.link().history().unwrap().push(AppRoute::UserDetails {
|
|
user_id: ctx.props().username.clone(),
|
|
});
|
|
}
|
|
response?;
|
|
Ok(true)
|
|
}
|
|
}
|
|
}
|
|
|
|
fn mut_common(&mut self) -> &mut CommonComponentParts<Self> {
|
|
&mut self.common
|
|
}
|
|
}
|
|
|
|
impl Component for ChangePasswordForm {
|
|
type Message = Msg;
|
|
type Properties = Props;
|
|
|
|
fn create(_: &Context<Self>) -> Self {
|
|
ChangePasswordForm {
|
|
common: CommonComponentParts::<Self>::create(),
|
|
form: yew_form::Form::<FormModel>::new(FormModel::default()),
|
|
opaque_data: OpaqueData::None,
|
|
}
|
|
}
|
|
|
|
fn update(&mut self, ctx: &Context<Self>, msg: Self::Message) -> bool {
|
|
CommonComponentParts::<Self>::update(self, ctx, msg)
|
|
}
|
|
|
|
fn view(&self, ctx: &Context<Self>) -> Html {
|
|
let is_admin = ctx.props().is_admin;
|
|
let link = ctx.link();
|
|
type Field = yew_form::Field<FormModel>;
|
|
html! {
|
|
<>
|
|
<div class="mb-2 mt-2">
|
|
<h5 class="fw-bold">
|
|
{"Change password"}
|
|
</h5>
|
|
</div>
|
|
{
|
|
if let Some(e) = &self.common.error {
|
|
html! {
|
|
<div class="alert alert-danger mt-3 mb-3">
|
|
{e.to_string() }
|
|
</div>
|
|
}
|
|
} else { html! {} }
|
|
}
|
|
<form
|
|
class="form">
|
|
{if !is_admin { html! {
|
|
<div class="form-group row">
|
|
<label for="old_password"
|
|
class="form-label col-sm-2 col-form-label">
|
|
{"Current password*:"}
|
|
</label>
|
|
<div class="col-sm-10">
|
|
<Field
|
|
form={&self.form}
|
|
field_name="old_password"
|
|
input_type="password"
|
|
class="form-control"
|
|
class_invalid="is-invalid has-error"
|
|
class_valid="has-success"
|
|
autocomplete="current-password"
|
|
oninput={link.callback(|_| Msg::FormUpdate)} />
|
|
<div class="invalid-feedback">
|
|
{&self.form.field_message("old_password")}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
}} else { html! {} }}
|
|
<div class="form-group row mb-3">
|
|
<label for="new_password"
|
|
class="form-label col-sm-2 col-form-label">
|
|
{"New Password"}
|
|
<span class="text-danger">{"*"}</span>
|
|
{":"}
|
|
</label>
|
|
<div class="col-sm-10">
|
|
<Field
|
|
form={&self.form}
|
|
field_name="password"
|
|
input_type="password"
|
|
class="form-control"
|
|
class_invalid="is-invalid has-error"
|
|
class_valid="has-success"
|
|
autocomplete="new-password"
|
|
oninput={link.callback(|_| Msg::FormUpdate)} />
|
|
<div class="invalid-feedback">
|
|
{&self.form.field_message("password")}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="form-group row mb-3">
|
|
<label for="confirm_password"
|
|
class="form-label col-sm-2 col-form-label">
|
|
{"Confirm Password"}
|
|
<span class="text-danger">{"*"}</span>
|
|
{":"}
|
|
</label>
|
|
<div class="col-sm-10">
|
|
<Field
|
|
form={&self.form}
|
|
field_name="confirm_password"
|
|
input_type="password"
|
|
class="form-control"
|
|
class_invalid="is-invalid has-error"
|
|
class_valid="has-success"
|
|
autocomplete="new-password"
|
|
oninput={link.callback(|_| Msg::FormUpdate)} />
|
|
<div class="invalid-feedback">
|
|
{&self.form.field_message("confirm_password")}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="form-group row justify-content-center">
|
|
<button
|
|
class="btn btn-primary col-auto col-form-label"
|
|
type="submit"
|
|
disabled={self.common.is_task_running()}
|
|
onclick={link.callback(|e: MouseEvent| {e.prevent_default(); Msg::Submit})}>
|
|
<i class="bi-save me-2"></i>
|
|
{"Save changes"}
|
|
</button>
|
|
<Link
|
|
classes="btn btn-secondary ms-2 col-auto col-form-label"
|
|
to={AppRoute::UserDetails{user_id: ctx.props().username.clone()}}>
|
|
<i class="bi-arrow-return-left me-2"></i>
|
|
{"Back"}
|
|
</Link>
|
|
</div>
|
|
</form>
|
|
</>
|
|
}
|
|
}
|
|
}
|