mirror of
https://github.com/bitwarden/browser
synced 2025-12-16 00:03:56 +00:00
198 lines
7.9 KiB
TypeScript
198 lines
7.9 KiB
TypeScript
import { mock } from "jest-mock-extended";
|
|
|
|
import { ApiService } from "@bitwarden/common/abstractions/api.service";
|
|
import { AuthRequestResponse } from "@bitwarden/common/auth/models/response/auth-request.response";
|
|
import { AppIdService } from "@bitwarden/common/platform/abstractions/app-id.service";
|
|
import { CryptoService } from "@bitwarden/common/platform/abstractions/crypto.service";
|
|
import { StateService } from "@bitwarden/common/platform/abstractions/state.service";
|
|
import { Utils } from "@bitwarden/common/platform/misc/utils";
|
|
import { EncString } from "@bitwarden/common/platform/models/domain/enc-string";
|
|
import { SymmetricCryptoKey } from "@bitwarden/common/platform/models/domain/symmetric-crypto-key";
|
|
import { MasterKey, UserKey } from "@bitwarden/common/types/key";
|
|
|
|
import { AuthRequestService } from "./auth-request.service";
|
|
|
|
describe("AuthRequestService", () => {
|
|
let sut: AuthRequestService;
|
|
|
|
const appIdService = mock<AppIdService>();
|
|
const cryptoService = mock<CryptoService>();
|
|
const apiService = mock<ApiService>();
|
|
const stateService = mock<StateService>();
|
|
|
|
let mockPrivateKey: Uint8Array;
|
|
|
|
beforeEach(() => {
|
|
jest.clearAllMocks();
|
|
|
|
sut = new AuthRequestService(appIdService, cryptoService, apiService, stateService);
|
|
|
|
mockPrivateKey = new Uint8Array(64);
|
|
});
|
|
|
|
describe("approveOrDenyAuthRequest", () => {
|
|
beforeEach(() => {
|
|
cryptoService.rsaEncrypt.mockResolvedValue({
|
|
encryptedString: "ENCRYPTED_STRING",
|
|
} as EncString);
|
|
appIdService.getAppId.mockResolvedValue("APP_ID");
|
|
});
|
|
it("should throw if auth request is missing id or key", async () => {
|
|
const authRequestNoId = new AuthRequestResponse({ id: "", key: "KEY" });
|
|
const authRequestNoPublicKey = new AuthRequestResponse({ id: "123", publicKey: "" });
|
|
|
|
await expect(sut.approveOrDenyAuthRequest(true, authRequestNoId)).rejects.toThrow(
|
|
"Auth request has no id",
|
|
);
|
|
await expect(sut.approveOrDenyAuthRequest(true, authRequestNoPublicKey)).rejects.toThrow(
|
|
"Auth request has no public key",
|
|
);
|
|
});
|
|
|
|
it("should use the master key and hash if they exist", async () => {
|
|
cryptoService.getMasterKey.mockResolvedValueOnce({ encKey: new Uint8Array(64) } as MasterKey);
|
|
stateService.getKeyHash.mockResolvedValueOnce("KEY_HASH");
|
|
|
|
await sut.approveOrDenyAuthRequest(
|
|
true,
|
|
new AuthRequestResponse({ id: "123", publicKey: "KEY" }),
|
|
);
|
|
|
|
expect(cryptoService.rsaEncrypt).toHaveBeenCalledWith(new Uint8Array(64), expect.anything());
|
|
});
|
|
|
|
it("should use the user key if the master key and hash do not exist", async () => {
|
|
cryptoService.getUserKey.mockResolvedValueOnce({ key: new Uint8Array(64) } as UserKey);
|
|
|
|
await sut.approveOrDenyAuthRequest(
|
|
true,
|
|
new AuthRequestResponse({ id: "123", publicKey: "KEY" }),
|
|
);
|
|
|
|
expect(cryptoService.rsaEncrypt).toHaveBeenCalledWith(new Uint8Array(64), expect.anything());
|
|
});
|
|
});
|
|
describe("setUserKeyAfterDecryptingSharedUserKey", () => {
|
|
it("decrypts and sets user key when given valid auth request response and private key", async () => {
|
|
// Arrange
|
|
const mockAuthReqResponse = {
|
|
key: "authReqPublicKeyEncryptedUserKey",
|
|
} as AuthRequestResponse;
|
|
|
|
const mockDecryptedUserKey = {} as UserKey;
|
|
jest.spyOn(sut, "decryptPubKeyEncryptedUserKey").mockResolvedValueOnce(mockDecryptedUserKey);
|
|
|
|
cryptoService.setUserKey.mockResolvedValueOnce(undefined);
|
|
|
|
// Act
|
|
await sut.setUserKeyAfterDecryptingSharedUserKey(mockAuthReqResponse, mockPrivateKey);
|
|
|
|
// Assert
|
|
expect(sut.decryptPubKeyEncryptedUserKey).toBeCalledWith(
|
|
mockAuthReqResponse.key,
|
|
mockPrivateKey,
|
|
);
|
|
expect(cryptoService.setUserKey).toBeCalledWith(mockDecryptedUserKey);
|
|
});
|
|
});
|
|
|
|
describe("setKeysAfterDecryptingSharedMasterKeyAndHash", () => {
|
|
it("decrypts and sets master key and hash and user key when given valid auth request response and private key", async () => {
|
|
// Arrange
|
|
const mockAuthReqResponse = {
|
|
key: "authReqPublicKeyEncryptedMasterKey",
|
|
masterPasswordHash: "authReqPublicKeyEncryptedMasterKeyHash",
|
|
} as AuthRequestResponse;
|
|
|
|
const mockDecryptedMasterKey = {} as MasterKey;
|
|
const mockDecryptedMasterKeyHash = "mockDecryptedMasterKeyHash";
|
|
const mockDecryptedUserKey = {} as UserKey;
|
|
|
|
jest.spyOn(sut, "decryptPubKeyEncryptedMasterKeyAndHash").mockResolvedValueOnce({
|
|
masterKey: mockDecryptedMasterKey,
|
|
masterKeyHash: mockDecryptedMasterKeyHash,
|
|
});
|
|
|
|
cryptoService.setMasterKey.mockResolvedValueOnce(undefined);
|
|
cryptoService.setMasterKeyHash.mockResolvedValueOnce(undefined);
|
|
cryptoService.decryptUserKeyWithMasterKey.mockResolvedValueOnce(mockDecryptedUserKey);
|
|
cryptoService.setUserKey.mockResolvedValueOnce(undefined);
|
|
|
|
// Act
|
|
await sut.setKeysAfterDecryptingSharedMasterKeyAndHash(mockAuthReqResponse, mockPrivateKey);
|
|
|
|
// Assert
|
|
expect(sut.decryptPubKeyEncryptedMasterKeyAndHash).toBeCalledWith(
|
|
mockAuthReqResponse.key,
|
|
mockAuthReqResponse.masterPasswordHash,
|
|
mockPrivateKey,
|
|
);
|
|
expect(cryptoService.setMasterKey).toBeCalledWith(mockDecryptedMasterKey);
|
|
expect(cryptoService.setMasterKeyHash).toBeCalledWith(mockDecryptedMasterKeyHash);
|
|
expect(cryptoService.decryptUserKeyWithMasterKey).toBeCalledWith(mockDecryptedMasterKey);
|
|
expect(cryptoService.setUserKey).toBeCalledWith(mockDecryptedUserKey);
|
|
});
|
|
});
|
|
|
|
describe("decryptAuthReqPubKeyEncryptedUserKey", () => {
|
|
it("returns a decrypted user key when given valid public key encrypted user key and an auth req private key", async () => {
|
|
// Arrange
|
|
const mockPubKeyEncryptedUserKey = "pubKeyEncryptedUserKey";
|
|
const mockDecryptedUserKeyBytes = new Uint8Array(64);
|
|
const mockDecryptedUserKey = new SymmetricCryptoKey(mockDecryptedUserKeyBytes) as UserKey;
|
|
|
|
cryptoService.rsaDecrypt.mockResolvedValueOnce(mockDecryptedUserKeyBytes);
|
|
|
|
// Act
|
|
const result = await sut.decryptPubKeyEncryptedUserKey(
|
|
mockPubKeyEncryptedUserKey,
|
|
mockPrivateKey,
|
|
);
|
|
|
|
// Assert
|
|
expect(cryptoService.rsaDecrypt).toBeCalledWith(mockPubKeyEncryptedUserKey, mockPrivateKey);
|
|
expect(result).toEqual(mockDecryptedUserKey);
|
|
});
|
|
});
|
|
|
|
describe("decryptAuthReqPubKeyEncryptedMasterKeyAndHash", () => {
|
|
it("returns a decrypted master key and hash when given a valid public key encrypted master key, public key encrypted master key hash, and an auth req private key", async () => {
|
|
// Arrange
|
|
const mockPubKeyEncryptedMasterKey = "pubKeyEncryptedMasterKey";
|
|
const mockPubKeyEncryptedMasterKeyHash = "pubKeyEncryptedMasterKeyHash";
|
|
|
|
const mockDecryptedMasterKeyBytes = new Uint8Array(64);
|
|
const mockDecryptedMasterKey = new SymmetricCryptoKey(
|
|
mockDecryptedMasterKeyBytes,
|
|
) as MasterKey;
|
|
const mockDecryptedMasterKeyHashBytes = new Uint8Array(64);
|
|
const mockDecryptedMasterKeyHash = Utils.fromBufferToUtf8(mockDecryptedMasterKeyHashBytes);
|
|
|
|
cryptoService.rsaDecrypt
|
|
.mockResolvedValueOnce(mockDecryptedMasterKeyBytes)
|
|
.mockResolvedValueOnce(mockDecryptedMasterKeyHashBytes);
|
|
|
|
// Act
|
|
const result = await sut.decryptPubKeyEncryptedMasterKeyAndHash(
|
|
mockPubKeyEncryptedMasterKey,
|
|
mockPubKeyEncryptedMasterKeyHash,
|
|
mockPrivateKey,
|
|
);
|
|
|
|
// Assert
|
|
expect(cryptoService.rsaDecrypt).toHaveBeenNthCalledWith(
|
|
1,
|
|
mockPubKeyEncryptedMasterKey,
|
|
mockPrivateKey,
|
|
);
|
|
expect(cryptoService.rsaDecrypt).toHaveBeenNthCalledWith(
|
|
2,
|
|
mockPubKeyEncryptedMasterKeyHash,
|
|
mockPrivateKey,
|
|
);
|
|
expect(result.masterKey).toEqual(mockDecryptedMasterKey);
|
|
expect(result.masterKeyHash).toEqual(mockDecryptedMasterKeyHash);
|
|
});
|
|
});
|
|
});
|