feat: reset password strength

This commit is contained in:
Max Hohlfeld 2024-08-19 20:43:47 +02:00
parent e3c1aab6e2
commit 0fd46e184b
4 changed files with 115 additions and 69 deletions

View File

@ -7,14 +7,12 @@ use lettre::{
};
use serde::Deserialize;
use sqlx::PgPool;
use zxcvbn::{
feedback::{Suggestion, Warning},
zxcvbn, Score,
};
use zxcvbn::{zxcvbn, Score};
use crate::{
auth,
auth::{self},
models::{PasswordReset, User},
utils::password_help,
};
#[derive(Deserialize)]
@ -106,53 +104,21 @@ Viele Grüße"##, user.name, reset_url))
let entropy = zxcvbn(form.password.as_ref().unwrap(), &user_inputs);
if entropy.score() < Score::Three {
let feedback = entropy.feedback().unwrap();
let message = password_help::generate_for_entropy(&entropy);
let warning = match feedback.warning() {
Some(Warning::ThisIsATop10Password) => {
"Das Passwort ist eins der 10 meist genutzten."
}
Some(Warning::ThisIsATop100Password) => {
"Das Passwort ist eins der 100 meist genutzten."
}
Some(Warning::ThisIsACommonPassword) => "Das ist ein zu übliches Password.",
Some(Warning::DatesAreOftenEasyToGuess) => {
"Datumsangaben lassen sich meist leicht erraten."
}
Some(Warning::RecentYearsAreEasyToGuess) => {
"Vergangene Jahreszahlen lassen sich leicht erraten."
}
Some(Warning::AWordByItselfIsEasyToGuess) => "abc",
Some(Warning::RepeatsLikeAaaAreEasyToGuess) => "abc",
Some(Warning::SequencesLikeAbcAreEasyToGuess) => "abc",
Some(Warning::StraightRowsOfKeysAreEasyToGuess) => "abc",
Some(Warning::ShortKeyboardPatternsAreEasyToGuess) => "abc",
Some(Warning::ThisIsSimilarToACommonlyUsedPassword) => "abc",
Some(Warning::CommonNamesAndSurnamesAreEasyToGuess) => "abc",
Some(Warning::NamesAndSurnamesByThemselvesAreEasyToGuess) => "abc",
Some(Warning::RepeatsLikeAbcAbcAreOnlySlightlyHarderToGuess) => "abc",
_ => "",
};
let suggestion = feedback
.suggestions()
.iter()
.map(|s| match s {
Suggestion::AvoidSequences => "abc",
Suggestion::AvoidRecentYears => "abc",
_ => "abc",
})
.collect::<Vec<&str>>()
.join("</br>");
return HttpResponse::BadRequest().body(format!("Passwort zu schwach.</br>Warnung: {warning}</br>Vorschlag: {suggestion}"));
return HttpResponse::BadRequest().body(message);
}
if form.dry.is_some_and(|b| b) {
return HttpResponse::NoContent().finish();
if entropy.score() == Score::Three {
return HttpResponse::Ok()
.body("<div id=\"password-strength\" class=\"mb-3 help content is-success\">Sicheres Passwort.</div>");
} else {
return HttpResponse::Ok()
.body("<div id=\"password-strength\" class=\"mb-3 help content is-success\">Sehr sicheres Passwort.</div>");
}
}
if form.password.as_ref().unwrap() != form.passwordretyped.as_ref().unwrap() {
return HttpResponse::BadRequest().body("Passwörter stimmen nicht überein!");
}

View File

@ -1,2 +1,3 @@
pub mod email;
pub mod manage_commands;
pub mod password_help;

View File

