user_management.rs 1.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647
  1. use crate::{schema, UIDCError};
  2. use microrm::prelude::*;
  3. pub fn change_auth(
  4. realm: &schema::Realm,
  5. username: &String,
  6. change_password: bool,
  7. change_totp: bool,
  8. ) -> Result<(), UIDCError> {
  9. // check that the user exists
  10. let user = realm
  11. .users
  12. .with(schema::User::Username, username)
  13. .first()
  14. .get()?
  15. .ok_or(UIDCError::Abort("no such user"))?;
  16. let user_id = user.id();
  17. let user = crate::user::User::from_schema(realm, user);
  18. if change_password {
  19. let raw_pass = rpassword::prompt_password("Enter new user password: ").unwrap();
  20. user.set_new_password(raw_pass.as_bytes())?;
  21. }
  22. if change_totp {
  23. let (new_secret, new_uri) = user.generate_totp_with_uri()?;
  24. println!("Please confirm you can generate tokens with the new secret:");
  25. qr2term::print_qr(new_uri.as_str())
  26. .map_err(|_| UIDCError::Abort("could not display QR code"))?;
  27. let new_challenge = schema::AuthChallenge {
  28. user_id,
  29. challenge_type: schema::AuthChallengeType::TOTP.into(),
  30. public: vec![],
  31. secret: new_secret.clone(),
  32. enabled: true,
  33. };
  34. loop {
  35. let digits = rpassword::prompt_password("TOTP code: ").unwrap();
  36. if new_challenge.verify_totp_challenge(digits.as_bytes())? {
  37. break;
  38. }
  39. }
  40. user.set_new_totp(new_secret.as_slice())?;
  41. }
  42. Ok(())
  43. }