mirror of
https://github.com/hay-kot/homebox.git
synced 2025-08-03 08:10:28 +00:00
implement password score UI and functions
This commit is contained in:
parent
31b34241e0
commit
5ecf0ec706
4 changed files with 152 additions and 0 deletions
40
frontend/components/global/PasswordScore.vue
Normal file
40
frontend/components/global/PasswordScore.vue
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
<template>
|
||||||
|
<div class="py-4">
|
||||||
|
<p class="text-sm">Password Strength: {{ message }}</p>
|
||||||
|
<progress
|
||||||
|
class="progress w-full progress-bar"
|
||||||
|
:value="score"
|
||||||
|
max="100"
|
||||||
|
:class="{
|
||||||
|
'progress-success': score > 50,
|
||||||
|
'progress-warning': score > 25 && score < 50,
|
||||||
|
'progress-error': score < 25,
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const props = defineProps({
|
||||||
|
password: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
valid: {
|
||||||
|
type: Boolean,
|
||||||
|
required: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const emits = defineEmits(["update:valid"]);
|
||||||
|
|
||||||
|
const { password } = toRefs(props);
|
||||||
|
|
||||||
|
const { score, message, isValid } = usePasswordScore(password);
|
||||||
|
|
||||||
|
watchEffect(() => {
|
||||||
|
emits("update:valid", isValid.value);
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped></style>
|
37
frontend/composables/use-password-score.ts
Normal file
37
frontend/composables/use-password-score.ts
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
import type { ComputedRef, Ref } from "vue";
|
||||||
|
import { scorePassword } from "~~/lib/passwords";
|
||||||
|
|
||||||
|
export interface PasswordScore {
|
||||||
|
score: ComputedRef<number>;
|
||||||
|
message: ComputedRef<string>;
|
||||||
|
isValid: ComputedRef<boolean>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function usePasswordScore(pw: Ref<string>, min = 30): PasswordScore {
|
||||||
|
const score = computed(() => {
|
||||||
|
return scorePassword(pw.value) || 0;
|
||||||
|
});
|
||||||
|
|
||||||
|
const message = computed(() => {
|
||||||
|
if (score.value < 20) {
|
||||||
|
return "Very weak";
|
||||||
|
} else if (score.value < 40) {
|
||||||
|
return "Weak";
|
||||||
|
} else if (score.value < 60) {
|
||||||
|
return "Good";
|
||||||
|
} else if (score.value < 80) {
|
||||||
|
return "Strong";
|
||||||
|
}
|
||||||
|
return "Very strong";
|
||||||
|
});
|
||||||
|
|
||||||
|
const isValid = computed(() => {
|
||||||
|
return score.value >= min;
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
score,
|
||||||
|
isValid,
|
||||||
|
message,
|
||||||
|
};
|
||||||
|
}
|
30
frontend/lib/passwords/index.test.ts
Normal file
30
frontend/lib/passwords/index.test.ts
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
import { describe, test, expect } from "vitest";
|
||||||
|
import { scorePassword } from ".";
|
||||||
|
|
||||||
|
describe("scorePassword tests", () => {
|
||||||
|
test("flagged words should return negative number", () => {
|
||||||
|
const flaggedWords = ["password", "homebox", "admin", "qwerty", "login"];
|
||||||
|
|
||||||
|
for (const word of flaggedWords) {
|
||||||
|
expect(scorePassword(word)).toBe(0);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return 0 for empty string", () => {
|
||||||
|
expect(scorePassword("")).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return 0 for strings less than 6", () => {
|
||||||
|
expect(scorePassword("12345")).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return positive number for long string", () => {
|
||||||
|
const result = expect(scorePassword("123456"));
|
||||||
|
result.toBeGreaterThan(0);
|
||||||
|
result.toBeLessThan(31);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return max number for long string with all variations", () => {
|
||||||
|
expect(scorePassword("3bYWcfYOwqxljqeOmQXTLlBwkrH6HV")).toBe(100);
|
||||||
|
});
|
||||||
|
});
|
45
frontend/lib/passwords/index.ts
Normal file
45
frontend/lib/passwords/index.ts
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
const flaggedWords = ["password", "homebox", "admin", "qwerty", "login"];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* scorePassword returns a score for a given password between 0 and 100.
|
||||||
|
* if a password contains a flagged word, it returns 0.
|
||||||
|
* @param pass
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
export function scorePassword(pass: string): number {
|
||||||
|
let score = 0;
|
||||||
|
if (!pass) return score;
|
||||||
|
|
||||||
|
if (pass.length < 6) return score;
|
||||||
|
|
||||||
|
// Check for flagged words
|
||||||
|
for (const word of flaggedWords) {
|
||||||
|
if (pass.toLowerCase().includes(word)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// award every unique letter until 5 repetitions
|
||||||
|
const letters: { [key: string]: number } = {};
|
||||||
|
|
||||||
|
for (let i = 0; i < pass.length; i++) {
|
||||||
|
letters[pass[i]] = (letters[pass[i]] || 0) + 1;
|
||||||
|
score += 5.0 / letters[pass[i]];
|
||||||
|
}
|
||||||
|
|
||||||
|
// bonus points for mixing it up
|
||||||
|
const variations: { [key: string]: boolean } = {
|
||||||
|
digits: /\d/.test(pass),
|
||||||
|
lower: /[a-z]/.test(pass),
|
||||||
|
upper: /[A-Z]/.test(pass),
|
||||||
|
nonWords: /\W/.test(pass),
|
||||||
|
};
|
||||||
|
|
||||||
|
let variationCount = 0;
|
||||||
|
for (const check in variations) {
|
||||||
|
variationCount += variations[check] === true ? 1 : 0;
|
||||||
|
}
|
||||||
|
score += (variationCount - 1) * 10;
|
||||||
|
|
||||||
|
return Math.max(Math.min(score, 100), 0);
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue