mirror of
https://github.com/bitwarden/browser.git
synced 2025-01-02 18:17:46 +01:00
Ps/allow state provider migrations from v 9 (#7263)
* Provide missing spec helpers and fakers * We need to be able to migrate v9 stuff to state providers
This commit is contained in:
parent
bc1f93d098
commit
69657a5ab5
@ -4,6 +4,9 @@ import {
|
|||||||
KeyDefinition,
|
KeyDefinition,
|
||||||
ActiveUserState,
|
ActiveUserState,
|
||||||
SingleUserState,
|
SingleUserState,
|
||||||
|
SingleUserStateProvider,
|
||||||
|
StateProvider,
|
||||||
|
ActiveUserStateProvider,
|
||||||
} from "../src/platform/state";
|
} from "../src/platform/state";
|
||||||
import { UserId } from "../src/types/guid";
|
import { UserId } from "../src/types/guid";
|
||||||
|
|
||||||
@ -26,7 +29,7 @@ export class FakeGlobalStateProvider implements GlobalStateProvider {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class FakeSingleUserStateProvider {
|
export class FakeSingleUserStateProvider implements SingleUserStateProvider {
|
||||||
states: Map<string, SingleUserState<unknown>> = new Map();
|
states: Map<string, SingleUserState<unknown>> = new Map();
|
||||||
get<T>(userId: UserId, keyDefinition: KeyDefinition<T>): SingleUserState<T> {
|
get<T>(userId: UserId, keyDefinition: KeyDefinition<T>): SingleUserState<T> {
|
||||||
let result = this.states.get(keyDefinition.buildCacheKey("user", userId)) as SingleUserState<T>;
|
let result = this.states.get(keyDefinition.buildCacheKey("user", userId)) as SingleUserState<T>;
|
||||||
@ -43,7 +46,7 @@ export class FakeSingleUserStateProvider {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class FakeActiveUserStateProvider {
|
export class FakeActiveUserStateProvider implements ActiveUserStateProvider {
|
||||||
states: Map<string, ActiveUserState<unknown>> = new Map();
|
states: Map<string, ActiveUserState<unknown>> = new Map();
|
||||||
get<T>(keyDefinition: KeyDefinition<T>): ActiveUserState<T> {
|
get<T>(keyDefinition: KeyDefinition<T>): ActiveUserState<T> {
|
||||||
let result = this.states.get(
|
let result = this.states.get(
|
||||||
@ -61,3 +64,21 @@ export class FakeActiveUserStateProvider {
|
|||||||
return this.get(keyDefinition) as FakeActiveUserState<T>;
|
return this.get(keyDefinition) as FakeActiveUserState<T>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class FakeStateProvider implements StateProvider {
|
||||||
|
getActive<T>(keyDefinition: KeyDefinition<T>): ActiveUserState<T> {
|
||||||
|
return this.activeUser.get(keyDefinition);
|
||||||
|
}
|
||||||
|
|
||||||
|
getGlobal<T>(keyDefinition: KeyDefinition<T>): GlobalState<T> {
|
||||||
|
return this.global.get(keyDefinition);
|
||||||
|
}
|
||||||
|
|
||||||
|
getUser<T>(userId: UserId, keyDefinition: KeyDefinition<T>): SingleUserState<T> {
|
||||||
|
return this.singleUser.get(userId, keyDefinition);
|
||||||
|
}
|
||||||
|
|
||||||
|
global: FakeGlobalStateProvider = new FakeGlobalStateProvider();
|
||||||
|
singleUser: FakeSingleUserStateProvider = new FakeSingleUserStateProvider();
|
||||||
|
activeUser: FakeActiveUserStateProvider = new FakeActiveUserStateProvider();
|
||||||
|
}
|
||||||
|
@ -69,7 +69,7 @@ describe("RemoveLegacyEtmKeyMigrator", () => {
|
|||||||
|
|
||||||
describe("getFromGlobal", () => {
|
describe("getFromGlobal", () => {
|
||||||
it("should return the correct value", async () => {
|
it("should return the correct value", async () => {
|
||||||
sut.currentVersion = 10;
|
sut.currentVersion = 9;
|
||||||
const value = await sut.getFromGlobal({
|
const value = await sut.getFromGlobal({
|
||||||
stateDefinition: { name: "serviceName" },
|
stateDefinition: { name: "serviceName" },
|
||||||
key: "key",
|
key: "key",
|
||||||
@ -77,33 +77,33 @@ describe("RemoveLegacyEtmKeyMigrator", () => {
|
|||||||
expect(value).toEqual("global_serviceName_key");
|
expect(value).toEqual("global_serviceName_key");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should throw if the current version is less than 10", () => {
|
it("should throw if the current version is less than 9", () => {
|
||||||
expect(() =>
|
expect(() =>
|
||||||
sut.getFromGlobal({ stateDefinition: { name: "serviceName" }, key: "key" }),
|
sut.getFromGlobal({ stateDefinition: { name: "serviceName" }, key: "key" }),
|
||||||
).toThrowError("No key builder should be used for versions prior to 10.");
|
).toThrowError("No key builder should be used for versions prior to 9.");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("setToGlobal", () => {
|
describe("setToGlobal", () => {
|
||||||
it("should set the correct value", async () => {
|
it("should set the correct value", async () => {
|
||||||
sut.currentVersion = 10;
|
sut.currentVersion = 9;
|
||||||
await sut.setToGlobal({ stateDefinition: { name: "serviceName" }, key: "key" }, "new_value");
|
await sut.setToGlobal({ stateDefinition: { name: "serviceName" }, key: "key" }, "new_value");
|
||||||
expect(storage.save).toHaveBeenCalledWith("global_serviceName_key", "new_value");
|
expect(storage.save).toHaveBeenCalledWith("global_serviceName_key", "new_value");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should throw if the current version is less than 10", () => {
|
it("should throw if the current version is less than 9", () => {
|
||||||
expect(() =>
|
expect(() =>
|
||||||
sut.setToGlobal(
|
sut.setToGlobal(
|
||||||
{ stateDefinition: { name: "serviceName" }, key: "key" },
|
{ stateDefinition: { name: "serviceName" }, key: "key" },
|
||||||
"global_serviceName_key",
|
"global_serviceName_key",
|
||||||
),
|
),
|
||||||
).toThrowError("No key builder should be used for versions prior to 10.");
|
).toThrowError("No key builder should be used for versions prior to 9.");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("getFromUser", () => {
|
describe("getFromUser", () => {
|
||||||
it("should return the correct value", async () => {
|
it("should return the correct value", async () => {
|
||||||
sut.currentVersion = 10;
|
sut.currentVersion = 9;
|
||||||
const value = await sut.getFromUser("userId", {
|
const value = await sut.getFromUser("userId", {
|
||||||
stateDefinition: { name: "serviceName" },
|
stateDefinition: { name: "serviceName" },
|
||||||
key: "key",
|
key: "key",
|
||||||
@ -111,16 +111,16 @@ describe("RemoveLegacyEtmKeyMigrator", () => {
|
|||||||
expect(value).toEqual("user_userId_serviceName_key");
|
expect(value).toEqual("user_userId_serviceName_key");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should throw if the current version is less than 10", () => {
|
it("should throw if the current version is less than 9", () => {
|
||||||
expect(() =>
|
expect(() =>
|
||||||
sut.getFromUser("userId", { stateDefinition: { name: "serviceName" }, key: "key" }),
|
sut.getFromUser("userId", { stateDefinition: { name: "serviceName" }, key: "key" }),
|
||||||
).toThrowError("No key builder should be used for versions prior to 10.");
|
).toThrowError("No key builder should be used for versions prior to 9.");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("setToUser", () => {
|
describe("setToUser", () => {
|
||||||
it("should set the correct value", async () => {
|
it("should set the correct value", async () => {
|
||||||
sut.currentVersion = 10;
|
sut.currentVersion = 9;
|
||||||
await sut.setToUser(
|
await sut.setToUser(
|
||||||
"userId",
|
"userId",
|
||||||
{ stateDefinition: { name: "serviceName" }, key: "key" },
|
{ stateDefinition: { name: "serviceName" }, key: "key" },
|
||||||
@ -129,31 +129,46 @@ describe("RemoveLegacyEtmKeyMigrator", () => {
|
|||||||
expect(storage.save).toHaveBeenCalledWith("user_userId_serviceName_key", "new_value");
|
expect(storage.save).toHaveBeenCalledWith("user_userId_serviceName_key", "new_value");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should throw if the current version is less than 10", () => {
|
it("should throw if the current version is less than 9", () => {
|
||||||
expect(() =>
|
expect(() =>
|
||||||
sut.setToUser(
|
sut.setToUser(
|
||||||
"userId",
|
"userId",
|
||||||
{ stateDefinition: { name: "serviceName" }, key: "key" },
|
{ stateDefinition: { name: "serviceName" }, key: "key" },
|
||||||
"new_value",
|
"new_value",
|
||||||
),
|
),
|
||||||
).toThrowError("No key builder should be used for versions prior to 10.");
|
).toThrowError("No key builder should be used for versions prior to 9.");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
/** Helper to create well-mocked migration helpers in migration tests */
|
/** Helper to create well-mocked migration helpers in migration tests */
|
||||||
export function mockMigrationHelper(storageJson: any): MockProxy<MigrationHelper> {
|
export function mockMigrationHelper(
|
||||||
|
storageJson: any,
|
||||||
|
stateVersion = 0,
|
||||||
|
): MockProxy<MigrationHelper> {
|
||||||
const logService: MockProxy<LogService> = mock();
|
const logService: MockProxy<LogService> = mock();
|
||||||
const storage: MockProxy<AbstractStorageService> = mock();
|
const storage: MockProxy<AbstractStorageService> = mock();
|
||||||
storage.get.mockImplementation((key) => (storageJson as any)[key]);
|
storage.get.mockImplementation((key) => (storageJson as any)[key]);
|
||||||
storage.save.mockImplementation(async (key, value) => {
|
storage.save.mockImplementation(async (key, value) => {
|
||||||
(storageJson as any)[key] = value;
|
(storageJson as any)[key] = value;
|
||||||
});
|
});
|
||||||
const helper = new MigrationHelper(0, storage, logService);
|
const helper = new MigrationHelper(stateVersion, storage, logService);
|
||||||
|
|
||||||
const mockHelper = mock<MigrationHelper>();
|
const mockHelper = mock<MigrationHelper>();
|
||||||
mockHelper.get.mockImplementation((key) => helper.get(key));
|
mockHelper.get.mockImplementation((key) => helper.get(key));
|
||||||
mockHelper.set.mockImplementation((key, value) => helper.set(key, value));
|
mockHelper.set.mockImplementation((key, value) => helper.set(key, value));
|
||||||
|
mockHelper.getFromGlobal.mockImplementation((keyDefinition) =>
|
||||||
|
helper.getFromGlobal(keyDefinition),
|
||||||
|
);
|
||||||
|
mockHelper.setToGlobal.mockImplementation((keyDefinition, value) =>
|
||||||
|
helper.setToGlobal(keyDefinition, value),
|
||||||
|
);
|
||||||
|
mockHelper.getFromUser.mockImplementation((userId, keyDefinition) =>
|
||||||
|
helper.getFromUser(userId, keyDefinition),
|
||||||
|
);
|
||||||
|
mockHelper.setToUser.mockImplementation((userId, keyDefinition, value) =>
|
||||||
|
helper.setToUser(userId, keyDefinition, value),
|
||||||
|
);
|
||||||
mockHelper.getAccounts.mockImplementation(() => helper.getAccounts());
|
mockHelper.getAccounts.mockImplementation(() => helper.getAccounts());
|
||||||
return mockHelper;
|
return mockHelper;
|
||||||
}
|
}
|
||||||
|
@ -123,8 +123,8 @@ export class MigrationHelper {
|
|||||||
* @returns
|
* @returns
|
||||||
*/
|
*/
|
||||||
private getUserKey(userId: string, keyDefinition: KeyDefinitionLike): string {
|
private getUserKey(userId: string, keyDefinition: KeyDefinitionLike): string {
|
||||||
if (this.currentVersion < 10) {
|
if (this.currentVersion < 9) {
|
||||||
return userKeyBuilderPre10();
|
return userKeyBuilderPre9();
|
||||||
} else {
|
} else {
|
||||||
return userKeyBuilder(userId, keyDefinition);
|
return userKeyBuilder(userId, keyDefinition);
|
||||||
}
|
}
|
||||||
@ -137,8 +137,8 @@ export class MigrationHelper {
|
|||||||
* @returns
|
* @returns
|
||||||
*/
|
*/
|
||||||
private getGlobalKey(keyDefinition: KeyDefinitionLike): string {
|
private getGlobalKey(keyDefinition: KeyDefinitionLike): string {
|
||||||
if (this.currentVersion < 10) {
|
if (this.currentVersion < 9) {
|
||||||
return globalKeyBuilderPre10();
|
return globalKeyBuilderPre9();
|
||||||
} else {
|
} else {
|
||||||
return globalKeyBuilder(keyDefinition);
|
return globalKeyBuilder(keyDefinition);
|
||||||
}
|
}
|
||||||
@ -158,8 +158,8 @@ function userKeyBuilder(userId: string, keyDefinition: KeyDefinitionLike): strin
|
|||||||
return `user_${userId}_${keyDefinition.stateDefinition.name}_${keyDefinition.key}`;
|
return `user_${userId}_${keyDefinition.stateDefinition.name}_${keyDefinition.key}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
function userKeyBuilderPre10(): string {
|
function userKeyBuilderPre9(): string {
|
||||||
throw Error("No key builder should be used for versions prior to 10.");
|
throw Error("No key builder should be used for versions prior to 9.");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -174,6 +174,6 @@ function globalKeyBuilder(keyDefinition: KeyDefinitionLike): string {
|
|||||||
return `global_${keyDefinition.stateDefinition.name}_${keyDefinition.key}`;
|
return `global_${keyDefinition.stateDefinition.name}_${keyDefinition.key}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
function globalKeyBuilderPre10(): string {
|
function globalKeyBuilderPre9(): string {
|
||||||
throw Error("No key builder should be used for versions prior to 10.");
|
throw Error("No key builder should be used for versions prior to 9.");
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user