1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00

Refactor StorageHelper to be clearer in its constructor (#2417)

This commit is contained in:
Sebastian Malton 2021-04-15 15:37:30 -04:00
parent c79b6c0015
commit 23c9255b9a
2 changed files with 54 additions and 69 deletions

View File

@ -26,7 +26,7 @@ describe("renderer/utils/StorageHelper", () => {
expect(storageHelper.defaultValue).toBe("test"); expect(storageHelper.defaultValue).toBe("test");
expect(storageHelper.get()).toBe("test"); expect(storageHelper.get()).toBe("test");
await storageHelper.init(); storageHelper.init();
expect(storageHelper.key).toBe(storageKey); expect(storageHelper.key).toBe(storageKey);
expect(storageHelper.defaultValue).toBe("test"); expect(storageHelper.defaultValue).toBe("test");

View File

@ -1,7 +1,7 @@
// Helper for working with storages (e.g. window.localStorage, NodeJS/file-system, etc.) // Helper for working with storages (e.g. window.localStorage, NodeJS/file-system, etc.)
import type { CreateObservableOptions } from "mobx/lib/api/observable"; import type { CreateObservableOptions } from "mobx/lib/api/observable";
import { action, comparer, observable, toJS, when } from "mobx"; import { action, comparer, observable, toJS, when, IObservableValue } from "mobx";
import produce, { Draft, enableMapSet, setAutoFreeze } from "immer"; import produce, { Draft, enableMapSet, setAutoFreeze } from "immer";
import { isEqual, isFunction, isPlainObject } from "lodash"; import { isEqual, isFunction, isPlainObject } from "lodash";
import logger from "../../main/logger"; import logger from "../../main/logger";
@ -25,7 +25,7 @@ export interface StorageHelperOptions<T> {
} }
export class StorageHelper<T> { export class StorageHelper<T> {
static defaultOptions: Partial<StorageHelperOptions<any>> = { static readonly defaultOptions: Partial<StorageHelperOptions<any>> = {
autoInit: true, autoInit: true,
observable: { observable: {
deep: true, deep: true,
@ -33,34 +33,33 @@ export class StorageHelper<T> {
} }
}; };
@observable private data = observable.box<T>(); private data: IObservableValue<T>;
@observable initialized = false; @observable initialized = false;
whenReady = when(() => this.initialized); whenReady = when(() => this.initialized);
get storage(): StorageAdapter<T> { public readonly storage: StorageAdapter<T>;
return this.options.storage; public readonly defaultValue: T;
}
get defaultValue(): T {
return this.options.defaultValue;
}
constructor(readonly key: string, private options: StorageHelperOptions<T>) { constructor(readonly key: string, private options: StorageHelperOptions<T>) {
this.options = { ...StorageHelper.defaultOptions, ...options }; this.data = observable.box<T>(this.options.defaultValue, {
this.configureObservable(); ...StorageHelper.defaultOptions.observable,
this.reset(); ...(options.observable ?? {})
});
this.data.observe(change => {
const { newValue, oldValue } = toJS(change, { recurseEverything: true });
this.onChange(newValue, oldValue);
});
this.storage = options.storage;
this.defaultValue = options.defaultValue;
if (this.options.autoInit) { if (this.options.autoInit) {
this.init(); this.init();
} }
} }
@action private onData = (data: T): void => {
init({ force = false } = {}) {
if (this.initialized && !force) return;
this.loadFromStorage({
onData: (data: T) => {
const notEmpty = data != null; const notEmpty = data != null;
const notDefault = !this.isDefaultValue(data); const notDefault = !this.isDefaultValue(data);
@ -69,49 +68,35 @@ export class StorageHelper<T> {
} }
this.initialized = true; this.initialized = true;
}, };
onError: (error?: any) => {
logger.error(`[init]: ${error}`, this);
},
});
}
private loadFromStorage(opts: { onData?(data: T): void, onError?(error?: any): void } = {}) { private onError = (error: any): void => {
let data: T | Promise<T>; logger.error(`[load]: ${error}`, this);
};
@action
init({ force = false } = {}) {
if (this.initialized && !force) {
return;
}
try { try {
data = this.storage.getItem(this.key); // sync reading from storage when exposed const data = this.storage.getItem(this.key);
if (data instanceof Promise) { if (data instanceof Promise) {
data.then(opts.onData, opts.onError); data.then(this.onData, this.onError);
} else { } else {
opts?.onData(data); this.onData(data);
} }
} catch (error) { } catch (error) {
logger.error(`[load]: ${error}`, this); this.onError(error);
opts?.onError(error);
} }
return data;
} }
isDefaultValue(value: T): boolean { isDefaultValue(value: T): boolean {
return isEqual(value, this.defaultValue); return isEqual(value, this.defaultValue);
} }
@action
private configureObservable(options = this.options.observable) {
this.data = observable.box<T>(this.data.get(), {
...StorageHelper.defaultOptions.observable, // inherit default observability options
...(options ?? {}),
});
this.data.observe(change => {
const { newValue, oldValue } = toJS(change, { recurseEverything: true });
this.onChange(newValue, oldValue);
});
}
protected onChange(value: T, oldValue?: T) { protected onChange(value: T, oldValue?: T) {
if (!this.initialized) return; if (!this.initialized) return;