2022-10-10 17:19:01 +02:00
|
|
|
// eslint-disable-next-line no-restricted-imports
|
2022-09-22 14:51:14 +02:00
|
|
|
import { Arg, Substitute, SubstituteOf } from "@fluffy-spoon/substitute";
|
2022-06-27 19:38:12 +02:00
|
|
|
|
|
|
|
import { LogService } from "@bitwarden/common/abstractions/log.service";
|
2022-08-16 14:05:03 +02:00
|
|
|
import {
|
2022-09-22 14:51:14 +02:00
|
|
|
MemoryStorageServiceInterface,
|
2022-08-16 14:05:03 +02:00
|
|
|
AbstractStorageService,
|
|
|
|
} from "@bitwarden/common/abstractions/storage.service";
|
2022-06-27 19:38:12 +02:00
|
|
|
import { SendType } from "@bitwarden/common/enums/sendType";
|
|
|
|
import { StateFactory } from "@bitwarden/common/factories/stateFactory";
|
2022-10-14 18:25:50 +02:00
|
|
|
import { GlobalState } from "@bitwarden/common/models/domain/global-state";
|
2022-06-27 19:38:12 +02:00
|
|
|
import { State } from "@bitwarden/common/models/domain/state";
|
2022-10-14 18:25:50 +02:00
|
|
|
import { SendView } from "@bitwarden/common/models/view/send.view";
|
2022-06-27 19:38:12 +02:00
|
|
|
import { StateMigrationService } from "@bitwarden/common/services/stateMigration.service";
|
|
|
|
|
|
|
|
import { Account } from "../models/account";
|
|
|
|
import { BrowserComponentState } from "../models/browserComponentState";
|
|
|
|
import { BrowserGroupingsComponentState } from "../models/browserGroupingsComponentState";
|
|
|
|
import { BrowserSendComponentState } from "../models/browserSendComponentState";
|
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
import { LocalBackedSessionStorageService } from "./localBackedSessionStorage.service";
|
2022-06-27 19:38:12 +02:00
|
|
|
import { StateService } from "./state.service";
|
|
|
|
|
|
|
|
describe("Browser State Service", () => {
|
|
|
|
let secureStorageService: SubstituteOf<AbstractStorageService>;
|
|
|
|
let diskStorageService: SubstituteOf<AbstractStorageService>;
|
|
|
|
let logService: SubstituteOf<LogService>;
|
|
|
|
let stateMigrationService: SubstituteOf<StateMigrationService>;
|
|
|
|
let stateFactory: SubstituteOf<StateFactory<GlobalState, Account>>;
|
|
|
|
let useAccountCache: boolean;
|
|
|
|
|
|
|
|
let state: State<GlobalState, Account>;
|
|
|
|
const userId = "userId";
|
|
|
|
|
|
|
|
let sut: StateService;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
secureStorageService = Substitute.for();
|
|
|
|
diskStorageService = Substitute.for();
|
|
|
|
logService = Substitute.for();
|
|
|
|
stateMigrationService = Substitute.for();
|
|
|
|
stateFactory = Substitute.for();
|
|
|
|
useAccountCache = true;
|
|
|
|
|
|
|
|
state = new State(new GlobalState());
|
|
|
|
state.accounts[userId] = new Account({
|
|
|
|
profile: { userId: userId },
|
|
|
|
});
|
|
|
|
state.activeUserId = userId;
|
|
|
|
});
|
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
describe("direct memory storage access", () => {
|
2022-09-22 14:51:14 +02:00
|
|
|
let memoryStorageService: LocalBackedSessionStorageService;
|
2022-08-16 14:05:03 +02:00
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
// We need `AbstractCachedStorageService` in the prototype chain to correctly test cache bypass.
|
|
|
|
memoryStorageService = Object.create(LocalBackedSessionStorageService.prototype);
|
|
|
|
|
|
|
|
sut = new StateService(
|
|
|
|
diskStorageService,
|
|
|
|
secureStorageService,
|
|
|
|
memoryStorageService,
|
|
|
|
logService,
|
|
|
|
stateMigrationService,
|
|
|
|
stateFactory,
|
|
|
|
useAccountCache
|
|
|
|
);
|
|
|
|
});
|
2022-06-27 19:38:12 +02:00
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
it("should bypass cache if possible", async () => {
|
|
|
|
const spyBypass = jest
|
|
|
|
.spyOn(memoryStorageService, "getBypassCache")
|
|
|
|
.mockResolvedValue("value");
|
|
|
|
const spyGet = jest.spyOn(memoryStorageService, "get");
|
|
|
|
const result = await sut.getFromSessionMemory("key");
|
|
|
|
expect(spyBypass).toHaveBeenCalled();
|
|
|
|
expect(spyGet).not.toHaveBeenCalled();
|
|
|
|
expect(result).toBe("value");
|
2022-06-27 19:38:12 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
describe("state methods", () => {
|
2022-09-22 14:51:14 +02:00
|
|
|
let memoryStorageService: SubstituteOf<AbstractStorageService & MemoryStorageServiceInterface>;
|
2022-08-16 14:05:03 +02:00
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
memoryStorageService = Substitute.for();
|
|
|
|
const stateGetter = (key: string) => Promise.resolve(JSON.parse(JSON.stringify(state)));
|
2022-09-22 14:51:14 +02:00
|
|
|
memoryStorageService.get("state", Arg.any()).mimicks(stateGetter);
|
2022-08-16 14:05:03 +02:00
|
|
|
|
|
|
|
sut = new StateService(
|
|
|
|
diskStorageService,
|
|
|
|
secureStorageService,
|
|
|
|
memoryStorageService,
|
|
|
|
logService,
|
|
|
|
stateMigrationService,
|
|
|
|
stateFactory,
|
|
|
|
useAccountCache
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("getBrowserGroupingComponentState", () => {
|
|
|
|
it("should return a BrowserGroupingsComponentState", async () => {
|
|
|
|
state.accounts[userId].groupings = new BrowserGroupingsComponentState();
|
2022-06-27 19:38:12 +02:00
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
const actual = await sut.getBrowserGroupingComponentState();
|
|
|
|
expect(actual).toBeInstanceOf(BrowserGroupingsComponentState);
|
|
|
|
});
|
2022-06-27 19:38:12 +02:00
|
|
|
});
|
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
describe("getBrowserCipherComponentState", () => {
|
|
|
|
it("should return a BrowserComponentState", async () => {
|
|
|
|
const componentState = new BrowserComponentState();
|
|
|
|
componentState.scrollY = 0;
|
|
|
|
componentState.searchText = "test";
|
|
|
|
state.accounts[userId].ciphers = componentState;
|
|
|
|
|
|
|
|
const actual = await sut.getBrowserCipherComponentState();
|
|
|
|
expect(actual).toStrictEqual(componentState);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("getBrowserSendComponentState", () => {
|
|
|
|
it("should return a BrowserSendComponentState", async () => {
|
|
|
|
const sendState = new BrowserSendComponentState();
|
|
|
|
sendState.sends = [new SendView(), new SendView()];
|
|
|
|
sendState.typeCounts = new Map<SendType, number>([
|
|
|
|
[SendType.File, 3],
|
|
|
|
[SendType.Text, 5],
|
|
|
|
]);
|
|
|
|
state.accounts[userId].send = sendState;
|
|
|
|
|
|
|
|
const actual = await sut.getBrowserSendComponentState();
|
|
|
|
expect(actual).toBeInstanceOf(BrowserSendComponentState);
|
|
|
|
expect(actual).toMatchObject(sendState);
|
|
|
|
});
|
2022-06-27 19:38:12 +02:00
|
|
|
});
|
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
describe("getBrowserSendTypeComponentState", () => {
|
|
|
|
it("should return a BrowserComponentState", async () => {
|
|
|
|
const componentState = new BrowserComponentState();
|
|
|
|
componentState.scrollY = 0;
|
|
|
|
componentState.searchText = "test";
|
|
|
|
state.accounts[userId].sendType = componentState;
|
2022-06-27 19:38:12 +02:00
|
|
|
|
2022-08-16 14:05:03 +02:00
|
|
|
const actual = await sut.getBrowserSendTypeComponentState();
|
|
|
|
expect(actual).toStrictEqual(componentState);
|
|
|
|
});
|
2022-06-27 19:38:12 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|