diff --git a/compiler/compiler.ts b/compiler/compiler.ts index ab801a09..4ff3aca7 100644 --- a/compiler/compiler.ts +++ b/compiler/compiler.ts @@ -2021,8 +2021,7 @@ export class Compiler { SCOPE.addJSTypeDefs = receiver != Runtime.endpoint && receiver != LOCAL_ENDPOINT; } - // add js type module only if http(s) url - const addTypeDefs = SCOPE.addJSTypeDefs && jsTypeDefModule && (jsTypeDefModule.toString().startsWith("http://") || jsTypeDefModule.toString().startsWith("https://")); + const addTypeDefs = SCOPE.addJSTypeDefs && jsTypeDefModule; if (addTypeDefs) { Compiler.builder.handleRequiredBufferSize(SCOPE.b_index+4, SCOPE); diff --git a/docs/manual/04 Pointer Synchronisation.md b/docs/manual/04 Pointer Synchronisation.md index 90d34cdb..a0dbe07f 100644 --- a/docs/manual/04 Pointer Synchronisation.md +++ b/docs/manual/04 Pointer Synchronisation.md @@ -167,3 +167,39 @@ const x2 = await $.ABCDEF assert (x1 === x2) ``` +The identity of a pointer is always preserved. Even if you receive the pointer +from a remote endpoint call, it is still identical to the local instance: + +```ts +// remote endpoint + +type User = {name: string, age: number} + +const happyBirthday = $$( + function (user: User) { + user.age++; + return user; + } +) +``` + +```ts +// local endpoint + +const user = $$({ + name: "Luke", + age: 20 +}) + +const happyBirthday = await $.DEFABCDEF // <- pointer id for the remote "happyBirthday" function +const olderUser = await happyBirthday(user); + +user === olderUser // true +olderUser.age === 21 // true +user.age === 21 // true +``` + +Like we would expect if this was a normal, local JavaScript function call, the returned +`user` object is identical to the object we passed to the function. +This is not only true for this simple example, but also for more complex scenarios. +For example, reference identities are also preserved within [eternal values](./05%20Eternal%20Pointers.md). \ No newline at end of file diff --git a/init.ts b/init.ts index 0d3fc566..5dd25a28 100644 --- a/init.ts +++ b/init.ts @@ -13,12 +13,34 @@ import { verboseArg } from "./utils/logger.ts"; import { MessageLogger } from "./utils/message_logger.ts"; +/** + * Load the subscriber cache from storage and + * reset if a remote pointer is required + */ +async function initSubscriberCache() { + try { + Runtime.subscriber_cache = (await Storage.loadOrCreate( + "Datex.Runtime.SUBSCRIBER_CACHE", + () => new Map(), + {onlyLocalPointers: true} + )).setAutoDefault(Set); + } + catch (e) { + logger.debug("resetting subscriber cache (" + e?.message + ")") + Runtime.subscriber_cache = (await Storage.loadOrCreate( + "Datex.Runtime.SUBSCRIBER_CACHE", + () => new Map(), + undefined, + true + )).setAutoDefault(Set); + } +} + /** * Runtime init (sets ENV, storage, endpoint, ...) */ export async function init() { - // register DatexStorage as pointer source registerStorageAsPointerSource(); // default storage config: @@ -56,7 +78,11 @@ export async function init() { Runtime.onEndpointChanged((endpoint) => { Pointer.pointer_prefix = endpoint.getPointerPrefix(); // has only local endpoint id (%0000) or global id? - if (endpoint != LOCAL_ENDPOINT) Pointer.is_local = false; + if (endpoint != LOCAL_ENDPOINT) { + Pointer.is_local = false; + // init subscriber cache as soon as endpoint is available + initSubscriberCache(); + } else Pointer.is_local = true; }) @@ -105,26 +131,6 @@ export async function init() { // init persistent memory Runtime.persistent_memory = (await Storage.loadOrCreate("Datex.Runtime.MEMORY", ()=>new Map())).setAutoDefault(Object); - // init persistent subscriber cache - (async () => { - try { - Runtime.subscriber_cache = (await Storage.loadOrCreate( - "Datex.Runtime.SUBSCRIBER_CACHE", - () => new Map(), - {onlyLocalPointers: true} - )).setAutoDefault(Set); - } - catch (e) { - logger.debug("resetting subscriber cache (" + e?.message + ")") - Runtime.subscriber_cache = (await Storage.loadOrCreate( - "Datex.Runtime.SUBSCRIBER_CACHE", - () => new Map(), - undefined, - true - )).setAutoDefault(Set); - } - })() - if (!globalThis.NO_INIT) { Runtime.init(); diff --git a/runtime/pointers.ts b/runtime/pointers.ts index fb38720c..520c3f2c 100644 --- a/runtime/pointers.ts +++ b/runtime/pointers.ts @@ -1199,7 +1199,8 @@ export class Pointer extends Ref { // update pointer ids if no longer local if (!this.#is_local) { for (const pointer of this.#local_pointers) { - pointer.id = Pointer.getUniquePointerID(pointer); + // still local? + if (pointer.origin == LOCAL_ENDPOINT) pointer.id = Pointer.getUniquePointerID(pointer); } this.#local_pointers.clear(); } @@ -1395,13 +1396,16 @@ export class Pointer extends Ref { } if (stored!=NOT_EXISTING) { - // if the value is a pointer with a tranform scope, copy the transform, not the value (TODO still just a workaround to preserve transforms in storage, maybe better solution?) - if (stored instanceof Pointer && stored.transform_scope) { - await pointer.handleTransformAsync(stored.transform_scope.internal_vars, stored.transform_scope); + // set value if pointer still not loaded during source.getPointer + if (!pointer.#loaded) { + // if the value is a pointer with a tranform scope, copy the transform, not the value (TODO still just a workaround to preserve transforms in storage, maybe better solution?) + if (stored instanceof Pointer && stored.transform_scope) { + await pointer.handleTransformAsync(stored.transform_scope.internal_vars, stored.transform_scope); + } + // set normal value + else pointer = pointer.setValue(stored); } - // set normal value - else pointer = pointer.setValue(stored); - + // now sync if source (pointer storage) can sync pointer if (source?.syncPointer) source.syncPointer(pointer); diff --git a/runtime/storage.ts b/runtime/storage.ts index 200d81ab..0c465bd8 100644 --- a/runtime/storage.ts +++ b/runtime/storage.ts @@ -12,6 +12,9 @@ import { displayFatalError } from "./display.ts" import { Type } from "../types/type.ts"; import { addPersistentListener } from "../utils/persistent-listeners.ts"; import { LOCAL_ENDPOINT } from "../types/addressing.ts"; +import { ESCAPE_SEQUENCES } from "../utils/logger.ts"; +import { StorageMap } from "../types/storage_map.ts"; +import { StorageSet } from "../types/storage_set.ts"; // displayInit(); @@ -123,6 +126,10 @@ type storage_options = { type storage_location_options = L extends StorageLocation ? storage_options : never +type StorageSnapshotOptions = { + internalItems: boolean, + expandStorageMapsAndSets: boolean +} export class Storage { @@ -135,7 +142,9 @@ export class Storage { static item_prefix = "dxitem::"+site_suffix+"::" static meta_prefix = "dxmeta::"+site_suffix+"::" - + static rc_prefix = "rc::" + static pointer_deps_prefix = "deps::dxptr::" + static item_deps_prefix = "deps::dxitem::" static #storage_active_pointers = new Set(); static #storage_active_pointer_ids = new Set(); @@ -371,6 +380,7 @@ export class Storage { static setItem(key:string, value:any, listen_for_pointer_changes = true, location:StorageLocation|null|undefined = this.#primary_location):Promise|boolean { Storage.cache.set(key, value); // save in cache + // cache deletion does not work, problems with storage item backup // setTimeout(()=>Storage.cache.delete(key), 10000); @@ -385,6 +395,7 @@ export class Storage { this.setDirty(location, true) // store value (might be pointer reference) const dependencies = await location.setItem(key, value); + this.updateItemDependencies(key, [...dependencies].map(p=>p.id)); await this.saveDependencyPointersAsync(dependencies, listen_for_pointer_changes, location); this.setDirty(location, false) return true; @@ -392,6 +403,7 @@ export class Storage { static setItemSync(location:SyncStorageLocation, key: string,value: unknown,listen_for_pointer_changes: boolean) { const dependencies = location.setItem(key, value); + this.updateItemDependencies(key, [...dependencies].map(p=>p.id)); this.saveDependencyPointersSync(dependencies, listen_for_pointer_changes, location); return true; } @@ -416,6 +428,7 @@ export class Storage { const dependencies = this.updatePointerSync(location, pointer, partialUpdateKey); dependencies.delete(pointer); + this.updatePointerDependencies(pointer.id, [...dependencies].map(p=>p.id)); this.saveDependencyPointersSync(dependencies, listen_for_changes, location); // listen for changes @@ -435,6 +448,7 @@ export class Storage { const dependencies = await this.updatePointerAsync(location, pointer, partialUpdateKey); dependencies.delete(pointer); + this.updatePointerDependencies(pointer.id, [...dependencies].map(p=>p.id)); await this.saveDependencyPointersAsync(dependencies, listen_for_changes, location); // listen for changes @@ -634,15 +648,22 @@ export class Storage { // create pointer with saved id and value + start syncing, if pointer not already created in DATEX if (pointerify) { - let pointer: Pointer; + let pointer = Pointer.get(pointer_id) // if the value is a pointer with a tranform scope, copy the transform, not the value (TODO still just a workaround to preserve transforms in storage, maybe better solution?) if (val instanceof Pointer && val.transform_scope) { console.log("init value",val); pointer = await Pointer.createTransformAsync(val.transform_scope.internal_vars, val.transform_scope); } - // normal pointer from value - else pointer = Pointer.create(pointer_id, val, false, Runtime.endpoint); + // set value of existing pointer + else if (pointer) { + if (pointer.value_initialized) logger.warn("pointer value " + pointer.idString() + " already initialized, setting new value from storage"); + pointer = pointer.setValue(val); + } + // create new pointer from value + else { + pointer = Pointer.create(pointer_id, val, false, Runtime.endpoint); + } this.syncPointer(pointer); this.#storage_active_pointers.add(pointer); @@ -656,12 +677,25 @@ export class Storage { } } - private static async removePointer(pointer_id:string, location?:StorageLocation) { - // remove from specific location + private static async removePointer(pointer_id:string, location?:StorageLocation, force_remove = false) { + if (!force_remove && this.getReferenceCount(pointer_id) > 0) { + logger.warn("Cannot remove pointer $" + pointer_id + ", still referenced"); + return; + } + logger.debug("Removing pointer $" + pointer_id + " from storage" + (location ? " (" + location.name + ")" : "")); + + // remove from specific location if (location) return location.removePointer(pointer_id); // remove from all else { + // clear dependencies + this.updatePointerDependencies(pointer_id, []) + + const ptr = Pointer.get(pointer_id) + if (ptr) this.#storage_active_pointers.delete(ptr); + this.#storage_active_pointer_ids.delete(pointer_id); + for (const location of this.#locations.keys()) { await location.removePointer(pointer_id); } @@ -812,6 +846,7 @@ export class Storage { Storage.cache.set(key, val); await this.initItemFromTrustedLocation(key, val, location) + return val; } @@ -838,18 +873,107 @@ export class Storage { } public static async removeItem(key:string, location?:StorageLocation):Promise { - if (Storage.cache.has(key)) Storage.cache.delete(key); // delete from cache + + logger.debug("Removing item '" + key + "' from storage" + (location ? " (" + location.name + ")" : "")) // remove from specific location if (location) return location.removeItem(key); // remove from all else { + Storage.cache.delete(key); // delete from cache + + // clear dependencies + this.updateItemDependencies(key, []) + for (const location of this.#locations.keys()) { await location.removeItem(key); } } } + /** + * Increase the reference count of a pointer in storage + */ + private static increaseReferenceCount(ptrId:string) { + localStorage.setItem(this.rc_prefix+ptrId, (this.getReferenceCount(ptrId) + 1).toString()); + } + /** + * Decrease the reference count of a pointer in storage + */ + private static decreaseReferenceCount(ptrId:string) { + const newCount = this.getReferenceCount(ptrId) - 1; + // RC is 0, delet pointer from storage + if (newCount <= 0) { + localStorage.removeItem(this.rc_prefix+ptrId); + this.removePointer(ptrId, undefined, true); + } + // decrease RC + else localStorage.setItem(this.rc_prefix+ptrId, newCount.toString()); + } + /** + * Get the current reference count of a pointer (number of entries that have a reference to this pointer) + * @returns + */ + private static getReferenceCount(ptrId:string) { + const entry = localStorage.getItem(this.rc_prefix+ptrId); + return entry ? Number(entry) : 0; + } + + private static addDependency(key:string, depPtrId:string, prefix:string) { + const uniqueKey = prefix+key; + if (localStorage.getItem(uniqueKey)?.includes(depPtrId)) return; + else localStorage.setItem(uniqueKey, (localStorage.getItem(uniqueKey) ?? "") + depPtrId); + } + private static removeDependency(key:string, depPtrId:string, prefix:string) { + const uniqueKey = prefix+key; + if (!localStorage.getItem(uniqueKey)?.includes(depPtrId)) return; + else { + const newDeps = localStorage.getItem(uniqueKey)!.replace(depPtrId, ""); + // remove key if no more dependencies + if (newDeps.length == 0) localStorage.removeItem(uniqueKey); + // remove dependency + else localStorage.setItem(uniqueKey, newDeps); + } + } + private static setDependencies(key:string, depPtrIds:string[], prefix:string) { + const uniqueKey = prefix+key; + if (!depPtrIds.length) localStorage.removeItem(uniqueKey); + else localStorage.setItem(uniqueKey, depPtrIds.join(",")); + } + + private static setItemDependencies(key:string, depPtrIds:string[]) { + this.setDependencies(key, depPtrIds, this.item_deps_prefix); + } + private static setPointerDependencies(key:string, depPtrIds:string[]) { + this.setDependencies(key, depPtrIds, this.pointer_deps_prefix); + } + private static getItemDependencies(key:string) { + const uniqueKey = this.item_deps_prefix+key; + return localStorage.getItem(uniqueKey)?.split(",") ?? []; + } + private static getPointerDependencies(key:string) { + const uniqueKey = this.pointer_deps_prefix+key; + return localStorage.getItem(uniqueKey)?.split(",") ?? []; + } + + private static updateItemDependencies(key:string, newDeps:string[]) { + const oldDeps = this.getItemDependencies(key); + const added = newDeps.filter(p=>!oldDeps.includes(p)); + const removed = oldDeps.filter(p=>!newDeps.includes(p)); + for (const ptrId of added) this.increaseReferenceCount(ptrId); + for (const ptrId of removed) this.decreaseReferenceCount(ptrId); + this.setItemDependencies(key, newDeps); + } + private static updatePointerDependencies(key:string, newDeps:string[]) { + const oldDeps = this.getPointerDependencies(key); + const added = newDeps.filter(p=>!oldDeps.includes(p)); + const removed = oldDeps.filter(p=>!newDeps.includes(p)); + for (const ptrId of added) this.increaseReferenceCount(ptrId); + for (const ptrId of removed) this.decreaseReferenceCount(ptrId); + this.setPointerDependencies(key, newDeps); + } + + public static clearAll(){ return this.clear() } @@ -903,6 +1027,199 @@ export class Storage { else throw new Error("Cannot find or create the state '" + id + "'") } + static #replaceLastOccurenceOf(search: string, replace: string, str: string) { + const n = str.lastIndexOf(search); + return str.substring(0, n) + replace + str.substring(n + search.length); + } + + public static async printSnapshot(options: StorageSnapshotOptions = {internalItems: false, expandStorageMapsAndSets: true}) { + const {items, pointers} = await this.getSnapshot(options); + + const COLOR_PTR = `\x1b[38;2;${[65,102,238].join(';')}m` + const COLOR_NUMBER = `\x1b[38;2;${[253,139,25].join(';')}m` + + let string = ESCAPE_SEQUENCES.BOLD+"Storage Locations\n\n"+ESCAPE_SEQUENCES.RESET + string += `${ESCAPE_SEQUENCES.ITALIC}A list of all currently used storage locations and their corresponding store strategies.\n${ESCAPE_SEQUENCES.RESET}` + + for (const [location, options] of this.#locations) { + string += `\n • ${location.name} ${ESCAPE_SEQUENCES.GREY}(${options.modes.map(m=>Storage.Mode[m]).join(', ')})${ESCAPE_SEQUENCES.RESET}` + } + + string += `\n\n${ESCAPE_SEQUENCES.BOLD}Trusted Location:${ESCAPE_SEQUENCES.RESET} ${this.#trusted_location?.name ?? "none"}` + string += `\n${ESCAPE_SEQUENCES.BOLD}Primary Location:${ESCAPE_SEQUENCES.RESET} ${this.#primary_location?.name ?? "none"}` + + console.log(string+"\n\n"); + + // pointers + string = ESCAPE_SEQUENCES.BOLD+"Pointers\n\n"+ESCAPE_SEQUENCES.RESET + string += `${ESCAPE_SEQUENCES.ITALIC}A list of all pointers stored in any storage location. Pointers are only stored as long as they are referenced somwhere else in the storage.\n\n${ESCAPE_SEQUENCES.RESET}` + + for (const [key, storageMap] of pointers.snapshot) { + // check if stored in all locations, otherwise print in which location it is stored (functional programming) + const locations = [...storageMap.keys()] + const storedInAll = [...this.#locations.keys()].every(l => locations.includes(l)); + + const value = [...storageMap.values()][0]; + string += ` • ${COLOR_PTR}$${key}${ESCAPE_SEQUENCES.GREY}${storedInAll ? "" : (` (only in ${locations.map(l=>l.name).join(",")})`)} = ${this.#replaceLastOccurenceOf("\n ", "", value.replaceAll("\n", "\n "))}\n` + } + console.log(string+"\n"); + + // items + string = ESCAPE_SEQUENCES.BOLD+"Items\n\n"+ESCAPE_SEQUENCES.RESET + string += `${ESCAPE_SEQUENCES.ITALIC}A list of all named items stored in any storage location.\n\n${ESCAPE_SEQUENCES.RESET}` + + for (const [key, storageMap] of items.snapshot) { + // check if stored in all locations, otherwise print in which location it is stored (functional programming) + const locations = [...storageMap.keys()] + const storedInAll = [...this.#locations.keys()].every(l => locations.includes(l)); + + const value = [...storageMap.values()][0]; + string += ` • ${key}${ESCAPE_SEQUENCES.GREY}${storedInAll ? "" : (` (only in ${locations.map(l=>l.name).join(",")})`)} = ${value}` + } + console.log(string+"\n"); + + // memory management + if (options?.internalItems) { + string = ESCAPE_SEQUENCES.BOLD+"Memory Management\n\n"+ESCAPE_SEQUENCES.RESET + string += `${ESCAPE_SEQUENCES.ITALIC}This section shows the reference count (rc::) of pointers and the dependencies (deps::) of items and pointers. The reference count of a pointer tracks the number of items and pointers that reference this pointer.\n\n${ESCAPE_SEQUENCES.RESET}` + let rc_string = "" + let item_deps_string = "" + let pointer_deps_string = "" + for (let i = 0, len = localStorage.length; i < len; ++i ) { + const key = localStorage.key(i)!; + if (key.startsWith(this.rc_prefix)) { + const ptrId = key.substring(this.rc_prefix.length); + const count = this.getReferenceCount(ptrId); + rc_string += `\x1b[0m • ${key} = ${COLOR_NUMBER}${count}\n` + } + else if (key.startsWith(this.item_deps_prefix)) { + const depsRaw = localStorage.getItem(key); + // single entry + if (!depsRaw?.includes(",")) { + item_deps_string += `\x1b[0m • ${key} = (${COLOR_PTR}${depsRaw}\x1b[0m)\n` + } + // multiple entries + else { + let deps = localStorage.getItem(key)!.split(",").join(`\x1b[0m,\n ${COLOR_PTR}$`) + if (deps) deps = ` ${COLOR_PTR}$`+deps + item_deps_string += `\x1b[0m • ${key} = (\n${COLOR_PTR}${deps}\x1b[0m\n )\n` + } + } + else if (key.startsWith(this.pointer_deps_prefix)) { + const depsRaw = localStorage.getItem(key); + // single entry + if (!depsRaw?.includes(",")) { + pointer_deps_string += `\x1b[0m • ${key} = (${COLOR_PTR}${depsRaw}\x1b[0m)\n` + } + // multiple entries + else { + let deps = localStorage.getItem(key)!.split(",").join(`\x1b[0m,\n ${COLOR_PTR}$`) + if (deps) deps = ` ${COLOR_PTR}$`+deps + pointer_deps_string += `\x1b[0m • ${key} = (\n${COLOR_PTR}${deps}\x1b[0m\n )\n` + } + } + } + + string += rc_string + "\n" + item_deps_string + "\n" + pointer_deps_string; + console.log(string+"\n"); + } + + // inconsistencies + if (pointers.inconsistencies.size > 0 || items.inconsistencies.size > 0) { + string = ESCAPE_SEQUENCES.BOLD+"Inconsistencies\n\n"+ESCAPE_SEQUENCES.RESET + string += `${ESCAPE_SEQUENCES.ITALIC}Inconsistencies between storage locations don't necessarily indicate that something is wrong. They can occur when a storage location is not updated immediately (e.g. when only using SAVE_ON_EXIT).\n\n${ESCAPE_SEQUENCES.RESET}` + for (const [key, storageMap] of pointers.inconsistencies) { + for (const [location, value] of storageMap) { + string += ` • ${COLOR_PTR}$${key}${ESCAPE_SEQUENCES.GREY} (${(location.name+")").padEnd(15, " ")} = ${this.#replaceLastOccurenceOf("\n ", "", value.replaceAll("\n", "\n "))}\n` + } + string += `\n` + } + for (const [key, storageMap] of items.inconsistencies) { + for (const [location, value] of storageMap) { + string += ` • ${key}${ESCAPE_SEQUENCES.GREY} (${(location.name+")").padEnd(15, " ")} = ${value}` + } + string += `\n` + } + + console.info(string+"\n"); + } + + + } + + public static async getSnapshot(options: StorageSnapshotOptions = {internalItems: false, expandStorageMapsAndSets: true}) { + const items = await this.createSnapshot(this.getItemKeys.bind(this), this.getItemDecompiled.bind(this)); + const pointers = await this.createSnapshot(this.getPointerKeys.bind(this), this.getPointerDecompiledFromLocation.bind(this)); + + // remove keys items that are unrelated to normal storage + for (const [key] of items.snapshot) { + if (key.startsWith("keys_")) items.snapshot.delete(key); + } + + // iterate over storage maps and sets and render all entries + if (options.expandStorageMapsAndSets) { + for (const [ptrId, storageMap] of pointers.snapshot) { + // display entry from first storage + const [location, value] = [...storageMap.entries()][0]; + + if (value.startsWith("\x1b[38;2;50;153;220m") || value.startsWith("\x1b[38;2;50;153;220m")) { + const ptr = await Pointer.load(ptrId, undefined, true); + if (ptr.val instanceof StorageMap) { + const map = ptr.val; + let inner = ""; + for await (const [key, val] of map) { + inner += ` ${Runtime.valueToDatexStringExperimental(key, true, true)}\x1b[0m => ${Runtime.valueToDatexStringExperimental(val, true, true)}\n` + } + if (inner) storageMap.set(location, "\x1b[38;2;50;153;220m \x1b[0m{\n"+inner+"\x1b[0m\n}") + } + else if (ptr.val instanceof StorageSet) { + const set = ptr.val; + let inner = ""; + for await (const val of set) { + inner += ` ${Runtime.valueToDatexStringExperimental(val, true, true)},\n` + } + if (inner) storageMap.set(location, "\x1b[38;2;50;153;220m \x1b[0m{\n"+inner+"\x1b[0m\n}") + } + } + } + } + + return {items, pointers} + } + + private static async createSnapshot( + keyGenerator: (location?: StorageLocation | undefined) => Promise>, + itemGetter: (key: string, colorized: boolean, location: StorageLocation) => Promise, + ) { + const snapshot = new Map>().setAutoDefault(Map); + const inconsistencies = new Map>().setAutoDefault(Map); + for (const location of new Set([this.#primary_location!, ...this.#locations.keys()].filter(l=>!!l))) { + for (const key of await keyGenerator(location)) { + const decompiled = await itemGetter(key, true, location); + if (typeof decompiled !== "string") { + console.error("Invalid entry in storage (" + location.name + "): " + key); + continue; + } + snapshot.getAuto(key).set(location, decompiled); + } + } + + + // find inconsistencies + for (const [key, storageMap] of snapshot) { + const [location, value] = [...storageMap.entries()][0]; + // compare with first entry + for (const [location2, value2] of storageMap) { + if (value !== value2) { + inconsistencies.getAuto(key).set(location, value); + inconsistencies.getAuto(key).set(location2, value2); + } + } + } + + return {snapshot, inconsistencies}; + } + } export namespace Storage { diff --git a/types/native_types.ts b/types/native_types.ts index 717c1966..ff9b1ab5 100644 --- a/types/native_types.ts +++ b/types/native_types.ts @@ -278,6 +278,11 @@ Type.std.Set.setJSInterface({ get_property: (parent:Set, key) => NOT_EXISTING, has_property: (parent:Set, key) => parent.has(key), + // implemented to support self-referencing serialization, not actual properties + // small issue with this approach: the Set always contains 'undefined' + set_property_silently: (parent:Set, key, value, pointer) => Set.prototype.add.call(parent, value), + set_property: (parent:Set, key, value) => parent.add(value), + count: (parent:Set) => parent.size, keys: (parent:Set) => [...parent], diff --git a/types/storage_map.ts b/types/storage_map.ts index 252e6ae4..65bd9b22 100644 --- a/types/storage_map.ts +++ b/types/storage_map.ts @@ -11,18 +11,17 @@ const logger = new Logger("StorageMap"); /** * WeakMap that outsources values to storage. - * In contrast to JS WeakMaps, primitive keys are also allowed - * Entries are not automatically garbage collected but must be - * explicitly deleted - * all methods are async + * The API is similar to the JS WeakMap API, but all methods are async. + * In contrast to JS WeakMaps, primitive keys are also allowed. + * The StorageWeakMap holds no strong references to its keys in storage. + * This means that the pointer of a key can be garbage collected. */ export class StorageWeakMap { #prefix?: string; constructor(){ - // TODO: does not work with eternal pointers! - // Pointer.proxifyValue(this) + Pointer.proxifyValue(this) } @@ -32,9 +31,8 @@ export class StorageWeakMap { return map; } - get prefix() { - // @ts-ignore - if (!this.#prefix) this.#prefix = 'dxmap::'+this[DX_PTR].idString()+'.'; + protected get _prefix() { + if (!this.#prefix) this.#prefix = 'dxmap::'+(this as any)[DX_PTR].idString()+'.'; return this.#prefix; } @@ -83,13 +81,13 @@ export class StorageWeakMap { protected async getStorageKey(key: K) { const keyHash = await Compiler.getUniqueValueIdentifier(key); // @ts-ignore DX_PTR - return this.prefix + keyHash; + return this._prefix + keyHash; } async clear() { const promises = []; - for (const key of await Storage.getItemKeysStartingWith(this.prefix)) { - promises.push(await Storage.removeItem(key)); + for (const key of await Storage.getItemKeysStartingWith(this._prefix)) { + promises.push(Storage.removeItem(key)); } await Promise.all(promises); } @@ -97,7 +95,8 @@ export class StorageWeakMap { } /** - * Map that outsources values to storage. + * Set that outsources values to storage. + * The API is similar to the JS Map API, but all methods are async. */ export class StorageMap extends StorageWeakMap { @@ -128,11 +127,14 @@ export class StorageMap extends StorageWeakMap { return Storage.removeItem(storage_item_key) } + /** + * Async iterator that returns all keys. + */ keys() { const self = this; const key_prefix = this.#key_prefix; return (async function*(){ - const keyGenerator = await Storage.getItemKeysStartingWith(self.prefix); + const keyGenerator = await Storage.getItemKeysStartingWith(self._prefix); for (const key of keyGenerator) { const keyValue = await Storage.getItem(key_prefix+key); @@ -140,16 +142,24 @@ export class StorageMap extends StorageWeakMap { } })() } + + /** + * Returns an array containing all keys. + * This can be used to iterate over the keys without using a (for await of) loop. + */ async keysArray() { const keys = []; for await (const key of this.keys()) keys.push(key); return keys; } + /** + * Async iterator that returns all values. + */ values() { const self = this; return (async function*(){ - const keyGenerator = await Storage.getItemKeysStartingWith(self.prefix); + const keyGenerator = await Storage.getItemKeysStartingWith(self._prefix); for (const key of keyGenerator) { const value = await Storage.getItem(key); @@ -157,15 +167,28 @@ export class StorageMap extends StorageWeakMap { } })() } + + /** + * Returns an array containing all values. + * This can be used to iterate over the values without using a (for await of) loop. + */ async valuesArray() { const values = []; for await (const value of this.values()) values.push(value); return values; } + /** + * Async iterator that returns all entries. + */ entries() { return this[Symbol.asyncIterator]() } + + /** + * Returns an array containing all entries. + * This can be used to iterate over the entries without using a (for await of) loop. + */ async entriesArray() { const entries = []; for await (const entry of this.entries()) entries.push(entry); @@ -173,7 +196,7 @@ export class StorageMap extends StorageWeakMap { } async *[Symbol.asyncIterator]() { - const keyGenerator = await Storage.getItemKeysStartingWith(this.prefix); + const keyGenerator = await Storage.getItemKeysStartingWith(this._prefix); for (const key of keyGenerator) { const keyValue = await Storage.getItem(this.#key_prefix+key); @@ -184,7 +207,7 @@ export class StorageMap extends StorageWeakMap { override async clear() { const promises = []; - for (const key of await Storage.getItemKeysStartingWith(this.prefix)) { + for (const key of await Storage.getItemKeysStartingWith(this._prefix)) { promises.push(await Storage.removeItem(key)); promises.push(await Storage.removeItem(this.#key_prefix+key)); } diff --git a/types/storage_set.ts b/types/storage_set.ts index 5c4116fd..2d2197b5 100644 --- a/types/storage_set.ts +++ b/types/storage_set.ts @@ -9,16 +9,18 @@ import { Logger } from "../utils/logger.ts"; const logger = new Logger("StorageSet"); /** - * Set that outsources values to storage. - * all methods are async + * WeakSet that outsources values to storage. + * The API is similar to the JS WeakSet API, but all methods are async. + * In contrast to JS WeakSets, primitive values are also allowed. + * The StorageWeakSet holds no strong references to its values in storage. + * This means that the pointer of a value can be garbage collected. */ -export class StorageSet { +export class StorageWeakSet { #prefix?: string; constructor(){ - // TODO: does not work with eternal pointers! - // Pointer.proxifyValue(this) + Pointer.proxifyValue(this) } static async from(values: readonly V[]) { @@ -27,18 +29,17 @@ export class StorageSet { return set; } - get prefix() { - // @ts-ignore - if (!this.#prefix) this.#prefix = 'dxset::'+this[DX_PTR].idString()+'.'; + protected get _prefix() { + if (!this.#prefix) this.#prefix = 'dxset::'+(this as any)[DX_PTR].idString()+'.'; return this.#prefix; } async add(value: V) { const storage_key = await this.getStorageKey(value); if (await this._has(storage_key)) return; // already exists - return this._add(storage_key, value); + return this._add(storage_key, null); } - protected _add(storage_key:string, value:V) { + protected _add(storage_key:string, value:V|null) { this.activateCacheTimeout(storage_key); return Storage.setItem(storage_key, value); } @@ -66,43 +67,79 @@ export class StorageSet { }, 60_000); } - protected getStorageKey(value: V) { - const keyHash = Compiler.getUniqueValueIdentifier(value); + protected async getStorageKey(value: V) { + const keyHash = await Compiler.getUniqueValueIdentifier(value); // @ts-ignore DX_PTR - return this.prefix + keyHash; + return this._prefix + keyHash; } async clear() { const promises = []; - for (const key of await Storage.getItemKeysStartingWith(this.prefix)) { + for (const key of await Storage.getItemKeysStartingWith(this._prefix)) { promises.push(await Storage.removeItem(key)); } await Promise.all(promises); } +} + +/** + * Set that outsources values to storage. + * The API is similar to the JS Set API, but all methods are async. + */ +export class StorageSet extends StorageWeakSet { + + /** + * Appends a new value to the StorageWeakSet. + */ + async add(value: V) { + const storage_key = await this.getStorageKey(value); + if (await this._has(storage_key)) return; // already exists + return this._add(storage_key, value); + } + + /** + * Async iterator that returns all keys. + */ keys() { return this[Symbol.asyncIterator]() } + + /** + * Returns an array containing all keys. + * This can be used to iterate over the keys without using a (for await of) loop. + */ async keysArray() { const keys = []; for await (const key of this.keys()) keys.push(key); return keys; } + /** + * Async iterator that returns all values. + */ values() { return this[Symbol.asyncIterator]() } + + /** + * Returns an array containing all values. + * This can be used to iterate over the values without using a (for await of) loop. + */ async valuesArray() { const values = []; for await (const value of this.values()) values.push(value); return values; } + /** + * Async iterator that returns all entries. + */ entries() { const self = this; return (async function*(){ - const keyGenerator = await Storage.getItemKeysStartingWith(self.prefix); + const keyGenerator = await Storage.getItemKeysStartingWith(self._prefix); for (const key of keyGenerator) { const value = await Storage.getItem(key); @@ -110,6 +147,11 @@ export class StorageSet { } })() } + + /** + * Returns an array containing all entries. + * This can be used to iterate over the entries without using a (for await of) loop. + */ async entriesArray() { const entries = []; for await (const entry of this.entries()) entries.push(entry); @@ -117,7 +159,7 @@ export class StorageSet { } async *[Symbol.asyncIterator]() { - const keyGenerator = await Storage.getItemKeysStartingWith(this.prefix); + const keyGenerator = await Storage.getItemKeysStartingWith(this._prefix); for (const key of keyGenerator) { const value = await Storage.getItem(key); diff --git a/types/type.ts b/types/type.ts index c03b547f..d590873b 100644 --- a/types/type.ts +++ b/types/type.ts @@ -20,7 +20,7 @@ import type { Task } from "./task.ts"; import { Assertion } from "./assertion.ts"; import type { Iterator } from "./iterator.ts"; import {StorageMap, StorageWeakMap} from "./storage_map.ts" -import {StorageSet} from "./storage_set.ts" +import {StorageSet, StorageWeakSet} from "./storage_set.ts" import { ExtensibleFunction } from "./function-utils.ts"; import type { JSTransferableFunction } from "./js-function.ts"; @@ -56,11 +56,17 @@ export class Type extends ExtensibleFunction { parameters:any[] // special type parameters #jsTypeDefModule?: string|URL // URL for the JS module that creates the corresponding type definition + #potentialJsTypeDefModule?: string|URL // remember jsTypeDefModule if jsTypeDefModuleMapper is added later get jsTypeDefModule():string|URL|undefined {return this.#jsTypeDefModule} set jsTypeDefModule(url: string|URL) { + // custom module mapper if (Type.#jsTypeDefModuleMapper) this.#jsTypeDefModule = Type.#jsTypeDefModuleMapper(url, this); - else this.#jsTypeDefModule = url; + // default: only allow http/https modules + else if (url.toString().startsWith("http://") || url.toString().startsWith("https://")) { + this.#jsTypeDefModule = url; + } + this.#potentialJsTypeDefModule = url; } root_type: Type; // DatexType without parameters and variation @@ -84,7 +90,7 @@ export class Type extends ExtensibleFunction { this.#jsTypeDefModuleMapper = fn; // update existing typedef modules for (const type of this.types.values()) { - if (type.#jsTypeDefModule) type.jsTypeDefModule = type.#jsTypeDefModule; + if (type.#potentialJsTypeDefModule) type.jsTypeDefModule = type.#potentialJsTypeDefModule; } } @@ -1012,6 +1018,7 @@ export class Type extends ExtensibleFunction { StorageMap: Type.get>("std:StorageMap"), StorageWeakMap: Type.get>("std:StorageWeakMap"), StorageSet: Type.get>("std:StorageSet"), + StorageWeakSet: Type.get>("std:StorageWeakSet"), Error: Type.get("std:Error"), SyntaxError: Type.get("std:SyntaxError"), @@ -1095,6 +1102,13 @@ Type.std.StorageMap.setJSInterface({ visible_children: new Set(), }) +Type.std.StorageWeakSet.setJSInterface({ + class: StorageWeakSet, + is_normal_object: true, + proxify_children: true, + visible_children: new Set(), +}) + Type.std.StorageSet.setJSInterface({ class: StorageSet, is_normal_object: true,