@ -0,0 +1,74 @@
use zxcvbn::{feedback::{Suggestion, Warning}, Entropy};
pub fn generate_for_entropy(entropy: &Entropy) -> String {
let feedback = entropy.feedback().unwrap();
let warning = match feedback.warning() {
Some(Warning::StraightRowsOfKeysAreEasyToGuess) => {
"Gerade Linien von Tasten auf der Tastatur sind leicht zu erraten."
}
Some(Warning::ShortKeyboardPatternsAreEasyToGuess) => {
"Kurze Tastaturmuster sind leicht zu erraten."
}
Some(Warning::RepeatsLikeAaaAreEasyToGuess) => {
"Sich wiederholende Zeichen wie 'aaa' sind leicht zu erraten."
}
Some(Warning::RepeatsLikeAbcAbcAreOnlySlightlyHarderToGuess) => {
"Sich wiederholende Zeichenmuster wie 'abcabcabc' sind leicht zu erraten."
}
Some(Warning::ThisIsATop10Password) => "Dies ist ein sehr häufig verwendetes Passwort.",
Some(Warning::ThisIsATop100Password) => "Dies ist ein häufig verwendetes Passwort.",
Some(Warning::ThisIsACommonPassword) => "Dies ist ein oft verwendetes Passwort.",
Some(Warning::ThisIsSimilarToACommonlyUsedPassword) => {
"Dieses Passwort weist Ähnlichkeit zu anderen, oft verwendeten Passwörtern auf."
}
Some(Warning::SequencesLikeAbcAreEasyToGuess) => {
"Häufige Zeichenfolgen wie 'abc' oder '1234' sind leicht zu erraten."
}
Some(Warning::RecentYearsAreEasyToGuess) => {
"Die jüngsten Jahreszahlen sind leicht zu erraten."
}
Some(Warning::AWordByItselfIsEasyToGuess) => "Einzelne Wörter sind leicht zu erraten.",
Some(Warning::DatesAreOftenEasyToGuess) => "Ein Datum ist leicht zu erraten.",
Some(Warning::NamesAndSurnamesByThemselvesAreEasyToGuess) => {
"Einzelne Namen oder Nachnamen sind leicht zu erraten."
}
Some(Warning::CommonNamesAndSurnamesAreEasyToGuess) => {
"Vornamen und Nachnamen sind leicht zu erraten."
}
_ => "Passwort ist zu schwach.",
};
let vorschlag_text = if feedback.suggestions().len() > 1 {
"Vorschläge"
} else {
"Vorschlag"
};
let suggestion = feedback
.suggestions()
.iter()
.map(|s| {
let inner = match s {
Suggestion::UseAFewWordsAvoidCommonPhrases => "Mehrere Wörter verwenden, aber allgemeine Phrasen vermeiden.",
Suggestion::NoNeedForSymbolsDigitsOrUppercaseLetters => "Es ist möglich, starke Passwörter zu erstellen, ohne Symbole, Zahlen oder Großbuchstaben zu verwenden.",
Suggestion::AddAnotherWordOrTwo => "Weitere Wörter, die weniger häufig vorkommen, hinzufügen.",
Suggestion::CapitalizationDoesntHelpVeryMuch => "Nicht nur den ersten Buchstaben groß schreiben.",
Suggestion::AllUppercaseIsAlmostAsEasyToGuessAsAllLowercase => "Einige, aber nicht alle Buchstaben groß schreiben.",
Suggestion::ReversedWordsArentMuchHarderToGuess => "Umgekehrte Schreibweise von gebräuchlichen Wörtern vermeiden.",
Suggestion::PredictableSubstitutionsDontHelpVeryMuch => "Vorhersehbare Buchstabenersetzungen wie '@' für 'a' vermeiden.",
Suggestion::UseALongerKeyboardPatternWithMoreTurns => "Längere Tastaturmuster in unterschiedlicher Tipprichtung verwenden.",
Suggestion::AvoidRepeatedWordsAndCharacters => "Wort- und Zeichenwiederholungen vermeiden.",
Suggestion::AvoidSequences => "Häufige Zeichenfolgen vermeiden.",
Suggestion::AvoidRecentYears => "Die jüngsten Jahreszahlen vermeiden.",
Suggestion::AvoidYearsThatAreAssociatedWithYou => "Jahre, die mit persönlichen Daten in Verbindung gebracht werden können, vermeiden.",
Suggestion::AvoidDatesAndYearsThatAreAssociatedWithYou => "Daten, die mit persönlichen Daten in Verbindung gebracht werden können, vermeiden.",
};
format!("<li>{inner}</li>")
})
.collect::<Vec<String>>()
.join("");
format!("<div id=\"password-strength\" class=\"mb-3 help content is-danger\"><p>{warning}</p><p>{vorschlag_text}:<ul>{suggestion}</ul></p></div>")
}

View File

@ -2,33 +2,38 @@
{% block body %}
<section class="section">
<div class="container">
<h1 class="title">Brass - Passwort zurücksetzen</h1>
<form class="box" hx-post="/reset-password" hx-params="not dry" hx-target-400="#error-message" hx-on:change="document.getElementById('error-message').innerHTML = ''">
<input type="hidden" name="token" value="{{ token }}"/>
<div class="container">
<h1 class="title">Brass - Passwort zurücksetzen</h1>
<form class="box" hx-post="/reset-password" hx-params="not dry" hx-target-400="#error-message-retype"
hx-on:keyup="document.getElementById('error-message-retype').innerHTML = ''">
<input type="hidden" name="token" value="{{ token }}" />
<input type="hidden" name="dry" value="true"/>
<input type="hidden" name="dry" value="true" />
<div class="field">
<label class="label" for="password">neues Passwort:</label>
<div class="control">
<input class="input" hx-post="/reset-password?dry=true" hx-params="*" hx-trigger="keyup changed delay:1s" hx-target-400="#error-message" placeholder="**********" name="password" type="password" required>
</div>
</div>
<div class="field">
<label class="label" for="password">neues Passwort:</label>
<div class="control">
<input class="input" hx-post="/reset-password?dry=true" hx-params="*" hx-trigger="keyup changed delay:1s"
hx-target="#password-strength" hx-target-400="#password-strength" placeholder="**********" name="password" type="password" required
hx-swap="outerHTML" hx-on:keyup="document.getElementById('password-strength').innerHTML = ''">
</div>
</div>
<div class="field">
<label class="label" for="passwordretyped">neues Passwort wiederholen:</label>
<div class="control">
<input class="input" placeholder="**********" name="passwordretyped" type="password" max=256 required>
</div>
</div>
<div id="password-strength" class="mb-3 help content"></div>
<div id="error-message" class="mb-3 help is-danger"></div>
<div class="field">
<label class="label" for="passwordretyped">neues Passwort wiederholen:</label>
<div class="control">
<input class="input" placeholder="**********" name="passwordretyped" type="password" max=256 required>
</div>
</div>
<div class="level">
<input class="button is-primary level-left" type="submit" value="Passwort zurücksetzen" />
</div>
</form>
</div>
<div id="error-message-retype" class="mb-3 help is-danger"></div>
<div class="level">
<input class="button is-primary level-left" type="submit" value="Passwort zurücksetzen" />
</div>
</form>
</div>
</section>
{% endblock %}