DataStore.ts 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384
  1. /* eslint-disable @typescript-eslint/no-explicit-any */
  2. import type { Prettify } from "./types.js";
  3. //#region types
  4. /** Function that takes the data in the old format and returns the data in the new format. Also supports an asynchronous migration. */
  5. type MigrationFunc = (oldData: any) => any | Promise<any>;
  6. /** Dictionary of format version numbers and the function that migrates to them from the previous whole integer */
  7. export type DataMigrationsDict = Record<number, MigrationFunc>;
  8. /** Options for the DataStore instance */
  9. export type DataStoreOptions<TData> = Prettify<
  10. & {
  11. /**
  12. * A unique internal ID for this data store.
  13. * To avoid conflicts with other scripts, it is recommended to use a prefix that is unique to your script.
  14. * If you want to change the ID, you should make use of the {@linkcode DataStore.migrateId()} method.
  15. */
  16. id: string;
  17. /**
  18. * The default data object to use if no data is saved in persistent storage yet.
  19. * Until the data is loaded from persistent storage with {@linkcode DataStore.loadData()}, this will be the data returned by {@linkcode DataStore.getData()}.
  20. *
  21. * ⚠️ This has to be an object that can be serialized to JSON using `JSON.stringify()`, so no functions or circular references are allowed, they will cause unexpected behavior.
  22. */
  23. defaultData: TData;
  24. /**
  25. * An incremental, whole integer version number of the current format of data.
  26. * If the format of the data is changed in any way, this number should be incremented, in which case all necessary functions of the migrations dictionary will be run consecutively.
  27. *
  28. * ⚠️ Never decrement this number and optimally don't skip any numbers either!
  29. */
  30. formatVersion: number;
  31. /**
  32. * A dictionary of functions that can be used to migrate data from older versions to newer ones.
  33. * The keys of the dictionary should be the format version that the functions can migrate to, from the previous whole integer value.
  34. * The values should be functions that take the data in the old format and return the data in the new format.
  35. * The functions will be run in order from the oldest to the newest version.
  36. * If the current format version is not in the dictionary, no migrations will be run.
  37. */
  38. migrations?: DataMigrationsDict;
  39. /**
  40. * If an ID or multiple IDs are passed here, the data will be migrated from the old ID(s) to the current ID.
  41. * This will happen once per page load, when {@linkcode DataStore.loadData()} is called.
  42. * All future calls to {@linkcode DataStore.loadData()} in the session will not check for the old ID(s) anymore.
  43. * To migrate IDs manually, use the method {@linkcode DataStore.migrateId()} instead.
  44. */
  45. migrateIds?: string | string[];
  46. /**
  47. * Where the data should be saved (`"GM"` by default).
  48. * The protected methods {@linkcode DataStore.getValue()}, {@linkcode DataStore.setValue()} and {@linkcode DataStore.deleteValue()} are used to interact with the storage.
  49. * `"GM"` storage, `"localStorage"` and `"sessionStorage"` are supported out of the box, but in an extended class you can overwrite those methods to implement any other storage method.
  50. */
  51. storageMethod?: "GM" | "localStorage" | "sessionStorage";
  52. }
  53. & (
  54. | {
  55. /**
  56. * Function to use to encode the data prior to saving it in persistent storage.
  57. * If this is specified, make sure to declare {@linkcode decodeData()} as well.
  58. *
  59. * You can make use of UserUtils' [`compress()`](https://github.com/Sv443-Network/UserUtils?tab=readme-ov-file#compress) function here to make the data use up less space at the cost of a little bit of performance.
  60. * @param data The input data as a serialized object (JSON string)
  61. */
  62. encodeData: (data: string) => string | Promise<string>,
  63. /**
  64. * Function to use to decode the data after reading it from persistent storage.
  65. * If this is specified, make sure to declare {@linkcode encodeData()} as well.
  66. *
  67. * You can make use of UserUtils' [`decompress()`](https://github.com/Sv443-Network/UserUtils?tab=readme-ov-file#decompress) function here to make the data use up less space at the cost of a little bit of performance.
  68. * @returns The resulting data as a valid serialized object (JSON string)
  69. */
  70. decodeData: (data: string) => string | Promise<string>,
  71. }
  72. | {
  73. encodeData?: never,
  74. decodeData?: never,
  75. }
  76. )
  77. >;
  78. //#region class
  79. /**
  80. * Manages a hybrid synchronous & asynchronous persistent JSON database that is cached in memory and persistently saved across sessions using [GM storage](https://wiki.greasespot.net/GM.setValue) (default), [localStorage](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage) or [sessionStorage](https://developer.mozilla.org/en-US/docs/Web/API/Window/sessionStorage).
  81. * Supports migrating data from older format versions to newer ones and populating the cache with default data if no persistent data is found.
  82. * Can be overridden to implement any other storage method.
  83. *
  84. * All methods are at least `protected`, so you can easily extend this class and overwrite them to use a different storage method or to add additional functionality.
  85. * Remember that you can call `super.methodName()` in the subclass to access the original method.
  86. *
  87. * ⚠️ The stored data has to be **serializable using `JSON.stringify()`**, meaning no undefined, circular references, etc. are allowed.
  88. * ⚠️ If the storageMethod is left as the default of `"GM"` the directives `@grant GM.getValue` and `@grant GM.setValue` are required. If you then also use the method {@linkcode DataStore.deleteData()}, the extra directive `@grant GM.deleteValue` is needed too.
  89. * ⚠️ Make sure to call {@linkcode loadData()} at least once after creating an instance, or the returned data will be the same as `options.defaultData`
  90. *
  91. * @template TData The type of the data that is saved in persistent storage for the currently set format version (will be automatically inferred from `defaultData` if not provided)
  92. */
  93. export class DataStore<TData extends object = object> {
  94. public readonly id: string;
  95. public readonly formatVersion: number;
  96. public readonly defaultData: TData;
  97. public readonly encodeData: DataStoreOptions<TData>["encodeData"];
  98. public readonly decodeData: DataStoreOptions<TData>["decodeData"];
  99. public readonly storageMethod: Required<DataStoreOptions<TData>>["storageMethod"];
  100. private cachedData: TData;
  101. private migrations?: DataMigrationsDict;
  102. private migrateIds: string[] = [];
  103. /**
  104. * Creates an instance of DataStore to manage a sync & async database that is cached in memory and persistently saved across sessions.
  105. * Supports migrating data from older versions to newer ones and populating the cache with default data if no persistent data is found.
  106. *
  107. * ⚠️ Requires the directives `@grant GM.getValue` and `@grant GM.setValue` if the storageMethod is left as the default of `"GM"`
  108. * ⚠️ Make sure to call {@linkcode loadData()} at least once after creating an instance, or the returned data will be the same as `options.defaultData`
  109. *
  110. * @template TData The type of the data that is saved in persistent storage for the currently set format version (will be automatically inferred from `defaultData` if not provided) - **This has to be a JSON-compatible object!** (no undefined, circular references, etc.)
  111. * @param options The options for this DataStore instance
  112. */
  113. constructor(options: DataStoreOptions<TData>) {
  114. this.id = options.id;
  115. this.formatVersion = options.formatVersion;
  116. this.defaultData = options.defaultData;
  117. this.cachedData = options.defaultData;
  118. this.migrations = options.migrations;
  119. if(options.migrateIds)
  120. this.migrateIds = Array.isArray(options.migrateIds) ? options.migrateIds : [options.migrateIds];
  121. this.storageMethod = options.storageMethod ?? "GM";
  122. this.encodeData = options.encodeData;
  123. this.decodeData = options.decodeData;
  124. }
  125. //#region public
  126. /**
  127. * Loads the data saved in persistent storage into the in-memory cache and also returns it.
  128. * Automatically populates persistent storage with default data if it doesn't contain any data yet.
  129. * Also runs all necessary migration functions if the data format has changed since the last time the data was saved.
  130. */
  131. public async loadData(): Promise<TData> {
  132. try {
  133. if(this.migrateIds.length > 0) {
  134. await this.migrateId(this.migrateIds);
  135. this.migrateIds = [];
  136. }
  137. const gmData = await this.getValue(`_uucfg-${this.id}`, JSON.stringify(this.defaultData));
  138. let gmFmtVer = Number(await this.getValue(`_uucfgver-${this.id}`, NaN));
  139. if(typeof gmData !== "string") {
  140. await this.saveDefaultData();
  141. return { ...this.defaultData };
  142. }
  143. const isEncoded = Boolean(await this.getValue(`_uucfgenc-${this.id}`, false));
  144. let saveData = false;
  145. if(isNaN(gmFmtVer)) {
  146. await this.setValue(`_uucfgver-${this.id}`, gmFmtVer = this.formatVersion);
  147. saveData = true;
  148. }
  149. let parsed = await this.deserializeData(gmData, isEncoded);
  150. if(gmFmtVer < this.formatVersion && this.migrations)
  151. parsed = await this.runMigrations(parsed, gmFmtVer);
  152. if(saveData)
  153. await this.setData(parsed);
  154. this.cachedData = { ...parsed };
  155. return this.cachedData;
  156. }
  157. catch(err) {
  158. console.warn("Error while parsing JSON data, resetting it to the default value.", err);
  159. await this.saveDefaultData();
  160. return this.defaultData;
  161. }
  162. }
  163. /**
  164. * Returns a copy of the data from the in-memory cache.
  165. * Use {@linkcode loadData()} to get fresh data from persistent storage (usually not necessary since the cache should always exactly reflect persistent storage).
  166. * @param deepCopy Whether to return a deep copy of the data (default: `false`) - only necessary if your data object is nested and may have a bigger performance impact if enabled
  167. */
  168. public getData(deepCopy = false): TData {
  169. return deepCopy
  170. ? this.deepCopy(this.cachedData)
  171. : { ...this.cachedData };
  172. }
  173. /** Saves the data synchronously to the in-memory cache and asynchronously to the persistent storage */
  174. public setData(data: TData): Promise<void> {
  175. this.cachedData = data;
  176. const useEncoding = this.encodingEnabled();
  177. return new Promise<void>(async (resolve) => {
  178. await Promise.all([
  179. this.setValue(`_uucfg-${this.id}`, await this.serializeData(data, useEncoding)),
  180. this.setValue(`_uucfgver-${this.id}`, this.formatVersion),
  181. this.setValue(`_uucfgenc-${this.id}`, useEncoding),
  182. ]);
  183. resolve();
  184. });
  185. }
  186. /** Saves the default data passed in the constructor synchronously to the in-memory cache and asynchronously to persistent storage */
  187. public async saveDefaultData(): Promise<void> {
  188. this.cachedData = this.defaultData;
  189. const useEncoding = this.encodingEnabled();
  190. return new Promise<void>(async (resolve) => {
  191. await Promise.all([
  192. this.setValue(`_uucfg-${this.id}`, await this.serializeData(this.defaultData, useEncoding)),
  193. this.setValue(`_uucfgver-${this.id}`, this.formatVersion),
  194. this.setValue(`_uucfgenc-${this.id}`, useEncoding),
  195. ]);
  196. resolve();
  197. });
  198. }
  199. /**
  200. * Call this method to clear all persistently stored data associated with this DataStore instance.
  201. * The in-memory cache will be left untouched, so you may still access the data with {@linkcode getData()}
  202. * Calling {@linkcode loadData()} or {@linkcode setData()} after this method was called will recreate persistent storage with the cached or default data.
  203. *
  204. * ⚠️ This requires the additional directive `@grant GM.deleteValue` if the storageMethod is left as the default of `"GM"`
  205. */
  206. public async deleteData(): Promise<void> {
  207. await Promise.all([
  208. this.deleteValue(`_uucfg-${this.id}`),
  209. this.deleteValue(`_uucfgver-${this.id}`),
  210. this.deleteValue(`_uucfgenc-${this.id}`),
  211. ]);
  212. }
  213. /** Returns whether encoding and decoding are enabled for this DataStore instance */
  214. public encodingEnabled(): this is Required<Pick<DataStoreOptions<TData>, "encodeData" | "decodeData">> {
  215. return Boolean(this.encodeData && this.decodeData);
  216. }
  217. //#region migrations
  218. /**
  219. * Runs all necessary migration functions consecutively and saves the result to the in-memory cache and persistent storage and also returns it.
  220. * This method is automatically called by {@linkcode loadData()} if the data format has changed since the last time the data was saved.
  221. * Though calling this method manually is not necessary, it can be useful if you want to run migrations for special occasions like a user importing potentially outdated data that has been previously exported.
  222. *
  223. * If one of the migrations fails, the data will be reset to the default value if `resetOnError` is set to `true` (default). Otherwise, an error will be thrown and no data will be saved.
  224. */
  225. public async runMigrations(oldData: any, oldFmtVer: number, resetOnError = true): Promise<TData> {
  226. if(!this.migrations)
  227. return oldData as TData;
  228. let newData = oldData;
  229. const sortedMigrations = Object.entries(this.migrations)
  230. .sort(([a], [b]) => Number(a) - Number(b));
  231. let lastFmtVer = oldFmtVer;
  232. for(const [fmtVer, migrationFunc] of sortedMigrations) {
  233. const ver = Number(fmtVer);
  234. if(oldFmtVer < this.formatVersion && oldFmtVer < ver) {
  235. try {
  236. const migRes = migrationFunc(newData);
  237. newData = migRes instanceof Promise ? await migRes : migRes;
  238. lastFmtVer = oldFmtVer = ver;
  239. }
  240. catch(err) {
  241. if(!resetOnError)
  242. throw new Error(`Error while running migration function for format version '${fmtVer}'`);
  243. console.error(`Error while running migration function for format version '${fmtVer}' - resetting to the default value.`, err);
  244. await this.saveDefaultData();
  245. return this.getData();
  246. }
  247. }
  248. }
  249. await Promise.all([
  250. this.setValue(`_uucfg-${this.id}`, await this.serializeData(newData)),
  251. this.setValue(`_uucfgver-${this.id}`, lastFmtVer),
  252. this.setValue(`_uucfgenc-${this.id}`, this.encodingEnabled()),
  253. ]);
  254. return this.cachedData = { ...newData as TData };
  255. }
  256. /**
  257. * Tries to migrate the currently saved persistent data from one or more old IDs to the ID set in the constructor.
  258. * If no data exist for the old ID(s), nothing will be done, but some time may still pass trying to fetch the non-existent data.
  259. */
  260. public async migrateId(oldIds: string | string[]): Promise<void> {
  261. const ids = Array.isArray(oldIds) ? oldIds : [oldIds];
  262. await Promise.all(ids.map(async id => {
  263. const data = await this.getValue(`_uucfg-${id}`, JSON.stringify(this.defaultData));
  264. const fmtVer = Number(await this.getValue(`_uucfgver-${id}`, NaN));
  265. const isEncoded = Boolean(await this.getValue(`_uucfgenc-${id}`, false));
  266. if(data === undefined || isNaN(fmtVer))
  267. return;
  268. const parsed = await this.deserializeData(data, isEncoded);
  269. await Promise.allSettled([
  270. this.setValue(`_uucfg-${this.id}`, await this.serializeData(parsed)),
  271. this.setValue(`_uucfgver-${this.id}`, fmtVer),
  272. this.setValue(`_uucfgenc-${this.id}`, isEncoded),
  273. this.deleteValue(`_uucfg-${id}`),
  274. this.deleteValue(`_uucfgver-${id}`),
  275. this.deleteValue(`_uucfgenc-${id}`),
  276. ]);
  277. }));
  278. }
  279. //#region serialization
  280. /** Serializes the data using the optional this.encodeData() and returns it as a string */
  281. protected async serializeData(data: TData, useEncoding = true): Promise<string> {
  282. const stringData = JSON.stringify(data);
  283. if(!this.encodingEnabled() || !useEncoding)
  284. return stringData;
  285. const encRes = this.encodeData(stringData);
  286. if(encRes instanceof Promise)
  287. return await encRes;
  288. return encRes;
  289. }
  290. /** Deserializes the data using the optional this.decodeData() and returns it as a JSON object */
  291. protected async deserializeData(data: string, useEncoding = true): Promise<TData> {
  292. let decRes = this.encodingEnabled() && useEncoding ? this.decodeData(data) : undefined;
  293. if(decRes instanceof Promise)
  294. decRes = await decRes;
  295. return JSON.parse(decRes ?? data) as TData;
  296. }
  297. //#region misc
  298. /** Copies a JSON-compatible object and loses all its internal references in the process */
  299. protected deepCopy<T>(obj: T): T {
  300. return JSON.parse(JSON.stringify(obj));
  301. }
  302. //#region storage
  303. /** Gets a value from persistent storage - can be overwritten in a subclass if you want to use something other than GM storage */
  304. protected async getValue<TValue extends GM.Value = string>(name: string, defaultValue: TValue): Promise<string | TValue> {
  305. switch(this.storageMethod) {
  306. case "localStorage":
  307. return localStorage.getItem(name) as TValue ?? defaultValue;
  308. case "sessionStorage":
  309. return sessionStorage.getItem(name) as string ?? defaultValue;
  310. default:
  311. return GM.getValue<TValue>(name, defaultValue);
  312. }
  313. }
  314. /**
  315. * Sets a value in persistent storage - can be overwritten in a subclass if you want to use something other than GM storage.
  316. * The default storage engines will stringify all passed values like numbers or booleans, so be aware of that.
  317. */
  318. protected async setValue(name: string, value: GM.Value): Promise<void> {
  319. switch(this.storageMethod) {
  320. case "localStorage":
  321. return localStorage.setItem(name, String(value));
  322. case "sessionStorage":
  323. return sessionStorage.setItem(name, String(value));
  324. default:
  325. return GM.setValue(name, String(value));
  326. }
  327. }
  328. /** Deletes a value from persistent storage - can be overwritten in a subclass if you want to use something other than GM storage */
  329. protected async deleteValue(name: string): Promise<void> {
  330. switch(this.storageMethod) {
  331. case "localStorage":
  332. return localStorage.removeItem(name);
  333. case "sessionStorage":
  334. return sessionStorage.removeItem(name);
  335. default:
  336. return GM.deleteValue(name);
  337. }
  338. }
  339. }