|
@@ -156,9 +156,11 @@ impl<'l> SessionHelper<'l> {
|
|
|
{
|
|
|
"challenge":
|
|
|
format!(r#"
|
|
|
- <input type="hidden" name="challenge_type" value="{:?}" />
|
|
|
- <div class="challenge-type">{}</div>
|
|
|
- <div class="challenge-content">{}</div>
|
|
|
+ <td class="challenge-type">
|
|
|
+ <input type="hidden" name="challenge_type" value="{:?}" />
|
|
|
+ {}
|
|
|
+ </td>
|
|
|
+ <td class="challenge-content">{}</td>
|
|
|
"#,
|
|
|
to_present, ty, ch),
|
|
|
"redirect": redirect,
|
|
@@ -184,6 +186,12 @@ impl<'l> SessionHelper<'l> {
|
|
|
r#"<input name="challenge" type="password" autofocus />"#,
|
|
|
));
|
|
|
}
|
|
|
+ schema::AuthChallengeType::TOTP => {
|
|
|
+ response.set_body(do_challenge(
|
|
|
+ "Authenticator code",
|
|
|
+ r#"<input name="challenge" type="text" autofocus />"#,
|
|
|
+ ));
|
|
|
+ }
|
|
|
_ => todo!(),
|
|
|
}
|
|
|
|
|
@@ -252,6 +260,7 @@ async fn v1_login_post(mut req: Request) -> tide::Result<tide::Response> {
|
|
|
let challenge: schema::AuthChallengeType = match body.challenge_type.as_str() {
|
|
|
"Username" => ChallengeType::Username,
|
|
|
"Password" => ChallengeType::Password,
|
|
|
+ "TOTP" => ChallengeType::TOTP,
|
|
|
_ => Err(tide::Error::from_str(400, "Unknown challenge type"))?,
|
|
|
};
|
|
|
|
|
@@ -283,12 +292,18 @@ async fn v1_login_post(mut req: Request) -> tide::Result<tide::Response> {
|
|
|
} else {
|
|
|
let user = user.unwrap();
|
|
|
|
|
|
- // TODO: set list of challenges to be whatever else this user has set up
|
|
|
+ let enabled = qi.get().by(schema::AuthChallenge::User, &user.id()).all()?;
|
|
|
+ let has_totp = enabled.iter().filter(|ac| ac.challenge_type == schema::AuthChallengeType::TOTP).count() > 0;
|
|
|
+
|
|
|
+ // TODO: support more flows than just username,password[,totp]
|
|
|
let sa = schema::SessionAuthentication {
|
|
|
session: session_id,
|
|
|
realm: realm,
|
|
|
user: user.id(),
|
|
|
- challenges_left: vec![schema::AuthChallengeType::Password],
|
|
|
+ challenges_left: if has_totp {
|
|
|
+ vec![schema::AuthChallengeType::Password, schema::AuthChallengeType::TOTP]
|
|
|
+ } else {
|
|
|
+ vec![schema::AuthChallengeType::Password] },
|
|
|
};
|
|
|
let id = qi.add(&sa).unwrap();
|
|
|
auth = Some(microrm::WithID::new(sa, id));
|