mirror of
https://github.com/bitwarden/browser
synced 2025-12-10 13:23:34 +00:00
* refactor: update clientType enum * refactor: update deviceType filename * refactor: update encryptedExportType filename * refactor: update encryptionType filename * refactor: update eventType filename * refactor: update fieldType filename * refactor: update fileUploadType filename * refactor: update hashPurpose filename * refactor: update htmlStorageLocation filename * refactor: update kdfType filename * refactor: update keySuffixOptions filename * refactor: update linkedIdType filename * refactor: update logLevelType filename * refactor: update nativeMessagingVersion filename * refactor: update notificationType filename * refactor: update productType filename * refactor: update secureNoteType filename * refactor: update stateVersion filename * refactor: update storageLocation filename * refactor: update themeType filename * refactor: update uriMatchType filename * fix: update kdfType classes missed in initial pass, refs AC-1266 * fix: missing import update for device-type * refactor: add barrel file for enums and update pathed import statements, refs AC-1266 * fix: incorrect import statements for web, refs AC-1266 * fix: missed import statement updates (browser), refs AC-1266 * fix: missed import statement changes (cli), refs AC-1266 * fix: missed import statement changes (desktop), refs AC-1266 * fix: prettier, refs AC-1266 * refactor: (libs) update relative paths to use barrel file, refs AC-1266 * fix: missed find/replace import statements for SecureNoteType, refs AC-1266 * refactor: apply .enum suffix to enums folder and modify leftover relative paths, refs AC-1266 * fix: find/replace errors for native-messaging-version, refs AC-1266
221 lines
6.9 KiB
TypeScript
221 lines
6.9 KiB
TypeScript
import { Directive, OnDestroy, OnInit } from "@angular/core";
|
|
import { Subject, takeUntil } from "rxjs";
|
|
|
|
import { CryptoService } from "@bitwarden/common/abstractions/crypto.service";
|
|
import { I18nService } from "@bitwarden/common/abstractions/i18n.service";
|
|
import { MessagingService } from "@bitwarden/common/abstractions/messaging.service";
|
|
import { PlatformUtilsService } from "@bitwarden/common/abstractions/platformUtils.service";
|
|
import { StateService } from "@bitwarden/common/abstractions/state.service";
|
|
import { PolicyService } from "@bitwarden/common/admin-console/abstractions/policy/policy.service.abstraction";
|
|
import { MasterPasswordPolicyOptions } from "@bitwarden/common/admin-console/models/domain/master-password-policy-options";
|
|
import { KdfConfig } from "@bitwarden/common/auth/models/domain/kdf-config";
|
|
import { KdfType } from "@bitwarden/common/enums";
|
|
import { Utils } from "@bitwarden/common/misc/utils";
|
|
import { EncString } from "@bitwarden/common/models/domain/enc-string";
|
|
import { SymmetricCryptoKey } from "@bitwarden/common/models/domain/symmetric-crypto-key";
|
|
import { PasswordGenerationServiceAbstraction } from "@bitwarden/common/tools/generator/password";
|
|
|
|
import { PasswordColorText } from "../../shared/components/password-strength/password-strength.component";
|
|
|
|
@Directive()
|
|
export class ChangePasswordComponent implements OnInit, OnDestroy {
|
|
masterPassword: string;
|
|
masterPasswordRetype: string;
|
|
formPromise: Promise<any>;
|
|
enforcedPolicyOptions: MasterPasswordPolicyOptions;
|
|
passwordStrengthResult: any;
|
|
color: string;
|
|
text: string;
|
|
leakedPassword: boolean;
|
|
minimumLength = Utils.minimumPasswordLength;
|
|
|
|
protected email: string;
|
|
protected kdf: KdfType;
|
|
protected kdfConfig: KdfConfig;
|
|
|
|
protected destroy$ = new Subject<void>();
|
|
|
|
constructor(
|
|
protected i18nService: I18nService,
|
|
protected cryptoService: CryptoService,
|
|
protected messagingService: MessagingService,
|
|
protected passwordGenerationService: PasswordGenerationServiceAbstraction,
|
|
protected platformUtilsService: PlatformUtilsService,
|
|
protected policyService: PolicyService,
|
|
protected stateService: StateService
|
|
) {}
|
|
|
|
async ngOnInit() {
|
|
this.email = await this.stateService.getEmail();
|
|
this.policyService
|
|
.masterPasswordPolicyOptions$()
|
|
.pipe(takeUntil(this.destroy$))
|
|
.subscribe(
|
|
(enforcedPasswordPolicyOptions) =>
|
|
(this.enforcedPolicyOptions ??= enforcedPasswordPolicyOptions)
|
|
);
|
|
}
|
|
|
|
ngOnDestroy(): void {
|
|
this.destroy$.next();
|
|
this.destroy$.complete();
|
|
}
|
|
|
|
async submit() {
|
|
if (!(await this.strongPassword())) {
|
|
return;
|
|
}
|
|
|
|
if (!(await this.setupSubmitActions())) {
|
|
return;
|
|
}
|
|
|
|
const email = await this.stateService.getEmail();
|
|
if (this.kdf == null) {
|
|
this.kdf = await this.stateService.getKdfType();
|
|
}
|
|
if (this.kdfConfig == null) {
|
|
this.kdfConfig = await this.stateService.getKdfConfig();
|
|
}
|
|
const key = await this.cryptoService.makeKey(
|
|
this.masterPassword,
|
|
email.trim().toLowerCase(),
|
|
this.kdf,
|
|
this.kdfConfig
|
|
);
|
|
const masterPasswordHash = await this.cryptoService.hashPassword(this.masterPassword, key);
|
|
|
|
let encKey: [SymmetricCryptoKey, EncString] = null;
|
|
const existingEncKey = await this.cryptoService.getEncKey();
|
|
if (existingEncKey == null) {
|
|
encKey = await this.cryptoService.makeEncKey(key);
|
|
} else {
|
|
encKey = await this.cryptoService.remakeEncKey(key);
|
|
}
|
|
|
|
await this.performSubmitActions(masterPasswordHash, key, encKey);
|
|
}
|
|
|
|
async setupSubmitActions(): Promise<boolean> {
|
|
// Override in sub-class
|
|
// Can be used for additional validation and/or other processes the should occur before changing passwords
|
|
return true;
|
|
}
|
|
|
|
async performSubmitActions(
|
|
masterPasswordHash: string,
|
|
key: SymmetricCryptoKey,
|
|
encKey: [SymmetricCryptoKey, EncString]
|
|
) {
|
|
// Override in sub-class
|
|
}
|
|
|
|
async strongPassword(): Promise<boolean> {
|
|
if (this.masterPassword == null || this.masterPassword === "") {
|
|
this.platformUtilsService.showToast(
|
|
"error",
|
|
this.i18nService.t("errorOccurred"),
|
|
this.i18nService.t("masterPasswordRequired")
|
|
);
|
|
return false;
|
|
}
|
|
if (this.masterPassword.length < this.minimumLength) {
|
|
this.platformUtilsService.showToast(
|
|
"error",
|
|
this.i18nService.t("errorOccurred"),
|
|
this.i18nService.t("masterPasswordMinimumlength", this.minimumLength)
|
|
);
|
|
return false;
|
|
}
|
|
if (this.masterPassword !== this.masterPasswordRetype) {
|
|
this.platformUtilsService.showToast(
|
|
"error",
|
|
this.i18nService.t("errorOccurred"),
|
|
this.i18nService.t("masterPassDoesntMatch")
|
|
);
|
|
return false;
|
|
}
|
|
|
|
const strengthResult = this.passwordStrengthResult;
|
|
|
|
if (
|
|
this.enforcedPolicyOptions != null &&
|
|
!this.policyService.evaluateMasterPassword(
|
|
strengthResult.score,
|
|
this.masterPassword,
|
|
this.enforcedPolicyOptions
|
|
)
|
|
) {
|
|
this.platformUtilsService.showToast(
|
|
"error",
|
|
this.i18nService.t("errorOccurred"),
|
|
this.i18nService.t("masterPasswordPolicyRequirementsNotMet")
|
|
);
|
|
return false;
|
|
}
|
|
|
|
const weakPassword = strengthResult != null && strengthResult.score < 3;
|
|
|
|
if (weakPassword && this.leakedPassword) {
|
|
const result = await this.platformUtilsService.showDialog(
|
|
this.i18nService.t("weakAndBreachedMasterPasswordDesc"),
|
|
this.i18nService.t("weakAndExposedMasterPassword"),
|
|
this.i18nService.t("yes"),
|
|
this.i18nService.t("no"),
|
|
"warning"
|
|
);
|
|
if (!result) {
|
|
return false;
|
|
}
|
|
} else {
|
|
if (weakPassword) {
|
|
const result = await this.platformUtilsService.showDialog(
|
|
this.i18nService.t("weakMasterPasswordDesc"),
|
|
this.i18nService.t("weakMasterPassword"),
|
|
this.i18nService.t("yes"),
|
|
this.i18nService.t("no"),
|
|
"warning"
|
|
);
|
|
if (!result) {
|
|
return false;
|
|
}
|
|
}
|
|
if (this.leakedPassword) {
|
|
const result = await this.platformUtilsService.showDialog(
|
|
this.i18nService.t("exposedMasterPasswordDesc"),
|
|
this.i18nService.t("exposedMasterPassword"),
|
|
this.i18nService.t("yes"),
|
|
this.i18nService.t("no"),
|
|
"warning"
|
|
);
|
|
if (!result) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
async logOut() {
|
|
const confirmed = await this.platformUtilsService.showDialog(
|
|
this.i18nService.t("logOutConfirmation"),
|
|
this.i18nService.t("logOut"),
|
|
this.i18nService.t("logOut"),
|
|
this.i18nService.t("cancel")
|
|
);
|
|
if (confirmed) {
|
|
this.messagingService.send("logout");
|
|
}
|
|
}
|
|
|
|
getStrengthResult(result: any) {
|
|
this.passwordStrengthResult = result;
|
|
}
|
|
|
|
getPasswordScoreText(event: PasswordColorText) {
|
|
this.color = event.color;
|
|
this.text = event.text;
|
|
}
|
|
}
|