1
0
Fork 0
mirror of https://github.com/sussy-code/smov.git synced 2025-01-17 01:51:24 +01:00

Made type-safe versioned store, migrated to it

Co-authored-by: mrjvs <mistrjvs@gmail.com>
This commit is contained in:
Jip Fr 2023-02-12 00:41:55 +01:00
parent dd14b575eb
commit 942a6cc9c0
10 changed files with 408 additions and 485 deletions

View file

@ -9,6 +9,7 @@ import "@/setup/i18n";
import "@/setup/index.css"; import "@/setup/index.css";
import "@/backend"; import "@/backend";
import { initializeChromecast } from "./setup/chromecast"; import { initializeChromecast } from "./setup/chromecast";
import { initializeStores } from "./utils/storage";
// initialize // initialize
const key = const key =
@ -42,12 +43,19 @@ initializeChromecast();
// TODO general todos: // TODO general todos:
// - localize everything (fix loading screen text (series vs movies)) // - localize everything (fix loading screen text (series vs movies))
const LazyLoadedApp = React.lazy(async () => {
await initializeStores();
return {
default: App,
};
});
ReactDOM.render( ReactDOM.render(
<React.StrictMode> <React.StrictMode>
<ErrorBoundary> <ErrorBoundary>
<HashRouter> <HashRouter>
<Suspense fallback=""> <Suspense fallback="">
<App /> <LazyLoadedApp />
</Suspense> </Suspense>
</HashRouter> </HashRouter>
</ErrorBoundary> </ErrorBoundary>

View file

@ -1,17 +1,8 @@
import { MWMediaMeta } from "@/backend/metadata/types"; import { MWMediaMeta } from "@/backend/metadata/types";
import { import { useStore } from "@/utils/storage";
createContext, import { createContext, ReactNode, useContext, useMemo } from "react";
ReactNode,
useCallback,
useContext,
useMemo,
useState,
} from "react";
import { BookmarkStore } from "./store"; import { BookmarkStore } from "./store";
import { BookmarkStoreData } from "./types";
interface BookmarkStoreData {
bookmarks: MWMediaMeta[];
}
interface BookmarkStoreDataWrapper { interface BookmarkStoreDataWrapper {
setItemBookmark(media: MWMediaMeta, bookedmarked: boolean): void; setItemBookmark(media: MWMediaMeta, bookedmarked: boolean): void;
@ -36,25 +27,7 @@ function getBookmarkIndexFromMedia(
} }
export function BookmarkContextProvider(props: { children: ReactNode }) { export function BookmarkContextProvider(props: { children: ReactNode }) {
const bookmarkLocalstorage = BookmarkStore.get(); const [bookmarkStorage, setBookmarked] = useStore(BookmarkStore);
const [bookmarkStorage, setBookmarkStore] = useState<BookmarkStoreData>(
bookmarkLocalstorage as BookmarkStoreData
);
const setBookmarked = useCallback(
(data: any) => {
setBookmarkStore((old) => {
const old2 = JSON.parse(JSON.stringify(old));
let newData = data;
if (data.constructor === Function) {
newData = data(old2);
}
bookmarkLocalstorage.save(newData);
return newData;
});
},
[bookmarkLocalstorage, setBookmarkStore]
);
const contextValue = useMemo( const contextValue = useMemo(
() => ({ () => ({

View file

@ -1,17 +1,18 @@
import { versionedStoreBuilder } from "@/utils/storage"; import { createVersionedStore } from "@/utils/storage";
import { BookmarkStoreData } from "./types";
export const BookmarkStore = versionedStoreBuilder() export const BookmarkStore = createVersionedStore<BookmarkStoreData>()
.setKey("mw-bookmarks") .setKey("mw-bookmarks")
.addVersion({ .addVersion({
version: 0, version: 0,
migrate() {
return {
bookmarks: [], // TODO migrate bookmarks
};
},
}) })
.addVersion({ .addVersion({
version: 1, version: 1,
migrate() {
return {
bookmarks: [],
};
},
create() { create() {
return { return {
bookmarks: [], bookmarks: [],

View file

@ -0,0 +1,5 @@
import { MWMediaMeta } from "@/backend/metadata/types";
export interface BookmarkStoreData {
bookmarks: MWMediaMeta[];
}

View file

@ -1,5 +1,6 @@
import { DetailedMeta } from "@/backend/metadata/getmeta"; import { DetailedMeta } from "@/backend/metadata/getmeta";
import { MWMediaMeta, MWMediaType } from "@/backend/metadata/types"; import { MWMediaType } from "@/backend/metadata/types";
import { useStore } from "@/utils/storage";
import { import {
createContext, createContext,
ReactNode, ReactNode,
@ -7,9 +8,9 @@ import {
useContext, useContext,
useMemo, useMemo,
useRef, useRef,
useState,
} from "react"; } from "react";
import { VideoProgressStore } from "./store"; import { VideoProgressStore } from "./store";
import { StoreMediaItem, WatchedStoreItem, WatchedStoreData } from "./types";
const FIVETEEN_MINUTES = 15 * 60; const FIVETEEN_MINUTES = 15 * 60;
const FIVE_MINUTES = 5 * 60; const FIVE_MINUTES = 5 * 60;
@ -34,29 +35,8 @@ function shouldSave(
return true; return true;
} }
interface MediaItem {
meta: MWMediaMeta;
series?: {
episodeId: string;
seasonId: string;
episode: number;
season: number;
};
}
export interface WatchedStoreItem {
item: MediaItem;
progress: number;
percentage: number;
watchedAt: number;
}
export interface WatchedStoreData {
items: WatchedStoreItem[];
}
interface WatchedStoreDataWrapper { interface WatchedStoreDataWrapper {
updateProgress(media: MediaItem, progress: number, total: number): void; updateProgress(media: StoreMediaItem, progress: number, total: number): void;
getFilteredWatched(): WatchedStoreItem[]; getFilteredWatched(): WatchedStoreItem[];
removeProgress(id: string): void; removeProgress(id: string): void;
watched: WatchedStoreData; watched: WatchedStoreData;
@ -72,7 +52,7 @@ const WatchedContext = createContext<WatchedStoreDataWrapper>({
}); });
WatchedContext.displayName = "WatchedContext"; WatchedContext.displayName = "WatchedContext";
function isSameEpisode(media: MediaItem, v: MediaItem) { function isSameEpisode(media: StoreMediaItem, v: StoreMediaItem) {
return ( return (
media.meta.id === v.meta.id && media.meta.id === v.meta.id &&
(!media.series || (!media.series ||
@ -82,24 +62,7 @@ function isSameEpisode(media: MediaItem, v: MediaItem) {
} }
export function WatchedContextProvider(props: { children: ReactNode }) { export function WatchedContextProvider(props: { children: ReactNode }) {
const watchedLocalstorage = VideoProgressStore.get(); const [watched, setWatched] = useStore(VideoProgressStore);
const [watched, setWatchedReal] = useState<WatchedStoreData>(
watchedLocalstorage as WatchedStoreData
);
const setWatched = useCallback(
(data: any) => {
setWatchedReal((old) => {
let newData = data;
if (data.constructor === Function) {
newData = data(old);
}
watchedLocalstorage.save(newData);
return newData;
});
},
[setWatchedReal, watchedLocalstorage]
);
const contextValue = useMemo( const contextValue = useMemo(
() => ({ () => ({
@ -110,7 +73,11 @@ export function WatchedContextProvider(props: { children: ReactNode }) {
return newData; return newData;
}); });
}, },
updateProgress(media: MediaItem, progress: number, total: number): void { updateProgress(
media: StoreMediaItem,
progress: number,
total: number
): void {
setWatched((data: WatchedStoreData) => { setWatched((data: WatchedStoreData) => {
const newData = { ...data }; const newData = { ...data };
let item = newData.items.find((v) => isSameEpisode(media, v.item)); let item = newData.items.find((v) => isSameEpisode(media, v.item));
@ -176,7 +143,7 @@ export function useWatchedContext() {
} }
function isSameEpisodeMeta( function isSameEpisodeMeta(
media: MediaItem, media: StoreMediaItem,
mediaTwo: DetailedMeta | null, mediaTwo: DetailedMeta | null,
episodeId?: string episodeId?: string
) { ) {

View file

@ -0,0 +1,170 @@
import { DetailedMeta, getMetaFromId } from "@/backend/metadata/getmeta";
import { searchForMedia } from "@/backend/metadata/search";
import { MWMediaMeta, MWMediaType } from "@/backend/metadata/types";
import { WatchedStoreData, WatchedStoreItem } from "../types";
interface OldMediaBase {
mediaId: number;
mediaType: MWMediaType;
percentage: number;
progress: number;
providerId: string;
title: string;
year: number;
}
interface OldMovie extends OldMediaBase {
mediaType: MWMediaType.MOVIE;
}
interface OldSeries extends OldMediaBase {
mediaType: MWMediaType.SERIES;
episodeId: number;
seasonId: number;
}
export interface OldData {
items: (OldMovie | OldSeries)[];
}
export async function migrateV2(old: OldData) {
const oldData = old;
if (!oldData) return;
const uniqueMedias: Record<string, any> = {};
oldData.items.forEach((item: any) => {
if (uniqueMedias[item.mediaId]) return;
uniqueMedias[item.mediaId] = item;
});
const yearsAreClose = (a: number, b: number) => {
return Math.abs(a - b) <= 1;
};
const mediaMetas: Record<string, Record<string, DetailedMeta | null>> = {};
const relevantItems = await Promise.all(
Object.values(uniqueMedias).map(async (item) => {
const year = Number(item.year.toString().split("-")[0]);
const data = await searchForMedia({
searchQuery: `${item.title} ${year}`,
type: item.mediaType,
});
const relevantItem = data.find((res) =>
yearsAreClose(Number(res.year), year)
);
if (!relevantItem) {
console.error("No item");
return;
}
return {
id: item.mediaId,
data: relevantItem,
};
})
);
for (const item of relevantItems.filter(Boolean)) {
if (!item) continue;
let keys: (string | null)[][] = [["0", "0"]];
if (item.data.type === "series") {
// TODO sort episodes by season & episode so it shows the "highest" episode as last
const meta = await getMetaFromId(item.data.type, item.data.id);
if (!meta || !meta?.meta.seasons) return;
const seasonNumbers = [
...new Set(
oldData.items
.filter((watchedEntry: any) => watchedEntry.mediaId === item.id)
.map((watchedEntry: any) => watchedEntry.seasonId)
),
];
const seasons = seasonNumbers.map((num) => ({
num,
season: meta.meta?.seasons?.[(num as number) - 1],
}));
keys = seasons
.map((season) => (season ? [season.num, season?.season?.id] : []))
.filter((entry) => entry.length > 0);
}
if (!mediaMetas[item.id]) mediaMetas[item.id] = {};
await Promise.all(
keys.map(async ([key, id]) => {
if (!key) return;
mediaMetas[item.id][key] = await getMetaFromId(
item.data.type,
item.data.id,
id === "0" || id === null ? undefined : id
);
})
);
}
// We've got all the metadata you can dream of now
// Now let's convert stuff into the new format.
const newData: WatchedStoreData = {
...oldData,
items: [],
};
for (const oldWatched of oldData.items) {
if (oldWatched.mediaType === "movie") {
if (!mediaMetas[oldWatched.mediaId]["0"]?.meta) continue;
const newItem: WatchedStoreItem = {
item: {
meta: mediaMetas[oldWatched.mediaId]["0"]?.meta as MWMediaMeta,
},
progress: oldWatched.progress,
percentage: oldWatched.percentage,
watchedAt: Date.now(), // There was no watchedAt in V2
};
oldData.items = oldData.items.filter(
(item) => JSON.stringify(item) !== JSON.stringify(oldWatched)
);
newData.items.push(newItem);
} else if (oldWatched.mediaType === "series") {
if (!mediaMetas[oldWatched.mediaId][oldWatched.seasonId]?.meta) continue;
const meta = mediaMetas[oldWatched.mediaId][oldWatched.seasonId]
?.meta as MWMediaMeta;
if (meta.type !== "series") return;
const newItem: WatchedStoreItem = {
item: {
meta,
series: {
episode: Number(oldWatched.episodeId),
season: Number(oldWatched.seasonId),
seasonId: meta.seasonData.id,
episodeId:
meta.seasonData.episodes[Number(oldWatched.episodeId) - 1].id,
},
},
progress: oldWatched.progress,
percentage: oldWatched.percentage,
watchedAt: Date.now(), // There was no watchedAt in V2
// Put watchedAt in the future to show last episode as most recently
};
if (
newData.items.find(
(item) =>
item.item.meta.id === newItem.item.meta.id &&
item.item.series?.episodeId === newItem.item.series?.episodeId
)
)
continue;
oldData.items = oldData.items.filter(
(item) => JSON.stringify(item) !== JSON.stringify(oldWatched)
);
newData.items.push(newItem);
}
}
return newData;
}

View file

@ -1,57 +1,25 @@
import { DetailedMeta, getMetaFromId } from "@/backend/metadata/getmeta"; import { createVersionedStore } from "@/utils/storage";
import { searchForMedia } from "@/backend/metadata/search"; import { migrateV2, OldData } from "./migrations/v2";
import { MWMediaMeta, MWMediaType } from "@/backend/metadata/types"; import { WatchedStoreData } from "./types";
import { versionedStoreBuilder } from "@/utils/storage";
import { WatchedStoreData, WatchedStoreItem } from "./context";
interface OldMediaBase { export const VideoProgressStore = createVersionedStore<WatchedStoreData>()
mediaId: number;
mediaType: MWMediaType;
percentage: number;
progress: number;
providerId: string;
title: string;
year: number;
}
interface OldMovie extends OldMediaBase {
mediaType: MWMediaType.MOVIE;
}
interface OldSeries extends OldMediaBase {
mediaType: MWMediaType.SERIES;
episodeId: number;
seasonId: number;
}
interface OldData {
items: (OldMovie | OldSeries)[];
}
export const VideoProgressStore = versionedStoreBuilder()
.setKey("video-progress") .setKey("video-progress")
.addVersion({ .addVersion({
version: 0, version: 0,
migrate() {
return {
items: [], // dont migrate from version 0 to version 1, unmigratable
};
},
}) })
.addVersion({ .addVersion({
version: 1, version: 1,
migrate() { async migrate(old: OldData) {
return { return migrateV2(old);
items: [],
};
}, },
}) })
.addVersion({ .addVersion({
version: 2, version: 2,
migrate(old: OldData) {
requestAnimationFrame(() => {
// eslint-disable-next-line no-use-before-define
migrateV2(old);
});
return {
items: [],
};
},
create() { create() {
return { return {
items: [], items: [],
@ -59,153 +27,3 @@ export const VideoProgressStore = versionedStoreBuilder()
}, },
}) })
.build(); .build();
async function migrateV2(old: OldData) {
const oldData = old;
if (!oldData) return;
const uniqueMedias: Record<string, any> = {};
oldData.items.forEach((item: any) => {
if (uniqueMedias[item.mediaId]) return;
uniqueMedias[item.mediaId] = item;
});
const yearsAreClose = (a: number, b: number) => {
return Math.abs(a - b) <= 1;
};
const mediaMetas: Record<string, Record<string, DetailedMeta | null>> = {};
const relevantItems = await Promise.all(
Object.values(uniqueMedias).map(async (item) => {
const year = Number(item.year.toString().split("-")[0]);
const data = await searchForMedia({
searchQuery: `${item.title} ${year}`,
type: item.mediaType,
});
const relevantItem = data.find((res) =>
yearsAreClose(Number(res.year), year)
);
if (!relevantItem) {
console.error("No item");
return;
}
return {
id: item.mediaId,
data: relevantItem,
};
})
);
for (const item of relevantItems.filter(Boolean)) {
if (!item) continue;
let keys: (string | null)[][] = [["0", "0"]];
if (item.data.type === "series") {
const meta = await getMetaFromId(item.data.type, item.data.id);
if (!meta || !meta?.meta.seasons) return;
const seasonNumbers = [
...new Set(
oldData.items
.filter((watchedEntry: any) => watchedEntry.mediaId === item.id)
.map((watchedEntry: any) => watchedEntry.seasonId)
),
];
const seasons = seasonNumbers
.map((num) => ({
num,
season: meta.meta?.seasons?.[(num as number) - 1],
}))
.filter(Boolean);
keys = seasons
.map((season) => (season ? [season.num, season?.season?.id] : []))
.filter((entry) => entry.length > 0); // Stupid TypeScript
}
if (!mediaMetas[item.id]) mediaMetas[item.id] = {};
await Promise.all(
keys.map(async ([key, id]) => {
if (!key) return;
mediaMetas[item.id][key] = await getMetaFromId(
item.data.type,
item.data.id,
id === "0" || id === null ? undefined : id
);
})
);
}
// We've got all the metadata you can dream of now
// Now let's convert stuff into the new format.
interface WatchedStoreDataWithVersion extends WatchedStoreData {
"--version": number;
}
const newData: WatchedStoreDataWithVersion = {
...oldData,
items: [],
"--version": 2,
};
for (const oldWatched of oldData.items) {
if (oldWatched.mediaType === "movie") {
if (!mediaMetas[oldWatched.mediaId]["0"]?.meta) continue;
const newItem: WatchedStoreItem = {
item: {
meta: mediaMetas[oldWatched.mediaId]["0"]?.meta as MWMediaMeta,
},
progress: oldWatched.progress,
percentage: oldWatched.percentage,
watchedAt: Date.now(), // There was no watchedAt in V2
};
oldData.items = oldData.items.filter(
(item) => JSON.stringify(item) !== JSON.stringify(oldWatched)
);
newData.items.push(newItem);
} else if (oldWatched.mediaType === "series") {
if (!mediaMetas[oldWatched.mediaId][oldWatched.seasonId]?.meta) continue;
const meta = mediaMetas[oldWatched.mediaId][oldWatched.seasonId]
?.meta as MWMediaMeta;
if (meta.type !== "series") return;
const newItem: WatchedStoreItem = {
item: {
meta,
series: {
episode: Number(oldWatched.episodeId),
season: Number(oldWatched.seasonId),
seasonId: meta.seasonData.id,
episodeId:
meta.seasonData.episodes[Number(oldWatched.episodeId) - 1].id,
},
},
progress: oldWatched.progress,
percentage: oldWatched.percentage,
watchedAt: Date.now(), // There was no watchedAt in V2
};
if (
newData.items.find(
(item) =>
item.item.meta.id === newItem.item.meta.id &&
item.item.series?.episodeId === newItem.item.series?.episodeId
)
)
continue;
oldData.items = oldData.items.filter(
(item) => JSON.stringify(item) !== JSON.stringify(oldWatched)
);
newData.items.push(newItem);
}
}
console.log(JSON.stringify(old), JSON.stringify(newData));
if (JSON.stringify(old.items) !== JSON.stringify(newData.items)) {
console.log(newData);
VideoProgressStore.get().save(newData);
}
}

View file

@ -0,0 +1,22 @@
import { MWMediaMeta } from "@/backend/metadata/types";
export interface StoreMediaItem {
meta: MWMediaMeta;
series?: {
episodeId: string;
seasonId: string;
episode: number;
season: number;
};
}
export interface WatchedStoreItem {
item: StoreMediaItem;
progress: number;
percentage: number;
watchedAt: number;
}
export interface WatchedStoreData {
items: WatchedStoreItem[];
}

View file

@ -1,232 +1,188 @@
// TODO make type and react safe!! import { useEffect, useState } from "react";
/*
it needs to be react-ified by having a save function not on the instance itself.
also type safety is important, this is all spaghetti with "any" everywhere
*/
function buildStoreObject(d: any) { interface StoreVersion<A> {
const data: any = { version: number;
versions: d.versions, migrate?(data: A): any;
currentVersion: d.maxVersion, create?: () => A;
id: d.storageString, }
interface StoreRet<T> {
save: (data: T) => void;
get: () => T;
_raw: () => any;
onChange: (cb: (data: T) => void) => {
destroy: () => void;
}; };
function update(this: any, obj2: any) {
let obj = obj2;
if (!obj) throw new Error("object to update is not an object");
// repeat until object fully updated
if (obj["--version"] === undefined) obj["--version"] = 0;
while (obj["--version"] !== this.currentVersion) {
// get version
let version: any = obj["--version"] || 0;
if (version.constructor !== Number || version < 0) version = -42;
// invalid on purpose so it will reset
else {
version = ((version as number) + 1).toString();
} }
// check if version exists export interface StoreBuilder<T> {
if (!this.versions[version]) { setKey: (key: string) => StoreBuilder<T>;
console.error( addVersion: <A>(ver: StoreVersion<A>) => StoreBuilder<T>;
`Version not found for storage item in store ${this.id}, resetting` build: () => StoreRet<T>;
);
obj = null;
break;
} }
// update object interface InternalStoreData {
obj = this.versions[version].update(obj); versions: StoreVersion<any>[];
key: string | null;
} }
// if resulting obj is null, use latest version as init object const storeCallbacks: Record<string, ((data: any) => void)[]> = {};
if (obj === null) { const stores: Record<string, [StoreRet<any>, InternalStoreData]> = {};
console.error(
`Storage item for store ${this.id} has been reset due to faulty updates` export async function initializeStores() {
); // migrate all stores
return this.versions[this.currentVersion.toString()].init(); for (const [store, internal] of Object.values(stores)) {
const versions = internal.versions.sort((a, b) => a.version - b.version);
const data = store._raw();
const dataVersion =
data["--version"] && typeof data["--version"] === "number"
? data["--version"]
: 0;
// Find which versions need to be used for migrations
const relevantVersions = versions.filter((v) => v.version >= dataVersion);
// Migrate over each version
let mostRecentData = data;
for (const version of relevantVersions) {
if (version.migrate)
mostRecentData = await version.migrate(mostRecentData);
} }
// updates succesful, return store.save(mostRecentData);
return obj;
}
function get(this: any) {
// get from storage api
const store = this;
let gottenData: any = localStorage.getItem(this.id);
// parse json if item exists
if (gottenData) {
try {
gottenData = JSON.parse(gottenData);
if (!gottenData.constructor) {
console.error(
`Storage item for store ${this.id} has not constructor`
);
throw new Error("storage item has no constructor");
}
if (gottenData.constructor !== Object) {
console.error(`Storage item for store ${this.id} is not an object`);
throw new Error("storage item is not an object");
}
} catch (_) {
// if errored, set to null so it generates new one, see below
console.error(`Failed to parse storage item for store ${this.id}`);
gottenData = null;
} }
} }
// if item doesnt exist, generate from version init function buildStorageObject<T>(store: InternalStoreData): StoreRet<T> {
if (!gottenData) { const key = store.key ?? "";
gottenData = this.versions[this.currentVersion.toString()].init(); const latestVersion = store.versions.sort((a, b) => b.version - a.version)[0];
}
// update the data if needed function onChange(cb: (data: T) => void) {
gottenData = this.update(gottenData); if (!storeCallbacks[key]) storeCallbacks[key] = [];
storeCallbacks[key].push(cb);
// add a save object to return value
gottenData.save = function save(newData: any) {
const dataToStore = newData || gottenData;
localStorage.setItem(store.id, JSON.stringify(dataToStore));
};
// add instance helpers
Object.entries(d.instanceHelpers).forEach(([name, helper]: any) => {
if (gottenData[name] !== undefined)
throw new Error(
`helper name: ${name} on instance of store ${this.id} is reserved`
);
gottenData[name] = helper.bind(gottenData);
});
// return data
return gottenData;
}
// add functions to store
data.get = get.bind(data);
data.update = update.bind(data);
// add static helpers
Object.entries(d.staticHelpers).forEach(([name, helper]: any) => {
if (data[name] !== undefined)
throw new Error(`helper name: ${name} on store ${data.id} is reserved`);
data[name] = helper.bind({});
});
return data;
}
/*
* Builds a versioned store
*
* manages versioning of localstorage items
*/
export function versionedStoreBuilder(): any {
return { return {
_data: { destroy() {
versionList: [], // remove function pointer from callbacks
maxVersion: 0, storeCallbacks[key] = storeCallbacks[key].filter((v) => v === cb);
versions: {},
storageString: undefined,
instanceHelpers: {},
staticHelpers: {},
}, },
};
setKey(str: string) {
this._data.storageString = str;
return this;
},
addVersion({ version, migrate, create }: any) {
// input checking
if (version < 0) throw new Error("Cannot add version below 0 in store");
if (version > 0 && !migrate)
throw new Error(
`Missing migration on version ${version} (needed for any version above 0)`
);
// update max version list
if (version > this._data.maxVersion) this._data.maxVersion = version;
// add to version list
this._data.versionList.push(version);
// register version
this._data.versions[version.toString()] = {
version, // version number
update: migrate
? (data: any) => {
// update function, and increment version
const newData = migrate(data);
newData["--version"] = version; // eslint-disable-line no-param-reassign
return newData;
} }
: undefined,
init: create function makeRaw() {
? () => { const data = latestVersion.create?.() ?? {};
// return an initial object data["--version"] = latestVersion.version;
const data = create();
data["--version"] = version;
return data; return data;
} }
: undefined,
function getRaw() {
const item = localStorage.getItem(key);
if (!item) return makeRaw();
try {
return JSON.parse(item);
} catch (err) {
// we assume user has fucked with the data, give them a fresh store
console.error(`FAILED TO PARSE LOCALSTORAGE FOR KEY ${key}`, err);
return makeRaw();
}
}
function save(data: T) {
const withVersion: any = { ...data };
withVersion["--version"] = latestVersion.version;
localStorage.setItem(key, JSON.stringify(withVersion));
if (!storeCallbacks[key]) storeCallbacks[key] = [];
storeCallbacks[key].forEach((v) => v(structuredClone(data)));
}
return {
get() {
const data = getRaw();
delete data["--version"];
return data as T;
},
_raw() {
return getRaw();
},
onChange,
save,
}; };
return this;
},
registerHelper({ name, helper, type }: any) {
// type
let helperType: string = type;
if (!helperType) helperType = "instance";
// input checking
if (!name || name.constructor !== String) {
throw new Error("helper name is not a string");
}
if (!helper || helper.constructor !== Function) {
throw new Error("helper function is not a function");
}
if (!["instance", "static"].includes(helperType)) {
throw new Error("helper type must be either 'instance' or 'static'");
} }
// register helper function assertStore(store: InternalStoreData) {
if (helperType === "instance") const versionListSorted = store.versions.sort(
this._data.instanceHelpers[name as string] = helper; (a, b) => a.version - b.version
else if (helperType === "static")
this._data.staticHelpers[name as string] = helper;
return this;
},
build() {
// check if version list doesnt skip versions
const versionListSorted = this._data.versionList.sort(
(a: number, b: number) => a - b
); );
versionListSorted.forEach((v: any, i: number, arr: any[]) => { versionListSorted.forEach((v, i, arr) => {
if (i === 0) return; if (i === 0) return;
if (v !== arr[i - 1] + 1) if (v.version !== arr[i - 1].version + 1)
throw new Error("Version list of store is not incremental"); throw new Error("Version list of store is not incremental");
}); });
versionListSorted.forEach((v) => {
if (v.version < 0) throw new Error("Versions cannot be negative");
});
// version zero must exist // version zero must exist
if (versionListSorted[0] !== 0) if (versionListSorted[0]?.version !== 0)
throw new Error("Version 0 doesn't exist in version list of store"); throw new Error("Version 0 doesn't exist in version list of store");
// max version must have init function // max version must have create function
if (!this._data.versions[this._data.maxVersion.toString()].init) if (!store.versions[store.versions.length - 1].create)
throw new Error( throw new Error(`Missing create function on latest version of store`);
`Missing create function on version ${this._data.maxVersion} (needed for latest version of store)`
);
// check storage string // check storage string
if (!this._data.storageString) if (!store.key) throw new Error("storage key not set in store");
throw new Error("storage key not set in store");
// build versioned store // check if all parts have migratio
return buildStoreObject(this._data); const migrations = [...versionListSorted];
migrations.pop();
migrations.forEach((v) => {
if (!v.migrate)
throw new Error(`Migration missing on version ${v.version}`);
});
}
export function createVersionedStore<T>(): StoreBuilder<T> {
const _data: InternalStoreData = {
versions: [],
key: null,
};
return {
setKey(key) {
_data.key = key;
return this;
},
addVersion(ver) {
_data.versions.push(ver);
return this;
},
build() {
assertStore(_data);
const storageObject = buildStorageObject<T>(_data);
stores[_data.key ?? ""] = [storageObject, _data];
return storageObject;
}, },
}; };
} }
export function useStore<T>(
store: StoreRet<T>
): [T, (cb: (old: T) => T) => void] {
const [data, setData] = useState<T>(store.get());
useEffect(() => {
const { destroy } = store.onChange((newData) => {
setData(newData);
});
return () => {
destroy();
};
}, [store]);
function setNewData(cb: (old: T) => T) {
const newData = cb(data);
store.save(newData);
}
return [data, setNewData];
}

View file

@ -1,6 +1,10 @@
import { versionedStoreBuilder } from "@/utils/storage"; import { createVersionedStore } from "@/utils/storage";
export const volumeStore = versionedStoreBuilder() interface VolumeStoreData {
volume: number;
}
export const volumeStore = createVersionedStore<VolumeStoreData>()
.setKey("mw-volume") .setKey("mw-volume")
.addVersion({ .addVersion({
version: 0, version: 0,
@ -18,8 +22,7 @@ export function getStoredVolume(): number {
} }
export function setStoredVolume(volume: number) { export function setStoredVolume(volume: number) {
const store = volumeStore.get(); volumeStore.save({
store.save({
volume, volume,
}); });
} }