@rest-hooks/normalizr
Advanced tools
Comparing version 9.5.1 to 10.0.0
@@ -127,6 +127,69 @@ define(['exports'], (function (exports) { 'use strict'; | ||
/** Maps entity dependencies to a value (usually their denormalized form) | ||
* | ||
* Dependencies store `Path` to enable quick traversal using only `State` | ||
* If *any* members of the dependency get cleaned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakEntityMap { | ||
constructor() { | ||
this.next = new WeakMap(); | ||
} | ||
get(entity, getEntity) { | ||
let curLink = this.next.get(entity); | ||
if (!curLink) return EMPTY; | ||
while (curLink.nextPath) { | ||
const nextEntity = getEntity(curLink.nextPath); | ||
curLink = curLink.next.get(nextEntity); | ||
if (!curLink) return EMPTY; | ||
} | ||
// curLink exists, but has no path - so must have a value | ||
return [curLink.value, curLink.journey]; | ||
} | ||
set(dependencies, value) { | ||
if (dependencies.length < 1) throw new KeySize(); | ||
let curLink = this; | ||
for (const { | ||
entity, | ||
path | ||
} of dependencies) { | ||
let nextLink = curLink.next.get(entity); | ||
if (!nextLink) { | ||
nextLink = new Link(); | ||
curLink.next.set(entity, nextLink); | ||
} | ||
curLink.nextPath = path; | ||
curLink = nextLink; | ||
} | ||
// in case there used to be more | ||
delete curLink.nextPath; | ||
curLink.value = value; | ||
// we could recompute this on get, but it would have a cost and we optimize for `get` | ||
curLink.journey = depToPaths(dependencies); | ||
} | ||
} | ||
const EMPTY = [undefined, undefined]; | ||
function getEntities(state) { | ||
const entityIsImmutable = isImmutable(state); | ||
if (entityIsImmutable) { | ||
return ({ | ||
key, | ||
pk | ||
}) => state.getIn([key, pk]); | ||
} else { | ||
return ({ | ||
key, | ||
pk | ||
}) => { | ||
var _state$key; | ||
return (_state$key = state[key]) == null ? void 0 : _state$key[pk]; | ||
}; | ||
} | ||
} | ||
function depToPaths(dependencies) { | ||
return dependencies.map(dep => dep.path); | ||
} | ||
/** Link in a chain */ | ||
class Link { | ||
constructor() { | ||
this.children = new WeakMap(); | ||
this.next = new WeakMap(); | ||
} | ||
@@ -141,57 +204,7 @@ } | ||
/** Maps from a list of objects (referentially) to any value | ||
* | ||
* If *any* members of the list get claned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakListMap { | ||
constructor() { | ||
this.first = new WeakMap(); | ||
} | ||
delete(key) { | ||
const link = this.traverse(key); | ||
link == null ? true : delete link.value; | ||
return !!link; | ||
} | ||
get(key) { | ||
const link = this.traverse(key); | ||
return link == null ? void 0 : link.value; | ||
} | ||
has(key) { | ||
const link = this.traverse(key); | ||
if (!link) return false; | ||
return Object.hasOwn(link, 'value'); | ||
} | ||
set(key, value) { | ||
if (key.length < 1) throw new KeySize(); | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
if (!cur.has(key[i])) { | ||
link = new Link(); | ||
cur.set(key[i], link); | ||
} else { | ||
link = cur.get(key[i]); | ||
} | ||
cur = link.children; | ||
// do on later iteration of loop. this makes typescript happy rather than putting after loop | ||
if (i === key.length - 1) { | ||
link.value = value; | ||
} | ||
} | ||
return this; | ||
} | ||
traverse(key) { | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
link = cur.get(key[i]); | ||
if (!link) return; | ||
cur = link.children; | ||
} | ||
return link; | ||
} | ||
} | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex) => { | ||
const entity = getEntity(entityOrId, schema); | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex) => { | ||
const entity = typeof entityOrId === 'object' ? entityOrId : getEntity({ | ||
pk: entityOrId, | ||
key: schema.key | ||
}); | ||
if (typeof entity === 'symbol' && entity.toString().includes('DELETED')) { | ||
@@ -224,45 +237,62 @@ return [undefined, true, true]; | ||
} | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
const localCacheKey = localCache[key]; | ||
const cycleCacheKey = cycleCache[key]; | ||
const entityCacheKey = entityCache[key]; | ||
let found = true; | ||
let deleted = false; | ||
// local cache lookup first | ||
if (!localCacheKey[pk]) { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push(entity); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk); | ||
const globalCache = getCache(pk, schema); | ||
const [cacheValue] = globalCache.get(entity, getEntity); | ||
// TODO: what if this just returned the deps - then we don't need to store them | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
if (!globalCacheEntry.has(localKey)) { | ||
globalCacheEntry.set(localKey, localCacheKey[pk]); | ||
} else { | ||
localCacheKey[pk] = globalCacheEntry.get(localKey); | ||
if (cacheValue) { | ||
localCacheKey[pk] = cacheValue.value[0]; | ||
// TODO: can we store the cache values instead of tracking *all* their sources? | ||
// this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it | ||
dependencies.push(...cacheValue.dependencies); | ||
return cacheValue.value; | ||
} | ||
// if we don't find in denormalize cache then do full denormalize | ||
else { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
const cacheValue = { | ||
dependencies: localKey, | ||
value: [localCacheKey[pk], found, deleted] | ||
}; | ||
globalCache.set(localKey, cacheValue); | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
} | ||
} | ||
@@ -275,3 +305,9 @@ } else { | ||
// with no cycle, globalCacheEntry will have already been set | ||
dependencies.push(entity); | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
} | ||
@@ -281,4 +317,6 @@ } | ||
}; | ||
const getUnvisit = (entities, entityCache, resultCache, localCache) => { | ||
const getUnvisit = (entities, entityCache, resultCache) => { | ||
const getEntity = getEntities(entities); | ||
const getCache = getEntityCaches(entityCache); | ||
const localCache = {}; | ||
const dependencies = []; | ||
@@ -297,2 +335,4 @@ const cycleIndex = { | ||
const hasDenormalize = typeof schema.denormalize === 'function'; | ||
// deserialize fields (like Date) | ||
if (!hasDenormalize && typeof schema === 'function') { | ||
@@ -312,3 +352,3 @@ if (input instanceof schema) return [input, true, false]; | ||
if (isEntity(schema)) { | ||
return unvisitEntity(input, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex); | ||
return unvisitEntity(input, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex); | ||
} | ||
@@ -320,50 +360,57 @@ if (hasDenormalize) { | ||
} | ||
//const wrappedUnvisit = withTrackedEntities(unvisit, globalKey); | ||
return (input, schema) => { | ||
const ret = unvisit(input, schema); | ||
// in the case where WeakMap cannot be used | ||
// this test ensures null is properly excluded from WeakMap | ||
if (Object(input) !== input) return ret; | ||
dependencies.push(input); | ||
if (!resultCache.has(dependencies)) { | ||
resultCache.set(dependencies, ret[0]); | ||
return ret; | ||
} else { | ||
return [resultCache.get(dependencies), ret[1], ret[2]]; | ||
const cachable = Object(input) === input && Object(schema) === schema; | ||
if (!cachable) { | ||
const ret = unvisit(input, schema); | ||
// this is faster than spread | ||
// https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples | ||
return [ret[0], ret[1], ret[2], depToPaths(dependencies)]; | ||
} | ||
let [ret, entityPaths] = resultCache.get(input, getEntity); | ||
if (ret === undefined) { | ||
ret = unvisit(input, schema); | ||
// we want to do this before we add our 'input' entry | ||
entityPaths = depToPaths(dependencies); | ||
// for the first entry, `path` is ignored so empty members is fine | ||
dependencies.unshift({ | ||
entity: input, | ||
path: { | ||
key: '', | ||
pk: '' | ||
} | ||
}); | ||
resultCache.set(dependencies, ret); | ||
} | ||
return [ret[0], ret[1], ret[2], entityPaths]; | ||
}; | ||
}; | ||
const getEntities = entities => { | ||
const entityIsImmutable = isImmutable(entities); | ||
return (entityOrId, schema) => { | ||
var _entities$schemaKey; | ||
const schemaKey = schema.key; | ||
if (typeof entityOrId === 'object') { | ||
return entityOrId; | ||
const getEntityCaches = entityCache => { | ||
return (pk, schema) => { | ||
const key = schema.key; | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
if (entityIsImmutable) { | ||
return entities.getIn([schemaKey, entityOrId]); | ||
const entityCacheKey = entityCache[key]; | ||
if (!entityCacheKey[pk]) entityCacheKey[pk] = new WeakMap(); | ||
let wem = entityCacheKey[pk].get(schema); | ||
if (!wem) { | ||
wem = new WeakEntityMap(); | ||
entityCacheKey[pk].set(schema, wem); | ||
} | ||
return (_entities$schemaKey = entities[schemaKey]) == null ? void 0 : _entities$schemaKey[entityOrId]; | ||
return wem; | ||
}; | ||
}; | ||
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => { | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakEntityMap()) => { | ||
// undefined mean don't do anything | ||
if (schema === undefined) { | ||
return [input, true, false, {}]; | ||
return [input, true, false, []]; | ||
} | ||
if (input === undefined) { | ||
return [undefined, false, false, {}]; | ||
return [undefined, false, false, []]; | ||
} | ||
const resolvedEntities = {}; | ||
const unvisit = getUnvisit(entities, entityCache, resultCache, resolvedEntities); | ||
return [...unvisit(input, schema), resolvedEntities]; | ||
return getUnvisit(entities, entityCache, resultCache)(input, schema); | ||
}; | ||
function getGlobalCacheEntry(entityCache, id) { | ||
if (!entityCache[id]) entityCache[id] = new WeakListMap(); | ||
return entityCache[id]; | ||
} | ||
@@ -592,3 +639,3 @@ // TODO(breaking): remove once unused | ||
exports.ExpiryStatus = ExpiryStatus; | ||
exports.WeakListMap = WeakListMap; | ||
exports.WeakEntityMap = WeakEntityMap; | ||
exports.denormalize = denormalize; | ||
@@ -595,0 +642,0 @@ exports.inferResults = inferResults; |
@@ -128,6 +128,69 @@ var rest_hooks_normalizr = (function (exports) { | ||
/** Maps entity dependencies to a value (usually their denormalized form) | ||
* | ||
* Dependencies store `Path` to enable quick traversal using only `State` | ||
* If *any* members of the dependency get cleaned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakEntityMap { | ||
constructor() { | ||
this.next = new WeakMap(); | ||
} | ||
get(entity, getEntity) { | ||
let curLink = this.next.get(entity); | ||
if (!curLink) return EMPTY; | ||
while (curLink.nextPath) { | ||
const nextEntity = getEntity(curLink.nextPath); | ||
curLink = curLink.next.get(nextEntity); | ||
if (!curLink) return EMPTY; | ||
} | ||
// curLink exists, but has no path - so must have a value | ||
return [curLink.value, curLink.journey]; | ||
} | ||
set(dependencies, value) { | ||
if (dependencies.length < 1) throw new KeySize(); | ||
let curLink = this; | ||
for (const { | ||
entity, | ||
path | ||
} of dependencies) { | ||
let nextLink = curLink.next.get(entity); | ||
if (!nextLink) { | ||
nextLink = new Link(); | ||
curLink.next.set(entity, nextLink); | ||
} | ||
curLink.nextPath = path; | ||
curLink = nextLink; | ||
} | ||
// in case there used to be more | ||
delete curLink.nextPath; | ||
curLink.value = value; | ||
// we could recompute this on get, but it would have a cost and we optimize for `get` | ||
curLink.journey = depToPaths(dependencies); | ||
} | ||
} | ||
const EMPTY = [undefined, undefined]; | ||
function getEntities(state) { | ||
const entityIsImmutable = isImmutable(state); | ||
if (entityIsImmutable) { | ||
return ({ | ||
key, | ||
pk | ||
}) => state.getIn([key, pk]); | ||
} else { | ||
return ({ | ||
key, | ||
pk | ||
}) => { | ||
var _state$key; | ||
return (_state$key = state[key]) == null ? void 0 : _state$key[pk]; | ||
}; | ||
} | ||
} | ||
function depToPaths(dependencies) { | ||
return dependencies.map(dep => dep.path); | ||
} | ||
/** Link in a chain */ | ||
class Link { | ||
constructor() { | ||
this.children = new WeakMap(); | ||
this.next = new WeakMap(); | ||
} | ||
@@ -142,57 +205,7 @@ } | ||
/** Maps from a list of objects (referentially) to any value | ||
* | ||
* If *any* members of the list get claned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakListMap { | ||
constructor() { | ||
this.first = new WeakMap(); | ||
} | ||
delete(key) { | ||
const link = this.traverse(key); | ||
link == null ? true : delete link.value; | ||
return !!link; | ||
} | ||
get(key) { | ||
const link = this.traverse(key); | ||
return link == null ? void 0 : link.value; | ||
} | ||
has(key) { | ||
const link = this.traverse(key); | ||
if (!link) return false; | ||
return Object.hasOwn(link, 'value'); | ||
} | ||
set(key, value) { | ||
if (key.length < 1) throw new KeySize(); | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
if (!cur.has(key[i])) { | ||
link = new Link(); | ||
cur.set(key[i], link); | ||
} else { | ||
link = cur.get(key[i]); | ||
} | ||
cur = link.children; | ||
// do on later iteration of loop. this makes typescript happy rather than putting after loop | ||
if (i === key.length - 1) { | ||
link.value = value; | ||
} | ||
} | ||
return this; | ||
} | ||
traverse(key) { | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
link = cur.get(key[i]); | ||
if (!link) return; | ||
cur = link.children; | ||
} | ||
return link; | ||
} | ||
} | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex) => { | ||
const entity = getEntity(entityOrId, schema); | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex) => { | ||
const entity = typeof entityOrId === 'object' ? entityOrId : getEntity({ | ||
pk: entityOrId, | ||
key: schema.key | ||
}); | ||
if (typeof entity === 'symbol' && entity.toString().includes('DELETED')) { | ||
@@ -225,45 +238,62 @@ return [undefined, true, true]; | ||
} | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
const localCacheKey = localCache[key]; | ||
const cycleCacheKey = cycleCache[key]; | ||
const entityCacheKey = entityCache[key]; | ||
let found = true; | ||
let deleted = false; | ||
// local cache lookup first | ||
if (!localCacheKey[pk]) { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push(entity); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk); | ||
const globalCache = getCache(pk, schema); | ||
const [cacheValue] = globalCache.get(entity, getEntity); | ||
// TODO: what if this just returned the deps - then we don't need to store them | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
if (!globalCacheEntry.has(localKey)) { | ||
globalCacheEntry.set(localKey, localCacheKey[pk]); | ||
} else { | ||
localCacheKey[pk] = globalCacheEntry.get(localKey); | ||
if (cacheValue) { | ||
localCacheKey[pk] = cacheValue.value[0]; | ||
// TODO: can we store the cache values instead of tracking *all* their sources? | ||
// this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it | ||
dependencies.push(...cacheValue.dependencies); | ||
return cacheValue.value; | ||
} | ||
// if we don't find in denormalize cache then do full denormalize | ||
else { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
const cacheValue = { | ||
dependencies: localKey, | ||
value: [localCacheKey[pk], found, deleted] | ||
}; | ||
globalCache.set(localKey, cacheValue); | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
} | ||
} | ||
@@ -276,3 +306,9 @@ } else { | ||
// with no cycle, globalCacheEntry will have already been set | ||
dependencies.push(entity); | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
} | ||
@@ -282,4 +318,6 @@ } | ||
}; | ||
const getUnvisit = (entities, entityCache, resultCache, localCache) => { | ||
const getUnvisit = (entities, entityCache, resultCache) => { | ||
const getEntity = getEntities(entities); | ||
const getCache = getEntityCaches(entityCache); | ||
const localCache = {}; | ||
const dependencies = []; | ||
@@ -298,2 +336,4 @@ const cycleIndex = { | ||
const hasDenormalize = typeof schema.denormalize === 'function'; | ||
// deserialize fields (like Date) | ||
if (!hasDenormalize && typeof schema === 'function') { | ||
@@ -313,3 +353,3 @@ if (input instanceof schema) return [input, true, false]; | ||
if (isEntity(schema)) { | ||
return unvisitEntity(input, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex); | ||
return unvisitEntity(input, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex); | ||
} | ||
@@ -321,50 +361,57 @@ if (hasDenormalize) { | ||
} | ||
//const wrappedUnvisit = withTrackedEntities(unvisit, globalKey); | ||
return (input, schema) => { | ||
const ret = unvisit(input, schema); | ||
// in the case where WeakMap cannot be used | ||
// this test ensures null is properly excluded from WeakMap | ||
if (Object(input) !== input) return ret; | ||
dependencies.push(input); | ||
if (!resultCache.has(dependencies)) { | ||
resultCache.set(dependencies, ret[0]); | ||
return ret; | ||
} else { | ||
return [resultCache.get(dependencies), ret[1], ret[2]]; | ||
const cachable = Object(input) === input && Object(schema) === schema; | ||
if (!cachable) { | ||
const ret = unvisit(input, schema); | ||
// this is faster than spread | ||
// https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples | ||
return [ret[0], ret[1], ret[2], depToPaths(dependencies)]; | ||
} | ||
let [ret, entityPaths] = resultCache.get(input, getEntity); | ||
if (ret === undefined) { | ||
ret = unvisit(input, schema); | ||
// we want to do this before we add our 'input' entry | ||
entityPaths = depToPaths(dependencies); | ||
// for the first entry, `path` is ignored so empty members is fine | ||
dependencies.unshift({ | ||
entity: input, | ||
path: { | ||
key: '', | ||
pk: '' | ||
} | ||
}); | ||
resultCache.set(dependencies, ret); | ||
} | ||
return [ret[0], ret[1], ret[2], entityPaths]; | ||
}; | ||
}; | ||
const getEntities = entities => { | ||
const entityIsImmutable = isImmutable(entities); | ||
return (entityOrId, schema) => { | ||
var _entities$schemaKey; | ||
const schemaKey = schema.key; | ||
if (typeof entityOrId === 'object') { | ||
return entityOrId; | ||
const getEntityCaches = entityCache => { | ||
return (pk, schema) => { | ||
const key = schema.key; | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
if (entityIsImmutable) { | ||
return entities.getIn([schemaKey, entityOrId]); | ||
const entityCacheKey = entityCache[key]; | ||
if (!entityCacheKey[pk]) entityCacheKey[pk] = new WeakMap(); | ||
let wem = entityCacheKey[pk].get(schema); | ||
if (!wem) { | ||
wem = new WeakEntityMap(); | ||
entityCacheKey[pk].set(schema, wem); | ||
} | ||
return (_entities$schemaKey = entities[schemaKey]) == null ? void 0 : _entities$schemaKey[entityOrId]; | ||
return wem; | ||
}; | ||
}; | ||
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => { | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakEntityMap()) => { | ||
// undefined mean don't do anything | ||
if (schema === undefined) { | ||
return [input, true, false, {}]; | ||
return [input, true, false, []]; | ||
} | ||
if (input === undefined) { | ||
return [undefined, false, false, {}]; | ||
return [undefined, false, false, []]; | ||
} | ||
const resolvedEntities = {}; | ||
const unvisit = getUnvisit(entities, entityCache, resultCache, resolvedEntities); | ||
return [...unvisit(input, schema), resolvedEntities]; | ||
return getUnvisit(entities, entityCache, resultCache)(input, schema); | ||
}; | ||
function getGlobalCacheEntry(entityCache, id) { | ||
if (!entityCache[id]) entityCache[id] = new WeakListMap(); | ||
return entityCache[id]; | ||
} | ||
@@ -593,3 +640,3 @@ // TODO(breaking): remove once unused | ||
exports.ExpiryStatus = ExpiryStatus; | ||
exports.WeakListMap = WeakListMap; | ||
exports.WeakEntityMap = WeakEntityMap; | ||
exports.denormalize = denormalize; | ||
@@ -596,0 +643,0 @@ exports.inferResults = inferResults; |
@@ -1,1 +0,1 @@ | ||
function e(e){return null!==e&&void 0!==e.pk}const t=e=>{if("production"!==process.env.NODE_ENV){if(Array.isArray(e)&&e.length>1)throw new Error(`Expected schema definition to be a single schema, but found ${e.length}.`)}return e[0]},n=([e,,t])=>void 0!==e&&!t,r=(e,n,r,o,i,s,c)=>{e=t(e);const u=(e=>Array.isArray(e)?e:Object.keys(e).map((t=>e[t])))(n);return u.map(((t,n)=>i(t,r,o,e,s,c)))},o=(e,r,o)=>(e=t(e),[r.map?r.map((t=>o(t,e))).filter(n).map((([e])=>e)):r,!0,!1]);function i(e,t,n,r){}function s(e){return!("function"!=typeof e.hasOwnProperty||!(Object.hasOwnProperty.call(e,"__ownerID")||e._map&&Object.hasOwnProperty.call(e._map,"__ownerID")))}const c=(e,t,n,r,o,i,s)=>{const c=Object.assign({},t);return Object.keys(e).forEach((n=>{const r=e[n],u=o(t[n],t,n,r,i,s);null==u?delete c[n]:c[n]=u})),c},u=(e,t,n)=>{if(s(t))return function(e,t,n){let r=!0,o=!1;return[Object.keys(e).reduce(((t,i)=>{const s=`${i}`,[c,u,a]=n(t.get(s),e[s]);return u||(r=!1),a&&(o=!0),t.has(s)?t.set(s,c):t}),t),r,o]}(e,t,n);const r=Object.assign({},t);let o=!0,i=!1;return Object.keys(e).forEach((t=>{const[s,c,u]=n(r[t],e[t]);void 0!==r[t]&&(r[t]=s),u&&(i=!0),c||(o=!1)})),[r,o,i]};function a(e,t,n,r,o){const i={};for(const s of Object.keys(e))i[s]=r(e[s],t,n,o);return i}class l{constructor(){this.children=new WeakMap}}class f extends Error{constructor(...e){super(...e),this.message="Keys must include at least one member"}}class d{constructor(){this.first=new WeakMap}delete(e){const t=this.traverse(e);return null==t||delete t.value,!!t}get(e){const t=this.traverse(e);return null==t?void 0:t.value}has(e){const t=this.traverse(e);return!!t&&Object.hasOwn(t,"value")}set(e,t){if(e.length<1)throw new f;let n,r=this.first;for(let o=0;o<e.length;o++)r.has(e[o])?n=r.get(e[o]):(n=new l,r.set(e[o],n)),r=n.children,o===e.length-1&&(n.value=t);return this}traverse(e){let t,n=this.first;for(let r=0;r<e.length;r++){if(t=n.get(e[r]),!t)return;n=t.children}return t}}const p=(e,t,n,r,o,i,c,u,a)=>{const l=r(e,t);if("symbol"==typeof l&&l.toString().includes("DELETED"))return[void 0,!0,!0];if("object"!=typeof l||null===l)return[l,!1,!1];const f="string"==typeof e?e:t.pk(s(l)?l.toJS():l);if(void 0===f||""===f||"undefined"===f)return[l,!1,!1];const p=t.key;p in o||(o[p]=Object.create(null)),p in i||(i[p]=Object.create(null)),p in c||(c[p]=Object.create(null));const y=o[p],h=i[p],g=c[p];let b=!0,m=!1;if(y[f])f in h?a.i=h[f]:u.push(l);else{const e=u.length;let r;u.push(l),t.createIfValid?r=y[f]=s(l)?t.createIfValid(l.toObject()):t.createIfValid(l):(r=l,n=function(e){const t=e.og||e,n=(e,n)=>t(e,n);return n.og=e,n}(n),n.setLocal=e=>y[f]=e),h[f]=e,void 0===r?(b=!1,m=!0):[y[f],b,m]=t.denormalize(r,n),delete h[f];const o=function(e,t){e[t]||(e[t]=new d);return e[t]}(g,f),i=u.slice(-1===a.i?e:a.i);o.has(i)?y[f]=o.get(i):o.set(i,y[f]),a.i===e&&(a.i=-1)}return[y[f],b,m]},y=e=>{const t=s(e);return(n,r)=>{var o;const i=r.key;return"object"==typeof n?n:t?e.getIn([i,n]):null==(o=e[i])?void 0:o[n]}},h=(t,n,r,i={},s=new d)=>{if(void 0===n)return[t,!0,!1,{}];if(void 0===t)return[void 0,!1,!1,{}];const c={},a=((t,n,r,i)=>{const s=y(t),c=[],a={i:-1},l={};function f(t,r){if(!r)return[t,!0,!1];if(null===t)return[t,!0,!1];const d="function"==typeof r.denormalize;return d||"function"!=typeof r?void 0===t?[t,!1,!1]:d||"object"!=typeof r?e(r)?p(t,r,f,s,i,l,n,c,a):d?r.denormalize(t,f):[t,!0,!1]:(Array.isArray(r)?o:u)(r,t,f):t instanceof r||void 0===t?[t,!0,!1]:[new r(t),!0,!1]}return(e,t)=>{const n=f(e,t);return Object(e)!==e?n:(c.push(e),r.has(c)?[r.get(c),n[1],n[2]]:(r.set(c,n[0]),n))}})(r,i,s,c);return[...a(t,n),c]};const g=Symbol("ENTITY WAS DELETED"),b=(e,t,n,o,i,s)=>{if(!e||!o)return e;if(o.normalize&&"function"==typeof o.normalize)return"object"!=typeof e?e:o.normalize(e,t,n,b,i,s);if("function"==typeof o)return new o(e);if("object"!=typeof e||"object"!=typeof o)return e;return(Array.isArray(o)?r:c)(o,e,t,n,b,i,s)};const m=(e,t,n={},r={},o={},i={date:Date.now(),expiresAt:1/0,fetchedAt:0})=>{if(null==t)return{entities:n,indexes:r,result:e,entityMeta:o};const s=function(e){return["object","function"].includes(typeof e)?"object":typeof e}(t);if(null===e||typeof e!==s&&(void 0===t.key||void 0!==t.pk||"string"!=typeof e)){if("production"!==process.env.NODE_ENV){const n=e=>{try{return"string"!=typeof JSON.parse(e)}catch(e){return!1}};throw"string"==typeof e&&n(e)?new Error(`Normalizing a string, but this does match schema.\n\nParsing this input string as JSON worked. This likely indicates fetch function did not parse\nthe JSON. By default, this only happens if "content-type" header includes "json".\nSee https://resthooks.io/rest/api/RestEndpoint#parseResponse for more information\n\n Schema: ${JSON.stringify(t,void 0,2)}\n Input: "${e}"`):new Error(`Unexpected input given to normalize. Expected type to be "${s}", found "${null===e?"null":typeof e}".\n\n Schema: ${JSON.stringify(t,void 0,2)}\n Input: "${e}"`)}throw new Error(`Unexpected input given to normalize. Expected type to be "${s}", found "${null===e?"null":typeof e}".`)}const c=Object.assign({},n),u=Object.assign({},r),a=Object.assign({},o),l=((e,t,n,r,o,i)=>(s,c,u)=>{const a=s.key;a in e||(e[a]={},n[a]=Object.assign({},n[a]),o[a]=Object.assign({},o[a]));const l=e[a][u];if(l)e[a][u]=s.merge(l,c);else{const t=s.expiresAt?s.expiresAt(i,c):i.expiresAt,r=n[a][u];if(r){var f,d,p;const n=o[a][u];e[a][u]=s.mergeWithStore?s.mergeWithStore(n,i,r,c):function(e,t,n,r,o){return!t||(e.useIncoming&&n.fetchedAt?e.useIncoming(t,n,r,o):t.date<=n.date)?typeof o!=typeof r?o:e.merge(r,o):r}(s,n,i,r,c),o[a][u]={expiresAt:Math.max(t,null==n?void 0:n.expiresAt),date:Math.max(i.date,null!=(f=null==n?void 0:n.date)?f:0),fetchedAt:Math.max(null!=(d=i.fetchedAt)?d:0,null!=(p=null==n?void 0:n.fetchedAt)?p:0)}}else{var y;e[a][u]=c,o[a][u]={expiresAt:t,date:i.date,fetchedAt:null!=(y=i.fetchedAt)?y:i.date}}}if(Array.isArray(s.indexes)){const o=e[a][u];a in t||(t[a]={},r[a]=Object.assign({},r[a]));for(const e of s.indexes){e in t[a]||(r[a][e]=t[a][e]={});const i=t[a][e];l&&delete i[l[e]],n[a]&&n[a][u]&&n[a][u][e]!==o[e]&&(i[n[a][u][e]]=g),e in o?i[o[e]]=u:"production"!==process.env.NODE_ENV&&console.warn(`Index not found in entity. Indexes must be top-level members of your entity.\nIndex: ${e}\nEntity: ${JSON.stringify(o,void 0,2)}`)}}n[a][u]=e[a][u]})({},{},c,u,a,i);return{entities:c,indexes:u,result:b(e,e,void 0,t,l,{}),entityMeta:a}};function v(e,t,n,r={}){if(function(e){return!!e&&"function"==typeof e.infer}(e))return e.infer(t,n,v,r);if("object"==typeof e&&e){return(Array.isArray(e)?i:a)(e,t,n,v,r)}return e}var O={Invalid:1,InvalidIfStale:2,Valid:3};Object.hasOwn=Object.hasOwn||function(e,t){return Object.prototype.hasOwnProperty.call(e,t)};export{g as DELETED,O as ExpiryStatus,d as WeakListMap,h as denormalize,v as inferResults,e as isEntity,m as normalize}; | ||
function e(e){return null!==e&&void 0!==e.pk}const t=e=>{if("production"!==process.env.NODE_ENV){if(Array.isArray(e)&&e.length>1)throw new Error(`Expected schema definition to be a single schema, but found ${e.length}.`)}return e[0]},n=([e,,t])=>void 0!==e&&!t,r=(e,n,r,o,i,s,c)=>{e=t(e);const a=(e=>Array.isArray(e)?e:Object.keys(e).map((t=>e[t])))(n);return a.map(((t,n)=>i(t,r,o,e,s,c)))},o=(e,r,o)=>(e=t(e),[r.map?r.map((t=>o(t,e))).filter(n).map((([e])=>e)):r,!0,!1]);function i(e,t,n,r){}function s(e){return!("function"!=typeof e.hasOwnProperty||!(Object.hasOwnProperty.call(e,"__ownerID")||e._map&&Object.hasOwnProperty.call(e._map,"__ownerID")))}const c=(e,t,n,r,o,i,s)=>{const c=Object.assign({},t);return Object.keys(e).forEach((n=>{const r=e[n],a=o(t[n],t,n,r,i,s);null==a?delete c[n]:c[n]=a})),c},a=(e,t,n)=>{if(s(t))return function(e,t,n){let r=!0,o=!1;return[Object.keys(e).reduce(((t,i)=>{const s=`${i}`,[c,a,u]=n(t.get(s),e[s]);return a||(r=!1),u&&(o=!0),t.has(s)?t.set(s,c):t}),t),r,o]}(e,t,n);const r=Object.assign({},t);let o=!0,i=!1;return Object.keys(e).forEach((t=>{const[s,c,a]=n(r[t],e[t]);void 0!==r[t]&&(r[t]=s),a&&(i=!0),c||(o=!1)})),[r,o,i]};function u(e,t,n,r,o){const i={};for(const s of Object.keys(e))i[s]=r(e[s],t,n,o);return i}class f{constructor(){this.next=new WeakMap}get(e,t){let n=this.next.get(e);if(!n)return l;for(;n.nextPath;){const e=t(n.nextPath);if(n=n.next.get(e),!n)return l}return[n.value,n.journey]}set(e,t){if(e.length<1)throw new y;let n=this;for(const{entity:t,path:r}of e){let e=n.next.get(t);e||(e=new d,n.next.set(t,e)),n.nextPath=r,n=e}delete n.nextPath,n.value=t,n.journey=p(e)}}const l=[void 0,void 0];function p(e){return e.map((e=>e.path))}class d{constructor(){this.next=new WeakMap}}class y extends Error{constructor(...e){super(...e),this.message="Keys must include at least one member"}}const h=(t,n,r)=>{const i=s(c=t)?({key:e,pk:t})=>c.getIn([e,t]):({key:e,pk:t})=>{var n;return null==(n=c[e])?void 0:n[t]};var c;const u=g(n),f={},l=[],d={i:-1},y={};function h(t,n){if(!n)return[t,!0,!1];if(null===t)return[t,!0,!1];const r="function"==typeof n.denormalize;if(!r&&"function"==typeof n)return t instanceof n||void 0===t?[t,!0,!1]:[new n(t),!0,!1];if(void 0===t)return[t,!1,!1];if(!r&&"object"==typeof n){return(Array.isArray(n)?o:a)(n,t,h)}return e(n)?((e,t,n,r,o,i,c,a,u)=>{const f="object"==typeof e?e:r({pk:e,key:t.key});if("symbol"==typeof f&&f.toString().includes("DELETED"))return[void 0,!0,!0];if("object"!=typeof f||null===f)return[f,!1,!1];const l="string"==typeof e?e:t.pk(s(f)?f.toJS():f);if(void 0===l||""===l||"undefined"===l)return[f,!1,!1];const p=t.key;p in i||(i[p]=Object.create(null)),p in c||(c[p]=Object.create(null));const d=i[p],y=c[p];let h=!0,g=!1;if(d[l])l in y?u.i=y[l]:a.push({entity:f,path:{key:p,pk:l}});else{const e=o(l,t),[i]=e.get(f,r);if(i)return d[l]=i.value[0],a.push(...i.dependencies),i.value;{const r=a.length;let o;a.push({entity:f,path:{key:p,pk:l}}),t.createIfValid?o=d[l]=s(f)?t.createIfValid(f.toObject()):t.createIfValid(f):(o=f,n=function(e){const t=e.og||e,n=(e,n)=>t(e,n);return n.og=e,n}(n),n.setLocal=e=>d[l]=e),y[l]=r,void 0===o?(h=!1,g=!0):[d[l],h,g]=t.denormalize(o,n),delete y[l];const i=a.slice(-1===u.i?r:u.i),c={dependencies:i,value:[d[l],h,g]};e.set(i,c),u.i===r&&(u.i=-1)}}return[d[l],h,g]})(t,n,h,i,u,f,y,l,d):r?n.denormalize(t,h):[t,!0,!1]}return(e,t)=>{if(!(Object(e)===e&&Object(t)===t)){const n=h(e,t);return[n[0],n[1],n[2],p(l)]}let[n,o]=r.get(e,i);return void 0===n&&(n=h(e,t),o=p(l),l.unshift({entity:e,path:{key:"",pk:""}}),r.set(l,n)),[n[0],n[1],n[2],o]}},g=e=>(t,n)=>{const r=n.key;r in e||(e[r]=Object.create(null));const o=e[r];o[t]||(o[t]=new WeakMap);let i=o[t].get(n);return i||(i=new f,o[t].set(n,i)),i},b=(e,t,n,r={},o=new f)=>void 0===t?[e,!0,!1,[]]:void 0===e?[void 0,!1,!1,[]]:h(n,r,o)(e,t);const m=Symbol("ENTITY WAS DELETED"),v=(e,t,n,o,i,s)=>{if(!e||!o)return e;if(o.normalize&&"function"==typeof o.normalize)return"object"!=typeof e?e:o.normalize(e,t,n,v,i,s);if("function"==typeof o)return new o(e);if("object"!=typeof e||"object"!=typeof o)return e;return(Array.isArray(o)?r:c)(o,e,t,n,v,i,s)};const O=(e,t,n={},r={},o={},i={date:Date.now(),expiresAt:1/0,fetchedAt:0})=>{if(null==t)return{entities:n,indexes:r,result:e,entityMeta:o};const s=function(e){return["object","function"].includes(typeof e)?"object":typeof e}(t);if(null===e||typeof e!==s&&(void 0===t.key||void 0!==t.pk||"string"!=typeof e)){if("production"!==process.env.NODE_ENV){const n=e=>{try{return"string"!=typeof JSON.parse(e)}catch(e){return!1}};throw"string"==typeof e&&n(e)?new Error(`Normalizing a string, but this does match schema.\n\nParsing this input string as JSON worked. This likely indicates fetch function did not parse\nthe JSON. By default, this only happens if "content-type" header includes "json".\nSee https://resthooks.io/rest/api/RestEndpoint#parseResponse for more information\n\n Schema: ${JSON.stringify(t,void 0,2)}\n Input: "${e}"`):new Error(`Unexpected input given to normalize. Expected type to be "${s}", found "${null===e?"null":typeof e}".\n\n Schema: ${JSON.stringify(t,void 0,2)}\n Input: "${e}"`)}throw new Error(`Unexpected input given to normalize. Expected type to be "${s}", found "${null===e?"null":typeof e}".`)}const c=Object.assign({},n),a=Object.assign({},r),u=Object.assign({},o),f=((e,t,n,r,o,i)=>(s,c,a)=>{const u=s.key;u in e||(e[u]={},n[u]=Object.assign({},n[u]),o[u]=Object.assign({},o[u]));const f=e[u][a];if(f)e[u][a]=s.merge(f,c);else{const t=s.expiresAt?s.expiresAt(i,c):i.expiresAt,r=n[u][a];if(r){var l,p,d;const n=o[u][a];e[u][a]=s.mergeWithStore?s.mergeWithStore(n,i,r,c):function(e,t,n,r,o){return!t||(e.useIncoming&&n.fetchedAt?e.useIncoming(t,n,r,o):t.date<=n.date)?typeof o!=typeof r?o:e.merge(r,o):r}(s,n,i,r,c),o[u][a]={expiresAt:Math.max(t,null==n?void 0:n.expiresAt),date:Math.max(i.date,null!=(l=null==n?void 0:n.date)?l:0),fetchedAt:Math.max(null!=(p=i.fetchedAt)?p:0,null!=(d=null==n?void 0:n.fetchedAt)?d:0)}}else{var y;e[u][a]=c,o[u][a]={expiresAt:t,date:i.date,fetchedAt:null!=(y=i.fetchedAt)?y:i.date}}}if(Array.isArray(s.indexes)){const o=e[u][a];u in t||(t[u]={},r[u]=Object.assign({},r[u]));for(const e of s.indexes){e in t[u]||(r[u][e]=t[u][e]={});const i=t[u][e];f&&delete i[f[e]],n[u]&&n[u][a]&&n[u][a][e]!==o[e]&&(i[n[u][a][e]]=m),e in o?i[o[e]]=a:"production"!==process.env.NODE_ENV&&console.warn(`Index not found in entity. Indexes must be top-level members of your entity.\nIndex: ${e}\nEntity: ${JSON.stringify(o,void 0,2)}`)}}n[u][a]=e[u][a]})({},{},c,a,u,i);return{entities:c,indexes:a,result:v(e,e,void 0,t,f,{}),entityMeta:u}};function j(e,t,n,r={}){if(function(e){return!!e&&"function"==typeof e.infer}(e))return e.infer(t,n,j,r);if("object"==typeof e&&e){return(Array.isArray(e)?i:u)(e,t,n,j,r)}return e}var x={Invalid:1,InvalidIfStale:2,Valid:3};Object.hasOwn=Object.hasOwn||function(e,t){return Object.prototype.hasOwnProperty.call(e,t)};export{m as DELETED,x as ExpiryStatus,f as WeakEntityMap,b as denormalize,j as inferResults,e as isEntity,O as normalize}; |
@@ -133,6 +133,69 @@ 'use strict'; | ||
/** Maps entity dependencies to a value (usually their denormalized form) | ||
* | ||
* Dependencies store `Path` to enable quick traversal using only `State` | ||
* If *any* members of the dependency get cleaned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakEntityMap { | ||
constructor() { | ||
this.next = new WeakMap(); | ||
} | ||
get(entity, getEntity) { | ||
let curLink = this.next.get(entity); | ||
if (!curLink) return EMPTY; | ||
while (curLink.nextPath) { | ||
const nextEntity = getEntity(curLink.nextPath); | ||
curLink = curLink.next.get(nextEntity); | ||
if (!curLink) return EMPTY; | ||
} | ||
// curLink exists, but has no path - so must have a value | ||
return [curLink.value, curLink.journey]; | ||
} | ||
set(dependencies, value) { | ||
if (dependencies.length < 1) throw new KeySize(); | ||
let curLink = this; | ||
for (const { | ||
entity, | ||
path | ||
} of dependencies) { | ||
let nextLink = curLink.next.get(entity); | ||
if (!nextLink) { | ||
nextLink = new Link(); | ||
curLink.next.set(entity, nextLink); | ||
} | ||
curLink.nextPath = path; | ||
curLink = nextLink; | ||
} | ||
// in case there used to be more | ||
delete curLink.nextPath; | ||
curLink.value = value; | ||
// we could recompute this on get, but it would have a cost and we optimize for `get` | ||
curLink.journey = depToPaths(dependencies); | ||
} | ||
} | ||
const EMPTY = [undefined, undefined]; | ||
function getEntities(state) { | ||
const entityIsImmutable = isImmutable(state); | ||
if (entityIsImmutable) { | ||
return ({ | ||
key, | ||
pk | ||
}) => state.getIn([key, pk]); | ||
} else { | ||
return ({ | ||
key, | ||
pk | ||
}) => { | ||
var _state$key; | ||
return (_state$key = state[key]) == null ? void 0 : _state$key[pk]; | ||
}; | ||
} | ||
} | ||
function depToPaths(dependencies) { | ||
return dependencies.map(dep => dep.path); | ||
} | ||
/** Link in a chain */ | ||
class Link { | ||
constructor() { | ||
this.children = new WeakMap(); | ||
this.next = new WeakMap(); | ||
} | ||
@@ -147,57 +210,7 @@ } | ||
/** Maps from a list of objects (referentially) to any value | ||
* | ||
* If *any* members of the list get claned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakListMap { | ||
constructor() { | ||
this.first = new WeakMap(); | ||
} | ||
delete(key) { | ||
const link = this.traverse(key); | ||
link == null ? true : delete link.value; | ||
return !!link; | ||
} | ||
get(key) { | ||
const link = this.traverse(key); | ||
return link == null ? void 0 : link.value; | ||
} | ||
has(key) { | ||
const link = this.traverse(key); | ||
if (!link) return false; | ||
return Object.hasOwn(link, 'value'); | ||
} | ||
set(key, value) { | ||
if (key.length < 1) throw new KeySize(); | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
if (!cur.has(key[i])) { | ||
link = new Link(); | ||
cur.set(key[i], link); | ||
} else { | ||
link = cur.get(key[i]); | ||
} | ||
cur = link.children; | ||
// do on later iteration of loop. this makes typescript happy rather than putting after loop | ||
if (i === key.length - 1) { | ||
link.value = value; | ||
} | ||
} | ||
return this; | ||
} | ||
traverse(key) { | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
link = cur.get(key[i]); | ||
if (!link) return; | ||
cur = link.children; | ||
} | ||
return link; | ||
} | ||
} | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex) => { | ||
const entity = getEntity(entityOrId, schema); | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex) => { | ||
const entity = typeof entityOrId === 'object' ? entityOrId : getEntity({ | ||
pk: entityOrId, | ||
key: schema.key | ||
}); | ||
if (typeof entity === 'symbol' && entity.toString().includes('DELETED')) { | ||
@@ -230,45 +243,62 @@ return [undefined, true, true]; | ||
} | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
const localCacheKey = localCache[key]; | ||
const cycleCacheKey = cycleCache[key]; | ||
const entityCacheKey = entityCache[key]; | ||
let found = true; | ||
let deleted = false; | ||
// local cache lookup first | ||
if (!localCacheKey[pk]) { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push(entity); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk); | ||
const globalCache = getCache(pk, schema); | ||
const [cacheValue] = globalCache.get(entity, getEntity); | ||
// TODO: what if this just returned the deps - then we don't need to store them | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
if (!globalCacheEntry.has(localKey)) { | ||
globalCacheEntry.set(localKey, localCacheKey[pk]); | ||
} else { | ||
localCacheKey[pk] = globalCacheEntry.get(localKey); | ||
if (cacheValue) { | ||
localCacheKey[pk] = cacheValue.value[0]; | ||
// TODO: can we store the cache values instead of tracking *all* their sources? | ||
// this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it | ||
dependencies.push(...cacheValue.dependencies); | ||
return cacheValue.value; | ||
} | ||
// if we don't find in denormalize cache then do full denormalize | ||
else { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
const cacheValue = { | ||
dependencies: localKey, | ||
value: [localCacheKey[pk], found, deleted] | ||
}; | ||
globalCache.set(localKey, cacheValue); | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
} | ||
} | ||
@@ -281,3 +311,9 @@ } else { | ||
// with no cycle, globalCacheEntry will have already been set | ||
dependencies.push(entity); | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
} | ||
@@ -287,4 +323,6 @@ } | ||
}; | ||
const getUnvisit = (entities, entityCache, resultCache, localCache) => { | ||
const getUnvisit = (entities, entityCache, resultCache) => { | ||
const getEntity = getEntities(entities); | ||
const getCache = getEntityCaches(entityCache); | ||
const localCache = {}; | ||
const dependencies = []; | ||
@@ -303,2 +341,4 @@ const cycleIndex = { | ||
const hasDenormalize = typeof schema.denormalize === 'function'; | ||
// deserialize fields (like Date) | ||
if (!hasDenormalize && typeof schema === 'function') { | ||
@@ -318,3 +358,3 @@ if (input instanceof schema) return [input, true, false]; | ||
if (isEntity(schema)) { | ||
return unvisitEntity(input, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex); | ||
return unvisitEntity(input, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex); | ||
} | ||
@@ -326,50 +366,57 @@ if (hasDenormalize) { | ||
} | ||
//const wrappedUnvisit = withTrackedEntities(unvisit, globalKey); | ||
return (input, schema) => { | ||
const ret = unvisit(input, schema); | ||
// in the case where WeakMap cannot be used | ||
// this test ensures null is properly excluded from WeakMap | ||
if (Object(input) !== input) return ret; | ||
dependencies.push(input); | ||
if (!resultCache.has(dependencies)) { | ||
resultCache.set(dependencies, ret[0]); | ||
return ret; | ||
} else { | ||
return [resultCache.get(dependencies), ret[1], ret[2]]; | ||
const cachable = Object(input) === input && Object(schema) === schema; | ||
if (!cachable) { | ||
const ret = unvisit(input, schema); | ||
// this is faster than spread | ||
// https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples | ||
return [ret[0], ret[1], ret[2], depToPaths(dependencies)]; | ||
} | ||
let [ret, entityPaths] = resultCache.get(input, getEntity); | ||
if (ret === undefined) { | ||
ret = unvisit(input, schema); | ||
// we want to do this before we add our 'input' entry | ||
entityPaths = depToPaths(dependencies); | ||
// for the first entry, `path` is ignored so empty members is fine | ||
dependencies.unshift({ | ||
entity: input, | ||
path: { | ||
key: '', | ||
pk: '' | ||
} | ||
}); | ||
resultCache.set(dependencies, ret); | ||
} | ||
return [ret[0], ret[1], ret[2], entityPaths]; | ||
}; | ||
}; | ||
const getEntities = entities => { | ||
const entityIsImmutable = isImmutable(entities); | ||
return (entityOrId, schema) => { | ||
var _entities$schemaKey; | ||
const schemaKey = schema.key; | ||
if (typeof entityOrId === 'object') { | ||
return entityOrId; | ||
const getEntityCaches = entityCache => { | ||
return (pk, schema) => { | ||
const key = schema.key; | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
if (entityIsImmutable) { | ||
return entities.getIn([schemaKey, entityOrId]); | ||
const entityCacheKey = entityCache[key]; | ||
if (!entityCacheKey[pk]) entityCacheKey[pk] = new WeakMap(); | ||
let wem = entityCacheKey[pk].get(schema); | ||
if (!wem) { | ||
wem = new WeakEntityMap(); | ||
entityCacheKey[pk].set(schema, wem); | ||
} | ||
return (_entities$schemaKey = entities[schemaKey]) == null ? void 0 : _entities$schemaKey[entityOrId]; | ||
return wem; | ||
}; | ||
}; | ||
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => { | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakEntityMap()) => { | ||
// undefined mean don't do anything | ||
if (schema === undefined) { | ||
return [input, true, false, {}]; | ||
return [input, true, false, []]; | ||
} | ||
if (input === undefined) { | ||
return [undefined, false, false, {}]; | ||
return [undefined, false, false, []]; | ||
} | ||
const resolvedEntities = {}; | ||
const unvisit = getUnvisit(entities, entityCache, resultCache, resolvedEntities); | ||
return [...unvisit(input, schema), resolvedEntities]; | ||
return getUnvisit(entities, entityCache, resultCache)(input, schema); | ||
}; | ||
function getGlobalCacheEntry(entityCache, id) { | ||
if (!entityCache[id]) entityCache[id] = new WeakListMap(); | ||
return entityCache[id]; | ||
} | ||
@@ -605,3 +652,3 @@ // TODO(breaking): remove once unused | ||
exports.ExpiryStatus = ExpiryStatus; | ||
exports.WeakListMap = WeakListMap; | ||
exports.WeakEntityMap = WeakEntityMap; | ||
exports.denormalize = denormalize; | ||
@@ -608,0 +655,0 @@ exports.inferResults = inferResults; |
@@ -131,6 +131,69 @@ (function (global, factory) { | ||
/** Maps entity dependencies to a value (usually their denormalized form) | ||
* | ||
* Dependencies store `Path` to enable quick traversal using only `State` | ||
* If *any* members of the dependency get cleaned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakEntityMap { | ||
constructor() { | ||
this.next = new WeakMap(); | ||
} | ||
get(entity, getEntity) { | ||
let curLink = this.next.get(entity); | ||
if (!curLink) return EMPTY; | ||
while (curLink.nextPath) { | ||
const nextEntity = getEntity(curLink.nextPath); | ||
curLink = curLink.next.get(nextEntity); | ||
if (!curLink) return EMPTY; | ||
} | ||
// curLink exists, but has no path - so must have a value | ||
return [curLink.value, curLink.journey]; | ||
} | ||
set(dependencies, value) { | ||
if (dependencies.length < 1) throw new KeySize(); | ||
let curLink = this; | ||
for (const { | ||
entity, | ||
path | ||
} of dependencies) { | ||
let nextLink = curLink.next.get(entity); | ||
if (!nextLink) { | ||
nextLink = new Link(); | ||
curLink.next.set(entity, nextLink); | ||
} | ||
curLink.nextPath = path; | ||
curLink = nextLink; | ||
} | ||
// in case there used to be more | ||
delete curLink.nextPath; | ||
curLink.value = value; | ||
// we could recompute this on get, but it would have a cost and we optimize for `get` | ||
curLink.journey = depToPaths(dependencies); | ||
} | ||
} | ||
const EMPTY = [undefined, undefined]; | ||
function getEntities(state) { | ||
const entityIsImmutable = isImmutable(state); | ||
if (entityIsImmutable) { | ||
return ({ | ||
key, | ||
pk | ||
}) => state.getIn([key, pk]); | ||
} else { | ||
return ({ | ||
key, | ||
pk | ||
}) => { | ||
var _state$key; | ||
return (_state$key = state[key]) == null ? void 0 : _state$key[pk]; | ||
}; | ||
} | ||
} | ||
function depToPaths(dependencies) { | ||
return dependencies.map(dep => dep.path); | ||
} | ||
/** Link in a chain */ | ||
class Link { | ||
constructor() { | ||
this.children = new WeakMap(); | ||
this.next = new WeakMap(); | ||
} | ||
@@ -145,57 +208,7 @@ } | ||
/** Maps from a list of objects (referentially) to any value | ||
* | ||
* If *any* members of the list get claned up, so does that key/value pair get removed. | ||
*/ | ||
class WeakListMap { | ||
constructor() { | ||
this.first = new WeakMap(); | ||
} | ||
delete(key) { | ||
const link = this.traverse(key); | ||
link == null ? true : delete link.value; | ||
return !!link; | ||
} | ||
get(key) { | ||
const link = this.traverse(key); | ||
return link == null ? void 0 : link.value; | ||
} | ||
has(key) { | ||
const link = this.traverse(key); | ||
if (!link) return false; | ||
return Object.hasOwn(link, 'value'); | ||
} | ||
set(key, value) { | ||
if (key.length < 1) throw new KeySize(); | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
if (!cur.has(key[i])) { | ||
link = new Link(); | ||
cur.set(key[i], link); | ||
} else { | ||
link = cur.get(key[i]); | ||
} | ||
cur = link.children; | ||
// do on later iteration of loop. this makes typescript happy rather than putting after loop | ||
if (i === key.length - 1) { | ||
link.value = value; | ||
} | ||
} | ||
return this; | ||
} | ||
traverse(key) { | ||
let cur = this.first; | ||
let link; | ||
for (let i = 0; i < key.length; i++) { | ||
link = cur.get(key[i]); | ||
if (!link) return; | ||
cur = link.children; | ||
} | ||
return link; | ||
} | ||
} | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex) => { | ||
const entity = getEntity(entityOrId, schema); | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex) => { | ||
const entity = typeof entityOrId === 'object' ? entityOrId : getEntity({ | ||
pk: entityOrId, | ||
key: schema.key | ||
}); | ||
if (typeof entity === 'symbol' && entity.toString().includes('DELETED')) { | ||
@@ -228,45 +241,62 @@ return [undefined, true, true]; | ||
} | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
const localCacheKey = localCache[key]; | ||
const cycleCacheKey = cycleCache[key]; | ||
const entityCacheKey = entityCache[key]; | ||
let found = true; | ||
let deleted = false; | ||
// local cache lookup first | ||
if (!localCacheKey[pk]) { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push(entity); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk); | ||
const globalCache = getCache(pk, schema); | ||
const [cacheValue] = globalCache.get(entity, getEntity); | ||
// TODO: what if this just returned the deps - then we don't need to store them | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
if (!globalCacheEntry.has(localKey)) { | ||
globalCacheEntry.set(localKey, localCacheKey[pk]); | ||
} else { | ||
localCacheKey[pk] = globalCacheEntry.get(localKey); | ||
if (cacheValue) { | ||
localCacheKey[pk] = cacheValue.value[0]; | ||
// TODO: can we store the cache values instead of tracking *all* their sources? | ||
// this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it | ||
dependencies.push(...cacheValue.dependencies); | ||
return cacheValue.value; | ||
} | ||
// if we don't find in denormalize cache then do full denormalize | ||
else { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
const cacheValue = { | ||
dependencies: localKey, | ||
value: [localCacheKey[pk], found, deleted] | ||
}; | ||
globalCache.set(localKey, cacheValue); | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
} | ||
} | ||
@@ -279,3 +309,9 @@ } else { | ||
// with no cycle, globalCacheEntry will have already been set | ||
dependencies.push(entity); | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
} | ||
@@ -285,4 +321,6 @@ } | ||
}; | ||
const getUnvisit = (entities, entityCache, resultCache, localCache) => { | ||
const getUnvisit = (entities, entityCache, resultCache) => { | ||
const getEntity = getEntities(entities); | ||
const getCache = getEntityCaches(entityCache); | ||
const localCache = {}; | ||
const dependencies = []; | ||
@@ -301,2 +339,4 @@ const cycleIndex = { | ||
const hasDenormalize = typeof schema.denormalize === 'function'; | ||
// deserialize fields (like Date) | ||
if (!hasDenormalize && typeof schema === 'function') { | ||
@@ -316,3 +356,3 @@ if (input instanceof schema) return [input, true, false]; | ||
if (isEntity(schema)) { | ||
return unvisitEntity(input, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex); | ||
return unvisitEntity(input, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex); | ||
} | ||
@@ -324,50 +364,57 @@ if (hasDenormalize) { | ||
} | ||
//const wrappedUnvisit = withTrackedEntities(unvisit, globalKey); | ||
return (input, schema) => { | ||
const ret = unvisit(input, schema); | ||
// in the case where WeakMap cannot be used | ||
// this test ensures null is properly excluded from WeakMap | ||
if (Object(input) !== input) return ret; | ||
dependencies.push(input); | ||
if (!resultCache.has(dependencies)) { | ||
resultCache.set(dependencies, ret[0]); | ||
return ret; | ||
} else { | ||
return [resultCache.get(dependencies), ret[1], ret[2]]; | ||
const cachable = Object(input) === input && Object(schema) === schema; | ||
if (!cachable) { | ||
const ret = unvisit(input, schema); | ||
// this is faster than spread | ||
// https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples | ||
return [ret[0], ret[1], ret[2], depToPaths(dependencies)]; | ||
} | ||
let [ret, entityPaths] = resultCache.get(input, getEntity); | ||
if (ret === undefined) { | ||
ret = unvisit(input, schema); | ||
// we want to do this before we add our 'input' entry | ||
entityPaths = depToPaths(dependencies); | ||
// for the first entry, `path` is ignored so empty members is fine | ||
dependencies.unshift({ | ||
entity: input, | ||
path: { | ||
key: '', | ||
pk: '' | ||
} | ||
}); | ||
resultCache.set(dependencies, ret); | ||
} | ||
return [ret[0], ret[1], ret[2], entityPaths]; | ||
}; | ||
}; | ||
const getEntities = entities => { | ||
const entityIsImmutable = isImmutable(entities); | ||
return (entityOrId, schema) => { | ||
var _entities$schemaKey; | ||
const schemaKey = schema.key; | ||
if (typeof entityOrId === 'object') { | ||
return entityOrId; | ||
const getEntityCaches = entityCache => { | ||
return (pk, schema) => { | ||
const key = schema.key; | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
if (entityIsImmutable) { | ||
return entities.getIn([schemaKey, entityOrId]); | ||
const entityCacheKey = entityCache[key]; | ||
if (!entityCacheKey[pk]) entityCacheKey[pk] = new WeakMap(); | ||
let wem = entityCacheKey[pk].get(schema); | ||
if (!wem) { | ||
wem = new WeakEntityMap(); | ||
entityCacheKey[pk].set(schema, wem); | ||
} | ||
return (_entities$schemaKey = entities[schemaKey]) == null ? void 0 : _entities$schemaKey[entityOrId]; | ||
return wem; | ||
}; | ||
}; | ||
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => { | ||
const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakEntityMap()) => { | ||
// undefined mean don't do anything | ||
if (schema === undefined) { | ||
return [input, true, false, {}]; | ||
return [input, true, false, []]; | ||
} | ||
if (input === undefined) { | ||
return [undefined, false, false, {}]; | ||
return [undefined, false, false, []]; | ||
} | ||
const resolvedEntities = {}; | ||
const unvisit = getUnvisit(entities, entityCache, resultCache, resolvedEntities); | ||
return [...unvisit(input, schema), resolvedEntities]; | ||
return getUnvisit(entities, entityCache, resultCache)(input, schema); | ||
}; | ||
function getGlobalCacheEntry(entityCache, id) { | ||
if (!entityCache[id]) entityCache[id] = new WeakListMap(); | ||
return entityCache[id]; | ||
} | ||
@@ -596,3 +643,3 @@ // TODO(breaking): remove once unused | ||
exports.ExpiryStatus = ExpiryStatus; | ||
exports.WeakListMap = WeakListMap; | ||
exports.WeakEntityMap = WeakEntityMap; | ||
exports.denormalize = denormalize; | ||
@@ -599,0 +646,0 @@ exports.inferResults = inferResults; |
@@ -5,5 +5,8 @@ import { isEntity } from './isEntity.js'; | ||
import { denormalize as objectDenormalize } from './schemas/Object.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex) => { | ||
const entity = getEntity(entityOrId, schema); | ||
import WeakEntityMap, { getEntities, depToPaths } from './WeakEntityMap.js'; | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex) => { | ||
const entity = typeof entityOrId === 'object' ? entityOrId : getEntity({ | ||
pk: entityOrId, | ||
key: schema.key | ||
}); | ||
if (typeof entity === 'symbol' && entity.toString().includes('DELETED')) { | ||
@@ -36,45 +39,62 @@ return [undefined, true, true]; | ||
} | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
const localCacheKey = localCache[key]; | ||
const cycleCacheKey = cycleCache[key]; | ||
const entityCacheKey = entityCache[key]; | ||
let found = true; | ||
let deleted = false; | ||
// local cache lookup first | ||
if (!localCacheKey[pk]) { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push(entity); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk); | ||
const globalCache = getCache(pk, schema); | ||
const [cacheValue] = globalCache.get(entity, getEntity); | ||
// TODO: what if this just returned the deps - then we don't need to store them | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
if (!globalCacheEntry.has(localKey)) { | ||
globalCacheEntry.set(localKey, localCacheKey[pk]); | ||
} else { | ||
localCacheKey[pk] = globalCacheEntry.get(localKey); | ||
if (cacheValue) { | ||
localCacheKey[pk] = cacheValue.value[0]; | ||
// TODO: can we store the cache values instead of tracking *all* their sources? | ||
// this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it | ||
dependencies.push(...cacheValue.dependencies); | ||
return cacheValue.value; | ||
} | ||
// if we don't find in denormalize cache then do full denormalize | ||
else { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
const cacheValue = { | ||
dependencies: localKey, | ||
value: [localCacheKey[pk], found, deleted] | ||
}; | ||
globalCache.set(localKey, cacheValue); | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
} | ||
} | ||
@@ -87,3 +107,9 @@ } else { | ||
// with no cycle, globalCacheEntry will have already been set | ||
dependencies.push(entity); | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
} | ||
@@ -93,4 +119,6 @@ } | ||
}; | ||
const getUnvisit = (entities, entityCache, resultCache, localCache) => { | ||
const getUnvisit = (entities, entityCache, resultCache) => { | ||
const getEntity = getEntities(entities); | ||
const getCache = getEntityCaches(entityCache); | ||
const localCache = {}; | ||
const dependencies = []; | ||
@@ -109,2 +137,4 @@ const cycleIndex = { | ||
const hasDenormalize = typeof schema.denormalize === 'function'; | ||
// deserialize fields (like Date) | ||
if (!hasDenormalize && typeof schema === 'function') { | ||
@@ -124,3 +154,3 @@ if (input instanceof schema) return [input, true, false]; | ||
if (isEntity(schema)) { | ||
return unvisitEntity(input, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex); | ||
return unvisitEntity(input, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex); | ||
} | ||
@@ -132,51 +162,58 @@ if (hasDenormalize) { | ||
} | ||
//const wrappedUnvisit = withTrackedEntities(unvisit, globalKey); | ||
return (input, schema) => { | ||
const ret = unvisit(input, schema); | ||
// in the case where WeakMap cannot be used | ||
// this test ensures null is properly excluded from WeakMap | ||
if (Object(input) !== input) return ret; | ||
dependencies.push(input); | ||
if (!resultCache.has(dependencies)) { | ||
resultCache.set(dependencies, ret[0]); | ||
return ret; | ||
} else { | ||
return [resultCache.get(dependencies), ret[1], ret[2]]; | ||
const cachable = Object(input) === input && Object(schema) === schema; | ||
if (!cachable) { | ||
const ret = unvisit(input, schema); | ||
// this is faster than spread | ||
// https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples | ||
return [ret[0], ret[1], ret[2], depToPaths(dependencies)]; | ||
} | ||
let [ret, entityPaths] = resultCache.get(input, getEntity); | ||
if (ret === undefined) { | ||
ret = unvisit(input, schema); | ||
// we want to do this before we add our 'input' entry | ||
entityPaths = depToPaths(dependencies); | ||
// for the first entry, `path` is ignored so empty members is fine | ||
dependencies.unshift({ | ||
entity: input, | ||
path: { | ||
key: '', | ||
pk: '' | ||
} | ||
}); | ||
resultCache.set(dependencies, ret); | ||
} | ||
return [ret[0], ret[1], ret[2], entityPaths]; | ||
}; | ||
}; | ||
const getEntities = entities => { | ||
const entityIsImmutable = isImmutable(entities); | ||
return (entityOrId, schema) => { | ||
var _entities$schemaKey; | ||
const schemaKey = schema.key; | ||
if (typeof entityOrId === 'object') { | ||
return entityOrId; | ||
const getEntityCaches = entityCache => { | ||
return (pk, schema) => { | ||
const key = schema.key; | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
if (entityIsImmutable) { | ||
return entities.getIn([schemaKey, entityOrId]); | ||
const entityCacheKey = entityCache[key]; | ||
if (!entityCacheKey[pk]) entityCacheKey[pk] = new WeakMap(); | ||
let wem = entityCacheKey[pk].get(schema); | ||
if (!wem) { | ||
wem = new WeakEntityMap(); | ||
entityCacheKey[pk].set(schema, wem); | ||
} | ||
return (_entities$schemaKey = entities[schemaKey]) == null ? void 0 : _entities$schemaKey[entityOrId]; | ||
return wem; | ||
}; | ||
}; | ||
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types | ||
export const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => { | ||
export const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakEntityMap()) => { | ||
// undefined mean don't do anything | ||
if (schema === undefined) { | ||
return [input, true, false, {}]; | ||
return [input, true, false, []]; | ||
} | ||
if (input === undefined) { | ||
return [undefined, false, false, {}]; | ||
return [undefined, false, false, []]; | ||
} | ||
const resolvedEntities = {}; | ||
const unvisit = getUnvisit(entities, entityCache, resultCache, resolvedEntities); | ||
return [...unvisit(input, schema), resolvedEntities]; | ||
return getUnvisit(entities, entityCache, resultCache)(input, schema); | ||
}; | ||
export const denormalizeSimple = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => denormalize(input, schema, entities, entityCache, resultCache).slice(0, 3); | ||
function getGlobalCacheEntry(entityCache, id) { | ||
if (!entityCache[id]) entityCache[id] = new WeakListMap(); | ||
return entityCache[id]; | ||
} | ||
export const denormalizeSimple = (input, schema, entities, entityCache, resultCache) => denormalize(input, schema, entities, entityCache, resultCache).slice(0, 3); | ||
@@ -192,2 +229,2 @@ // TODO(breaking): remove once unused | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["isEntity","denormalize","arrayDenormalize","isImmutable","objectDenormalize","WeakListMap","unvisitEntity","entityOrId","schema","unvisit","getEntity","localCache","cycleCache","entityCache","dependencies","cycleIndex","entity","toString","includes","undefined","pk","toJS","key","Object","create","localCacheKey","cycleCacheKey","entityCacheKey","found","deleted","trackingIndex","length","push","entityCopy","createIfValid","toObject","withTrackedEntities","setLocal","globalCacheEntry","getGlobalCacheEntry","localKey","slice","i","has","set","get","getUnvisit","entities","resultCache","getEntities","input","hasDenormalize","method","Array","isArray","ret","entityIsImmutable","schemaKey","getIn","resolvedEntities","denormalizeSimple","id","originalUnvisit","og","wrappedUnvisit"],"sources":["../src/denormalize.ts"],"sourcesContent":["import type { Schema, EntityInterface, UnvisitFunction } from './interface.js';\nimport { isEntity } from './isEntity.js';\nimport { denormalize as arrayDenormalize } from './schemas/Array.js';\nimport { isImmutable } from './schemas/ImmutableUtils.js';\nimport { denormalize as objectDenormalize } from './schemas/Object.js';\nimport type {\n  Denormalize,\n  DenormalizeNullable,\n  DenormalizeCache,\n} from './types.js';\nimport WeakListMap from './WeakListMap.js';\n\nconst unvisitEntity = (\n  entityOrId: Record<string, any> | string,\n  schema: EntityInterface,\n  unvisit: UnvisitFunction,\n  getEntity: (\n    entityOrId: Record<string, any> | string,\n    schema: EntityInterface,\n  ) => object | symbol,\n  localCache: Record<string, Record<string, any>>,\n  cycleCache: Record<string, Record<string, number>>,\n  entityCache: DenormalizeCache['entities'],\n  dependencies: object[],\n  cycleIndex: { i: number },\n): [denormalized: object | undefined, found: boolean, deleted: boolean] => {\n  const entity = getEntity(entityOrId, schema);\n  if (\n    typeof entity === 'symbol' &&\n    (entity as symbol).toString().includes('DELETED')\n  ) {\n    return [undefined, true, true];\n    // TODO: Change to this as breaking change once we only support newer entities\n    // also remove `(entity as symbol).toString().includes('DELETED')` and do this for all symbols\n    // return schema.denormalize(entity, unvisit);\n  }\n\n  if (typeof entity !== 'object' || entity === null) {\n    return [entity as any, false, false];\n  }\n\n  const pk =\n    // normalize must always place a string, because pk() return value is string | undefined\n    // therefore no need to check for numbers\n    typeof entityOrId === 'string'\n      ? entityOrId\n      : schema.pk(isImmutable(entity) ? (entity as any).toJS() : entity);\n  // if we can't generate a working pk; this is hopeless so let's give them what's already there\n  // otherwise, even when we aren't doing a lookup we want to turn the entityOrId object into the\n  // expected class, and cache that class for referential equality. PK is used for global equality lookup.\n  if (pk === undefined || pk === '' || pk === 'undefined') {\n    return [entity, false, false];\n  }\n\n  const key = schema.key;\n  if (!(key in localCache)) {\n    localCache[key] = Object.create(null);\n  }\n  if (!(key in cycleCache)) {\n    cycleCache[key] = Object.create(null);\n  }\n  if (!(key in entityCache)) {\n    entityCache[key] = Object.create(null);\n  }\n  const localCacheKey = localCache[key];\n  const cycleCacheKey = cycleCache[key];\n  const entityCacheKey = entityCache[key];\n\n  let found = true;\n  let deleted = false;\n\n  if (!localCacheKey[pk]) {\n    const trackingIndex = dependencies.length;\n    dependencies.push(entity);\n\n    let entityCopy: any;\n    if (schema.createIfValid) {\n      entityCopy = localCacheKey[pk] = isImmutable(entity)\n        ? schema.createIfValid(entity.toObject())\n        : schema.createIfValid(entity);\n      // TODO(breaking): remove once old verions no longer supported\n    } else {\n      entityCopy = entity;\n      unvisit = withTrackedEntities(unvisit);\n      unvisit.setLocal = entityCopy => (localCacheKey[pk] = entityCopy);\n    }\n\n    cycleCacheKey[pk] = trackingIndex;\n    if (entityCopy === undefined) {\n      // undefined indicates we should suspense (perhaps failed validation)\n      found = false;\n      deleted = true;\n    } else {\n      [localCacheKey[pk], found, deleted] = schema.denormalize(\n        entityCopy,\n        unvisit,\n      );\n    }\n    delete cycleCacheKey[pk];\n\n    const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk);\n\n    // if in cycle, use the start of the cycle to track all deps\n    // otherwise, we use our own trackingIndex\n    const localKey = dependencies.slice(\n      cycleIndex.i === -1 ? trackingIndex : cycleIndex.i,\n    );\n\n    if (!globalCacheEntry.has(localKey)) {\n      globalCacheEntry.set(localKey, localCacheKey[pk]);\n    } else {\n      localCacheKey[pk] = globalCacheEntry.get(localKey);\n    }\n\n    // start of cycle - reset cycle detection\n    if (cycleIndex.i === trackingIndex) {\n      cycleIndex.i = -1;\n    }\n  } else {\n    // cycle detected\n    if (pk in cycleCacheKey) {\n      cycleIndex.i = cycleCacheKey[pk];\n    } else {\n      // with no cycle, globalCacheEntry will have already been set\n      dependencies.push(entity);\n    }\n  }\n\n  return [localCacheKey[pk], found, deleted];\n};\n\nconst getUnvisit = (\n  entities: Record<string, Record<string, any>>,\n  entityCache: DenormalizeCache['entities'],\n  resultCache: WeakListMap<object, any>,\n  localCache: Record<string, Record<string, any>>,\n) => {\n  const getEntity = getEntities(entities);\n  const dependencies: object[] = [];\n  const cycleIndex = { i: -1 };\n  const cycleCache = {};\n\n  function unvisit(\n    input: any,\n    schema: any,\n  ): [denormalized: any, found: boolean, deleted: boolean] {\n    if (!schema) return [input, true, false];\n\n    // null is considered intentional, thus always 'found' as true\n    if (input === null) {\n      return [input, true, false];\n    }\n\n    const hasDenormalize = typeof schema.denormalize === 'function';\n\n    if (!hasDenormalize && typeof schema === 'function') {\n      if (input instanceof schema) return [input, true, false];\n      // field deserialization should never count against 'found' (whether to used inferred results)\n      if (input === undefined) return [input, true, false];\n      return [new schema(input), true, false];\n    }\n\n    if (input === undefined) {\n      return [input, false, false];\n    }\n\n    if (!hasDenormalize && typeof schema === 'object') {\n      const method = Array.isArray(schema)\n        ? arrayDenormalize\n        : objectDenormalize;\n      return method(schema, input, unvisit);\n    }\n\n    if (isEntity(schema)) {\n      return unvisitEntity(\n        input,\n        schema,\n        unvisit,\n        getEntity,\n        localCache,\n        cycleCache,\n        entityCache,\n        dependencies,\n        cycleIndex,\n      );\n    }\n\n    if (hasDenormalize) {\n      return schema.denormalize(input, unvisit);\n    }\n\n    return [input, true, false];\n  }\n\n  //const wrappedUnvisit = withTrackedEntities(unvisit, globalKey);\n\n  return (\n    input: any,\n    schema: any,\n  ): [denormalized: any, found: boolean, deleted: boolean] => {\n    const ret = unvisit(input, schema);\n    // in the case where WeakMap cannot be used\n    // this test ensures null is properly excluded from WeakMap\n    if (Object(input) !== input) return ret;\n\n    dependencies.push(input);\n    if (!resultCache.has(dependencies)) {\n      resultCache.set(dependencies, ret[0]);\n      return ret;\n    } else {\n      return [resultCache.get(dependencies), ret[1], ret[2]];\n    }\n  };\n};\n\nconst getEntities = (entities: Record<string, any>) => {\n  const entityIsImmutable = isImmutable(entities);\n\n  return (\n    entityOrId: Record<string, any> | string,\n    schema: EntityInterface,\n  ) => {\n    const schemaKey = schema.key;\n\n    if (typeof entityOrId === 'object') {\n      return entityOrId;\n    }\n\n    if (entityIsImmutable) {\n      return entities.getIn([schemaKey, entityOrId]);\n    }\n\n    return entities[schemaKey]?.[entityOrId];\n  };\n};\n\ntype DenormalizeReturn<S extends Schema> =\n  | [\n      denormalized: Denormalize<S>,\n      found: true,\n      deleted: false,\n      resolvedEntities: Record<string, Record<string, any>>,\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: boolean,\n      deleted: true,\n      resolvedEntities: Record<string, Record<string, any>>,\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: false,\n      deleted: boolean,\n      resolvedEntities: Record<string, Record<string, any>>,\n    ];\n\n// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types\nexport const denormalize = <S extends Schema>(\n  input: unknown,\n  schema: S | undefined,\n  entities: any,\n  entityCache: DenormalizeCache['entities'] = {},\n  resultCache: WeakListMap<object, any> = new WeakListMap(),\n): DenormalizeReturn<S> => {\n  // undefined mean don't do anything\n  if (schema === undefined) {\n    return [input, true, false, {}] as [any, boolean, boolean, any];\n  }\n  if (input === undefined) {\n    return [undefined, false, false, {}] as [any, boolean, boolean, any];\n  }\n  const resolvedEntities: Record<string, Record<string, any>> = {};\n  const unvisit = getUnvisit(\n    entities,\n    entityCache,\n    resultCache,\n    resolvedEntities,\n  );\n  return [...unvisit(input, schema), resolvedEntities] as [\n    any,\n    boolean,\n    boolean,\n    any,\n  ];\n};\n\nexport const denormalizeSimple = <S extends Schema>(\n  input: any,\n  schema: S | undefined,\n  entities: any,\n  entityCache: DenormalizeCache['entities'] = {},\n  resultCache: WeakListMap<object, any> = new WeakListMap(),\n):\n  | [denormalized: Denormalize<S>, found: true, deleted: false]\n  | [denormalized: DenormalizeNullable<S>, found: boolean, deleted: true]\n  | [denormalized: DenormalizeNullable<S>, found: false, deleted: boolean] =>\n  denormalize(input, schema, entities, entityCache, resultCache).slice(\n    0,\n    3,\n  ) as any;\n\nfunction getGlobalCacheEntry(\n  entityCache: { [pk: string]: WeakListMap<object, EntityInterface<any>> },\n\n  id: any,\n) {\n  if (!entityCache[id]) entityCache[id] = new WeakListMap();\n  return entityCache[id];\n}\n\n// TODO(breaking): remove once unused\nfunction withTrackedEntities(unvisit: UnvisitFunction): UnvisitFunction {\n  // every time we nest, we want to unwrap back to the top.\n  // this is due to only needed the next level of nested entities for lookup\n  const originalUnvisit = unvisit.og || unvisit;\n  const wrappedUnvisit = (input: any, schema: any) =>\n    originalUnvisit(input, schema);\n  wrappedUnvisit.og = unvisit;\n  return wrappedUnvisit;\n}\n"],"mappings":"AACA,SAASA,QAAQ,QAAQ,eAAe;AACxC,SAASC,WAAW,IAAIC,gBAAgB,QAAQ,oBAAoB;AACpE,SAASC,WAAW,QAAQ,6BAA6B;AACzD,SAASF,WAAW,IAAIG,iBAAiB,QAAQ,qBAAqB;AAMtE,OAAOC,WAAW,MAAM,kBAAkB;AAE1C,MAAMC,aAAa,GAAG,CACpBC,UAAwC,EACxCC,MAAuB,EACvBC,OAAwB,EACxBC,SAGoB,EACpBC,UAA+C,EAC/CC,UAAkD,EAClDC,WAAyC,EACzCC,YAAsB,EACtBC,UAAyB,KACgD;EACzE,MAAMC,MAAM,GAAGN,SAAS,CAACH,UAAU,EAAEC,MAAM,CAAC;EAC5C,IACE,OAAOQ,MAAM,KAAK,QAAQ,IACzBA,MAAM,CAAYC,QAAQ,EAAE,CAACC,QAAQ,CAAC,SAAS,CAAC,EACjD;IACA,OAAO,CAACC,SAAS,EAAE,IAAI,EAAE,IAAI,CAAC;IAC9B;IACA;IACA;EACF;;EAEA,IAAI,OAAOH,MAAM,KAAK,QAAQ,IAAIA,MAAM,KAAK,IAAI,EAAE;IACjD,OAAO,CAACA,MAAM,EAAS,KAAK,EAAE,KAAK,CAAC;EACtC;EAEA,MAAMI,EAAE;EACN;EACA;EACA,OAAOb,UAAU,KAAK,QAAQ,GAC1BA,UAAU,GACVC,MAAM,CAACY,EAAE,CAACjB,WAAW,CAACa,MAAM,CAAC,GAAIA,MAAM,CAASK,IAAI,EAAE,GAAGL,MAAM,CAAC;EACtE;EACA;EACA;EACA,IAAII,EAAE,KAAKD,SAAS,IAAIC,EAAE,KAAK,EAAE,IAAIA,EAAE,KAAK,WAAW,EAAE;IACvD,OAAO,CAACJ,MAAM,EAAE,KAAK,EAAE,KAAK,CAAC;EAC/B;EAEA,MAAMM,GAAG,GAAGd,MAAM,CAACc,GAAG;EACtB,IAAI,EAAEA,GAAG,IAAIX,UAAU,CAAC,EAAE;IACxBA,UAAU,CAACW,GAAG,CAAC,GAAGC,MAAM,CAACC,MAAM,CAAC,IAAI,CAAC;EACvC;EACA,IAAI,EAAEF,GAAG,IAAIV,UAAU,CAAC,EAAE;IACxBA,UAAU,CAACU,GAAG,CAAC,GAAGC,MAAM,CAACC,MAAM,CAAC,IAAI,CAAC;EACvC;EACA,IAAI,EAAEF,GAAG,IAAIT,WAAW,CAAC,EAAE;IACzBA,WAAW,CAACS,GAAG,CAAC,GAAGC,MAAM,CAACC,MAAM,CAAC,IAAI,CAAC;EACxC;EACA,MAAMC,aAAa,GAAGd,UAAU,CAACW,GAAG,CAAC;EACrC,MAAMI,aAAa,GAAGd,UAAU,CAACU,GAAG,CAAC;EACrC,MAAMK,cAAc,GAAGd,WAAW,CAACS,GAAG,CAAC;EAEvC,IAAIM,KAAK,GAAG,IAAI;EAChB,IAAIC,OAAO,GAAG,KAAK;EAEnB,IAAI,CAACJ,aAAa,CAACL,EAAE,CAAC,EAAE;IACtB,MAAMU,aAAa,GAAGhB,YAAY,CAACiB,MAAM;IACzCjB,YAAY,CAACkB,IAAI,CAAChB,MAAM,CAAC;IAEzB,IAAIiB,UAAe;IACnB,IAAIzB,MAAM,CAAC0B,aAAa,EAAE;MACxBD,UAAU,GAAGR,aAAa,CAACL,EAAE,CAAC,GAAGjB,WAAW,CAACa,MAAM,CAAC,GAChDR,MAAM,CAAC0B,aAAa,CAAClB,MAAM,CAACmB,QAAQ,EAAE,CAAC,GACvC3B,MAAM,CAAC0B,aAAa,CAAClB,MAAM,CAAC;MAChC;IACF,CAAC,MAAM;MACLiB,UAAU,GAAGjB,MAAM;MACnBP,OAAO,GAAG2B,mBAAmB,CAAC3B,OAAO,CAAC;MACtCA,OAAO,CAAC4B,QAAQ,GAAGJ,UAAU,IAAKR,aAAa,CAACL,EAAE,CAAC,GAAGa,UAAW;IACnE;IAEAP,aAAa,CAACN,EAAE,CAAC,GAAGU,aAAa;IACjC,IAAIG,UAAU,KAAKd,SAAS,EAAE;MAC5B;MACAS,KAAK,GAAG,KAAK;MACbC,OAAO,GAAG,IAAI;IAChB,CAAC,MAAM;MACL,CAACJ,aAAa,CAACL,EAAE,CAAC,EAAEQ,KAAK,EAAEC,OAAO,CAAC,GAAGrB,MAAM,CAACP,WAAW,CACtDgC,UAAU,EACVxB,OAAO,CACR;IACH;IACA,OAAOiB,aAAa,CAACN,EAAE,CAAC;IAExB,MAAMkB,gBAAgB,GAAGC,mBAAmB,CAACZ,cAAc,EAAEP,EAAE,CAAC;;IAEhE;IACA;IACA,MAAMoB,QAAQ,GAAG1B,YAAY,CAAC2B,KAAK,CACjC1B,UAAU,CAAC2B,CAAC,KAAK,CAAC,CAAC,GAAGZ,aAAa,GAAGf,UAAU,CAAC2B,CAAC,CACnD;IAED,IAAI,CAACJ,gBAAgB,CAACK,GAAG,CAACH,QAAQ,CAAC,EAAE;MACnCF,gBAAgB,CAACM,GAAG,CAACJ,QAAQ,EAAEf,aAAa,CAACL,EAAE,CAAC,CAAC;IACnD,CAAC,MAAM;MACLK,aAAa,CAACL,EAAE,CAAC,GAAGkB,gBAAgB,CAACO,GAAG,CAACL,QAAQ,CAAC;IACpD;;IAEA;IACA,IAAIzB,UAAU,CAAC2B,CAAC,KAAKZ,aAAa,EAAE;MAClCf,UAAU,CAAC2B,CAAC,GAAG,CAAC,CAAC;IACnB;EACF,CAAC,MAAM;IACL;IACA,IAAItB,EAAE,IAAIM,aAAa,EAAE;MACvBX,UAAU,CAAC2B,CAAC,GAAGhB,aAAa,CAACN,EAAE,CAAC;IAClC,CAAC,MAAM;MACL;MACAN,YAAY,CAACkB,IAAI,CAAChB,MAAM,CAAC;IAC3B;EACF;EAEA,OAAO,CAACS,aAAa,CAACL,EAAE,CAAC,EAAEQ,KAAK,EAAEC,OAAO,CAAC;AAC5C,CAAC;AAED,MAAMiB,UAAU,GAAG,CACjBC,QAA6C,EAC7ClC,WAAyC,EACzCmC,WAAqC,EACrCrC,UAA+C,KAC5C;EACH,MAAMD,SAAS,GAAGuC,WAAW,CAACF,QAAQ,CAAC;EACvC,MAAMjC,YAAsB,GAAG,EAAE;EACjC,MAAMC,UAAU,GAAG;IAAE2B,CAAC,EAAE,CAAC;EAAE,CAAC;EAC5B,MAAM9B,UAAU,GAAG,CAAC,CAAC;EAErB,SAASH,OAAO,CACdyC,KAAU,EACV1C,MAAW,EAC4C;IACvD,IAAI,CAACA,MAAM,EAAE,OAAO,CAAC0C,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;;IAExC;IACA,IAAIA,KAAK,KAAK,IAAI,EAAE;MAClB,OAAO,CAACA,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;IAC7B;IAEA,MAAMC,cAAc,GAAG,OAAO3C,MAAM,CAACP,WAAW,KAAK,UAAU;IAE/D,IAAI,CAACkD,cAAc,IAAI,OAAO3C,MAAM,KAAK,UAAU,EAAE;MACnD,IAAI0C,KAAK,YAAY1C,MAAM,EAAE,OAAO,CAAC0C,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;MACxD;MACA,IAAIA,KAAK,KAAK/B,SAAS,EAAE,OAAO,CAAC+B,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;MACpD,OAAO,CAAC,IAAI1C,MAAM,CAAC0C,KAAK,CAAC,EAAE,IAAI,EAAE,KAAK,CAAC;IACzC;IAEA,IAAIA,KAAK,KAAK/B,SAAS,EAAE;MACvB,OAAO,CAAC+B,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC;IAC9B;IAEA,IAAI,CAACC,cAAc,IAAI,OAAO3C,MAAM,KAAK,QAAQ,EAAE;MACjD,MAAM4C,MAAM,GAAGC,KAAK,CAACC,OAAO,CAAC9C,MAAM,CAAC,GAChCN,gBAAgB,GAChBE,iBAAiB;MACrB,OAAOgD,MAAM,CAAC5C,MAAM,EAAE0C,KAAK,EAAEzC,OAAO,CAAC;IACvC;IAEA,IAAIT,QAAQ,CAACQ,MAAM,CAAC,EAAE;MACpB,OAAOF,aAAa,CAClB4C,KAAK,EACL1C,MAAM,EACNC,OAAO,EACPC,SAAS,EACTC,UAAU,EACVC,UAAU,EACVC,WAAW,EACXC,YAAY,EACZC,UAAU,CACX;IACH;IAEA,IAAIoC,cAAc,EAAE;MAClB,OAAO3C,MAAM,CAACP,WAAW,CAACiD,KAAK,EAAEzC,OAAO,CAAC;IAC3C;IAEA,OAAO,CAACyC,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;EAC7B;;EAEA;;EAEA,OAAO,CACLA,KAAU,EACV1C,MAAW,KAC+C;IAC1D,MAAM+C,GAAG,GAAG9C,OAAO,CAACyC,KAAK,EAAE1C,MAAM,CAAC;IAClC;IACA;IACA,IAAIe,MAAM,CAAC2B,KAAK,CAAC,KAAKA,KAAK,EAAE,OAAOK,GAAG;IAEvCzC,YAAY,CAACkB,IAAI,CAACkB,KAAK,CAAC;IACxB,IAAI,CAACF,WAAW,CAACL,GAAG,CAAC7B,YAAY,CAAC,EAAE;MAClCkC,WAAW,CAACJ,GAAG,CAAC9B,YAAY,EAAEyC,GAAG,CAAC,CAAC,CAAC,CAAC;MACrC,OAAOA,GAAG;IACZ,CAAC,MAAM;MACL,OAAO,CAACP,WAAW,CAACH,GAAG,CAAC/B,YAAY,CAAC,EAAEyC,GAAG,CAAC,CAAC,CAAC,EAAEA,GAAG,CAAC,CAAC,CAAC,CAAC;IACxD;EACF,CAAC;AACH,CAAC;AAED,MAAMN,WAAW,GAAIF,QAA6B,IAAK;EACrD,MAAMS,iBAAiB,GAAGrD,WAAW,CAAC4C,QAAQ,CAAC;EAE/C,OAAO,CACLxC,UAAwC,EACxCC,MAAuB,KACpB;IAAA;IACH,MAAMiD,SAAS,GAAGjD,MAAM,CAACc,GAAG;IAE5B,IAAI,OAAOf,UAAU,KAAK,QAAQ,EAAE;MAClC,OAAOA,UAAU;IACnB;IAEA,IAAIiD,iBAAiB,EAAE;MACrB,OAAOT,QAAQ,CAACW,KAAK,CAAC,CAACD,SAAS,EAAElD,UAAU,CAAC,CAAC;IAChD;IAEA,8BAAOwC,QAAQ,CAACU,SAAS,CAAC,qBAAnB,oBAAsBlD,UAAU,CAAC;EAC1C,CAAC;AACH,CAAC;AAsBD;AACA,OAAO,MAAMN,WAAW,GAAG,CACzBiD,KAAc,EACd1C,MAAqB,EACrBuC,QAAa,EACblC,WAAyC,GAAG,CAAC,CAAC,EAC9CmC,WAAqC,GAAG,IAAI3C,WAAW,EAAE,KAChC;EACzB;EACA,IAAIG,MAAM,KAAKW,SAAS,EAAE;IACxB,OAAO,CAAC+B,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;EACjC;EACA,IAAIA,KAAK,KAAK/B,SAAS,EAAE;IACvB,OAAO,CAACA,SAAS,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;EACtC;EACA,MAAMwC,gBAAqD,GAAG,CAAC,CAAC;EAChE,MAAMlD,OAAO,GAAGqC,UAAU,CACxBC,QAAQ,EACRlC,WAAW,EACXmC,WAAW,EACXW,gBAAgB,CACjB;EACD,OAAO,CAAC,GAAGlD,OAAO,CAACyC,KAAK,EAAE1C,MAAM,CAAC,EAAEmD,gBAAgB,CAAC;AAMtD,CAAC;AAED,OAAO,MAAMC,iBAAiB,GAAG,CAC/BV,KAAU,EACV1C,MAAqB,EACrBuC,QAAa,EACblC,WAAyC,GAAG,CAAC,CAAC,EAC9CmC,WAAqC,GAAG,IAAI3C,WAAW,EAAE,KAKzDJ,WAAW,CAACiD,KAAK,EAAE1C,MAAM,EAAEuC,QAAQ,EAAElC,WAAW,EAAEmC,WAAW,CAAC,CAACP,KAAK,CAClE,CAAC,EACD,CAAC,CACK;AAEV,SAASF,mBAAmB,CAC1B1B,WAAwE,EAExEgD,EAAO,EACP;EACA,IAAI,CAAChD,WAAW,CAACgD,EAAE,CAAC,EAAEhD,WAAW,CAACgD,EAAE,CAAC,GAAG,IAAIxD,WAAW,EAAE;EACzD,OAAOQ,WAAW,CAACgD,EAAE,CAAC;AACxB;;AAEA;AACA,SAASzB,mBAAmB,CAAC3B,OAAwB,EAAmB;EACtE;EACA;EACA,MAAMqD,eAAe,GAAGrD,OAAO,CAACsD,EAAE,IAAItD,OAAO;EAC7C,MAAMuD,cAAc,GAAG,CAACd,KAAU,EAAE1C,MAAW,KAC7CsD,eAAe,CAACZ,KAAK,EAAE1C,MAAM,CAAC;EAChCwD,cAAc,CAACD,EAAE,GAAGtD,OAAO;EAC3B,OAAOuD,cAAc;AACvB"} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["isEntity","denormalize","arrayDenormalize","isImmutable","objectDenormalize","WeakEntityMap","getEntities","depToPaths","unvisitEntity","entityOrId","schema","unvisit","getEntity","getCache","localCache","cycleCache","dependencies","cycleIndex","entity","pk","key","toString","includes","undefined","toJS","Object","create","localCacheKey","cycleCacheKey","found","deleted","globalCache","cacheValue","get","value","push","trackingIndex","length","path","entityCopy","createIfValid","toObject","withTrackedEntities","setLocal","localKey","slice","i","set","getUnvisit","entities","entityCache","resultCache","getEntityCaches","input","hasDenormalize","method","Array","isArray","cachable","ret","entityPaths","unshift","entityCacheKey","WeakMap","wem","denormalizeSimple","originalUnvisit","og","wrappedUnvisit"],"sources":["../src/denormalize.ts"],"sourcesContent":["import type { Schema, EntityInterface, UnvisitFunction } from './interface.js';\nimport { isEntity } from './isEntity.js';\nimport { denormalize as arrayDenormalize } from './schemas/Array.js';\nimport { isImmutable } from './schemas/ImmutableUtils.js';\nimport { denormalize as objectDenormalize } from './schemas/Object.js';\nimport type {\n  Denormalize,\n  DenormalizeNullable,\n  DenormalizeCache,\n  Path,\n} from './types.js';\nimport WeakEntityMap, {\n  type Dep,\n  getEntities,\n  type GetEntity,\n  depToPaths,\n} from './WeakEntityMap.js';\n\ninterface EntityCacheValue {\n  dependencies: Dep[];\n  value: [any, boolean, boolean];\n}\nconst unvisitEntity = (\n  entityOrId: Record<string, any> | string,\n  schema: EntityInterface,\n  unvisit: UnvisitFunction,\n  getEntity: GetEntity,\n  getCache: ReturnType<typeof getEntityCaches>,\n  localCache: Record<string, Record<string, any>>,\n  cycleCache: Record<string, Record<string, number>>,\n  dependencies: Dep[],\n  cycleIndex: { i: number },\n): [denormalized: object | undefined, found: boolean, deleted: boolean] => {\n  const entity =\n    typeof entityOrId === 'object'\n      ? entityOrId\n      : getEntity({ pk: entityOrId, key: schema.key });\n  if (\n    typeof entity === 'symbol' &&\n    (entity as symbol).toString().includes('DELETED')\n  ) {\n    return [undefined, true, true];\n    // TODO: Change to this as breaking change once we only support newer entities\n    // also remove `(entity as symbol).toString().includes('DELETED')` and do this for all symbols\n    // return schema.denormalize(entity, unvisit);\n  }\n\n  if (typeof entity !== 'object' || entity === null) {\n    return [entity as any, false, false];\n  }\n\n  const pk =\n    // normalize must always place a string, because pk() return value is string | undefined\n    // therefore no need to check for numbers\n    typeof entityOrId === 'string'\n      ? entityOrId\n      : schema.pk(isImmutable(entity) ? (entity as any).toJS() : entity);\n  // if we can't generate a working pk; this is hopeless so let's give them what's already there\n  // otherwise, even when we aren't doing a lookup we want to turn the entityOrId object into the\n  // expected class, and cache that class for referential equality. PK is used for global equality lookup.\n  if (pk === undefined || pk === '' || pk === 'undefined') {\n    return [entity, false, false];\n  }\n\n  const key = schema.key;\n  if (!(key in localCache)) {\n    localCache[key] = Object.create(null);\n  }\n  if (!(key in cycleCache)) {\n    cycleCache[key] = Object.create(null);\n  }\n  const localCacheKey = localCache[key];\n  const cycleCacheKey = cycleCache[key];\n\n  let found = true;\n  let deleted = false;\n\n  // local cache lookup first\n  if (!localCacheKey[pk]) {\n    const globalCache: WeakEntityMap<object, EntityCacheValue> = getCache(\n      pk,\n      schema,\n    );\n    const [cacheValue] = globalCache.get(entity, getEntity);\n    // TODO: what if this just returned the deps - then we don't need to store them\n\n    if (cacheValue) {\n      localCacheKey[pk] = cacheValue.value[0];\n      // TODO: can we store the cache values instead of tracking *all* their sources?\n      // this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it\n      dependencies.push(...cacheValue.dependencies);\n      return cacheValue.value;\n    }\n    // if we don't find in denormalize cache then do full denormalize\n    else {\n      const trackingIndex = dependencies.length;\n      dependencies.push({ entity, path: { key, pk } });\n\n      let entityCopy: any;\n      if (schema.createIfValid) {\n        entityCopy = localCacheKey[pk] = isImmutable(entity)\n          ? schema.createIfValid(entity.toObject())\n          : schema.createIfValid(entity);\n        // TODO(breaking): remove once old verions no longer supported\n      } else {\n        entityCopy = entity;\n        unvisit = withTrackedEntities(unvisit);\n        unvisit.setLocal = entityCopy => (localCacheKey[pk] = entityCopy);\n      }\n\n      cycleCacheKey[pk] = trackingIndex;\n      if (entityCopy === undefined) {\n        // undefined indicates we should suspense (perhaps failed validation)\n        found = false;\n        deleted = true;\n      } else {\n        [localCacheKey[pk], found, deleted] = schema.denormalize(\n          entityCopy,\n          unvisit,\n        );\n      }\n      delete cycleCacheKey[pk];\n\n      // if in cycle, use the start of the cycle to track all deps\n      // otherwise, we use our own trackingIndex\n      const localKey = dependencies.slice(\n        cycleIndex.i === -1 ? trackingIndex : cycleIndex.i,\n      );\n      const cacheValue: EntityCacheValue = {\n        dependencies: localKey,\n        value: [localCacheKey[pk], found, deleted],\n      };\n      globalCache.set(localKey, cacheValue);\n\n      // start of cycle - reset cycle detection\n      if (cycleIndex.i === trackingIndex) {\n        cycleIndex.i = -1;\n      }\n    }\n  } else {\n    // cycle detected\n    if (pk in cycleCacheKey) {\n      cycleIndex.i = cycleCacheKey[pk];\n    } else {\n      // with no cycle, globalCacheEntry will have already been set\n      dependencies.push({ entity, path: { key, pk } });\n    }\n  }\n\n  return [localCacheKey[pk], found, deleted];\n};\n\nconst getUnvisit = (\n  entities: Record<string, Record<string, any>>,\n  entityCache: DenormalizeCache['entities'],\n  resultCache: DenormalizeCache['results'][string],\n) => {\n  const getEntity = getEntities(entities);\n  const getCache = getEntityCaches(entityCache);\n  const localCache: Record<string, Record<string, any>> = {};\n  const dependencies: Dep[] = [];\n  const cycleIndex = { i: -1 };\n  const cycleCache = {};\n\n  function unvisit(\n    input: any,\n    schema: any,\n  ): [denormalized: any, found: boolean, deleted: boolean] {\n    if (!schema) return [input, true, false];\n\n    // null is considered intentional, thus always 'found' as true\n    if (input === null) {\n      return [input, true, false];\n    }\n\n    const hasDenormalize = typeof schema.denormalize === 'function';\n\n    // deserialize fields (like Date)\n    if (!hasDenormalize && typeof schema === 'function') {\n      if (input instanceof schema) return [input, true, false];\n      // field deserialization should never count against 'found' (whether to used inferred results)\n      if (input === undefined) return [input, true, false];\n      return [new schema(input), true, false];\n    }\n\n    if (input === undefined) {\n      return [input, false, false];\n    }\n\n    if (!hasDenormalize && typeof schema === 'object') {\n      const method = Array.isArray(schema)\n        ? arrayDenormalize\n        : objectDenormalize;\n      return method(schema, input, unvisit);\n    }\n\n    if (isEntity(schema)) {\n      return unvisitEntity(\n        input,\n        schema,\n        unvisit,\n        getEntity,\n        getCache,\n        localCache,\n        cycleCache,\n        dependencies,\n        cycleIndex,\n      );\n    }\n\n    if (hasDenormalize) {\n      return schema.denormalize(input, unvisit);\n    }\n\n    return [input, true, false];\n  }\n\n  return (\n    input: any,\n    schema: any,\n  ): [\n    denormalized: any,\n    found: boolean,\n    deleted: boolean,\n    entityPaths: Path[],\n  ] => {\n    // in the case where WeakMap cannot be used\n    // this test ensures null is properly excluded from WeakMap\n    const cachable = Object(input) === input && Object(schema) === schema;\n    if (!cachable) {\n      const ret = unvisit(input, schema);\n      // this is faster than spread\n      // https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples\n      return [ret[0], ret[1], ret[2], depToPaths(dependencies)];\n    }\n\n    let [ret, entityPaths] = resultCache.get(input, getEntity);\n\n    if (ret === undefined) {\n      ret = unvisit(input, schema);\n      // we want to do this before we add our 'input' entry\n      entityPaths = depToPaths(dependencies);\n      // for the first entry, `path` is ignored so empty members is fine\n      dependencies.unshift({ entity: input, path: { key: '', pk: '' } });\n      resultCache.set(dependencies, ret);\n    }\n\n    return [ret[0], ret[1], ret[2], entityPaths as Path[]];\n  };\n};\n\nconst getEntityCaches = (entityCache: DenormalizeCache['entities']) => {\n  return (pk: string, schema: EntityInterface) => {\n    const key = schema.key;\n\n    if (!(key in entityCache)) {\n      entityCache[key] = Object.create(null);\n    }\n    const entityCacheKey = entityCache[key];\n    if (!entityCacheKey[pk])\n      entityCacheKey[pk] = new WeakMap<\n        EntityInterface,\n        WeakEntityMap<object, any>\n      >();\n\n    let wem: WeakEntityMap<object, any> = entityCacheKey[pk].get(schema) as any;\n    if (!wem) {\n      wem = new WeakEntityMap<object, any>();\n      entityCacheKey[pk].set(schema, wem);\n    }\n\n    return wem;\n  };\n};\n\ntype DenormalizeReturn<S extends Schema> =\n  | [\n      denormalized: Denormalize<S>,\n      found: true,\n      deleted: false,\n      entityPaths: Path[],\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: boolean,\n      deleted: true,\n      entityPaths: Path[],\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: false,\n      deleted: boolean,\n      entityPaths: Path[],\n    ];\n\n// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types\nexport const denormalize = <S extends Schema>(\n  input: unknown,\n  schema: S | undefined,\n  entities: any,\n  entityCache: DenormalizeCache['entities'] = {},\n  resultCache: DenormalizeCache['results'][string] = new WeakEntityMap(),\n): DenormalizeReturn<S> => {\n  // undefined mean don't do anything\n  if (schema === undefined) {\n    return [input, true, false, []] as [any, boolean, boolean, any];\n  }\n  if (input === undefined) {\n    return [undefined, false, false, []] as [any, boolean, boolean, any];\n  }\n  return getUnvisit(entities, entityCache, resultCache)(input, schema);\n};\n\nexport const denormalizeSimple = <S extends Schema>(\n  input: any,\n  schema: S | undefined,\n  entities: any,\n  entityCache?: DenormalizeCache['entities'],\n  resultCache?: DenormalizeCache['results'][string],\n):\n  | [denormalized: Denormalize<S>, found: true, deleted: false]\n  | [denormalized: DenormalizeNullable<S>, found: boolean, deleted: true]\n  | [denormalized: DenormalizeNullable<S>, found: false, deleted: boolean] =>\n  denormalize(input, schema, entities, entityCache, resultCache).slice(\n    0,\n    3,\n  ) as any;\n\n// TODO(breaking): remove once unused\nfunction withTrackedEntities(unvisit: UnvisitFunction): UnvisitFunction {\n  // every time we nest, we want to unwrap back to the top.\n  // this is due to only needed the next level of nested entities for lookup\n  const originalUnvisit = unvisit.og || unvisit;\n  const wrappedUnvisit = (input: any, schema: any) =>\n    originalUnvisit(input, schema);\n  wrappedUnvisit.og = unvisit;\n  return wrappedUnvisit;\n}\n"],"mappings":"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"} |
@@ -7,3 +7,3 @@ Object.hasOwn = Object.hasOwn || /* istanbul ignore next */function hasOwn(it, key) { | ||
import { normalize } from './normalize.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
import WeakEntityMap from './WeakEntityMap.js'; | ||
export { default as inferResults } from './inferResults.js'; | ||
@@ -15,3 +15,3 @@ export { DELETED } from './special.js'; | ||
export * from './normal.js'; | ||
export { denormalize, normalize, isEntity, WeakListMap }; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJPYmplY3QiLCJoYXNPd24iLCJpdCIsImtleSIsInByb3RvdHlwZSIsImhhc093blByb3BlcnR5IiwiY2FsbCIsImRlbm9ybWFsaXplIiwiaXNFbnRpdHkiLCJub3JtYWxpemUiLCJXZWFrTGlzdE1hcCIsImRlZmF1bHQiLCJpbmZlclJlc3VsdHMiLCJERUxFVEVEIl0sInNvdXJjZXMiOlsiLi4vc3JjL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbIk9iamVjdC5oYXNPd24gPVxuICBPYmplY3QuaGFzT3duIHx8XG4gIC8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICovIGZ1bmN0aW9uIGhhc093bihpdCwga2V5KSB7XG4gICAgcmV0dXJuIE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbChpdCwga2V5KTtcbiAgfTtcbmltcG9ydCB7IGRlbm9ybWFsaXplIH0gZnJvbSAnLi9kZW5vcm1hbGl6ZS5qcyc7XG5pbXBvcnQgeyBpc0VudGl0eSB9IGZyb20gJy4vaXNFbnRpdHkuanMnO1xuaW1wb3J0IHsgbm9ybWFsaXplIH0gZnJvbSAnLi9ub3JtYWxpemUuanMnO1xuaW1wb3J0IFdlYWtMaXN0TWFwIGZyb20gJy4vV2Vha0xpc3RNYXAuanMnO1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBpbmZlclJlc3VsdHMgfSBmcm9tICcuL2luZmVyUmVzdWx0cy5qcyc7XG5leHBvcnQgeyBERUxFVEVEIH0gZnJvbSAnLi9zcGVjaWFsLmpzJztcblxuZXhwb3J0IHR5cGUge1xuICBBYnN0cmFjdEluc3RhbmNlVHlwZSxcbiAgTm9ybWFsaXplUmV0dXJuVHlwZSxcbiAgTm9ybWFsaXplZFNjaGVtYSxcbiAgRGVub3JtYWxpemVSZXR1cm5UeXBlLFxuICBEZW5vcm1hbGl6ZUNhY2hlLFxufSBmcm9tICcuL3R5cGVzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vZW5kcG9pbnQvdHlwZXMuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9FeHBpcnkuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9ub3JtYWwuanMnO1xuXG5leHBvcnQgeyBkZW5vcm1hbGl6ZSwgbm9ybWFsaXplLCBpc0VudGl0eSwgV2Vha0xpc3RNYXAgfTtcbiJdLCJtYXBwaW5ncyI6IkFBQUFBLE1BQU0sQ0FBQ0MsTUFBTSxHQUNYRCxNQUFNLENBQUNDLE1BQU0sSUFDYiwwQkFBMkIsU0FBU0EsTUFBTSxDQUFDQyxFQUFFLEVBQUVDLEdBQUcsRUFBRTtFQUNsRCxPQUFPSCxNQUFNLENBQUNJLFNBQVMsQ0FBQ0MsY0FBYyxDQUFDQyxJQUFJLENBQUNKLEVBQUUsRUFBRUMsR0FBRyxDQUFDO0FBQ3RELENBQUM7QUFDSCxTQUFTSSxXQUFXLFFBQVEsa0JBQWtCO0FBQzlDLFNBQVNDLFFBQVEsUUFBUSxlQUFlO0FBQ3hDLFNBQVNDLFNBQVMsUUFBUSxnQkFBZ0I7QUFDMUMsT0FBT0MsV0FBVyxNQUFNLGtCQUFrQjtBQUMxQyxTQUFTQyxPQUFPLElBQUlDLFlBQVksUUFBUSxtQkFBbUI7QUFDM0QsU0FBU0MsT0FBTyxRQUFRLGNBQWM7QUFTdEMsY0FBYyxxQkFBcUI7QUFDbkMsY0FBYyxnQkFBZ0I7QUFDOUIsY0FBYyxhQUFhO0FBQzNCLGNBQWMsYUFBYTtBQUUzQixTQUFTTixXQUFXLEVBQUVFLFNBQVMsRUFBRUQsUUFBUSxFQUFFRSxXQUFXIn0= | ||
export { denormalize, normalize, isEntity, WeakEntityMap }; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJPYmplY3QiLCJoYXNPd24iLCJpdCIsImtleSIsInByb3RvdHlwZSIsImhhc093blByb3BlcnR5IiwiY2FsbCIsImRlbm9ybWFsaXplIiwiaXNFbnRpdHkiLCJub3JtYWxpemUiLCJXZWFrRW50aXR5TWFwIiwiZGVmYXVsdCIsImluZmVyUmVzdWx0cyIsIkRFTEVURUQiXSwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiT2JqZWN0Lmhhc093biA9XG4gIE9iamVjdC5oYXNPd24gfHxcbiAgLyogaXN0YW5idWwgaWdub3JlIG5leHQgKi8gZnVuY3Rpb24gaGFzT3duKGl0LCBrZXkpIHtcbiAgICByZXR1cm4gT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eS5jYWxsKGl0LCBrZXkpO1xuICB9O1xuaW1wb3J0IHsgZGVub3JtYWxpemUgfSBmcm9tICcuL2Rlbm9ybWFsaXplLmpzJztcbmltcG9ydCB7IGlzRW50aXR5IH0gZnJvbSAnLi9pc0VudGl0eS5qcyc7XG5pbXBvcnQgeyBub3JtYWxpemUgfSBmcm9tICcuL25vcm1hbGl6ZS5qcyc7XG5pbXBvcnQgV2Vha0VudGl0eU1hcCBmcm9tICcuL1dlYWtFbnRpdHlNYXAuanMnO1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBpbmZlclJlc3VsdHMgfSBmcm9tICcuL2luZmVyUmVzdWx0cy5qcyc7XG5leHBvcnQgeyBERUxFVEVEIH0gZnJvbSAnLi9zcGVjaWFsLmpzJztcblxuZXhwb3J0IHR5cGUge1xuICBBYnN0cmFjdEluc3RhbmNlVHlwZSxcbiAgTm9ybWFsaXplUmV0dXJuVHlwZSxcbiAgTm9ybWFsaXplZFNjaGVtYSxcbiAgRGVub3JtYWxpemVSZXR1cm5UeXBlLFxuICBEZW5vcm1hbGl6ZUNhY2hlLFxuICBQYXRoLFxufSBmcm9tICcuL3R5cGVzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vZW5kcG9pbnQvdHlwZXMuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9FeHBpcnkuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9ub3JtYWwuanMnO1xuXG5leHBvcnQgeyBkZW5vcm1hbGl6ZSwgbm9ybWFsaXplLCBpc0VudGl0eSwgV2Vha0VudGl0eU1hcCB9O1xuIl0sIm1hcHBpbmdzIjoiQUFBQUEsTUFBTSxDQUFDQyxNQUFNLEdBQ1hELE1BQU0sQ0FBQ0MsTUFBTSxJQUNiLDBCQUEyQixTQUFTQSxNQUFNLENBQUNDLEVBQUUsRUFBRUMsR0FBRyxFQUFFO0VBQ2xELE9BQU9ILE1BQU0sQ0FBQ0ksU0FBUyxDQUFDQyxjQUFjLENBQUNDLElBQUksQ0FBQ0osRUFBRSxFQUFFQyxHQUFHLENBQUM7QUFDdEQsQ0FBQztBQUNILFNBQVNJLFdBQVcsUUFBUSxrQkFBa0I7QUFDOUMsU0FBU0MsUUFBUSxRQUFRLGVBQWU7QUFDeEMsU0FBU0MsU0FBUyxRQUFRLGdCQUFnQjtBQUMxQyxPQUFPQyxhQUFhLE1BQU0sb0JBQW9CO0FBQzlDLFNBQVNDLE9BQU8sSUFBSUMsWUFBWSxRQUFRLG1CQUFtQjtBQUMzRCxTQUFTQyxPQUFPLFFBQVEsY0FBYztBQVV0QyxjQUFjLHFCQUFxQjtBQUNuQyxjQUFjLGdCQUFnQjtBQUM5QixjQUFjLGFBQWE7QUFDM0IsY0FBYyxhQUFhO0FBRTNCLFNBQVNOLFdBQVcsRUFBRUUsU0FBUyxFQUFFRCxRQUFRLEVBQUVFLGFBQWEifQ== |
export {}; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL3R5cGVzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgU2NoZW1hLFxuICBTZXJpYWxpemFibGUsXG4gIEVudGl0eUludGVyZmFjZSxcbiAgTm9ybWFsaXplZEluZGV4LFxuICBTY2hlbWFDbGFzcyxcbn0gZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuaW1wb3J0IHR5cGUgV2Vha0xpc3RNYXAgZnJvbSAnLi9XZWFrTGlzdE1hcC5qcyc7XG5cbmV4cG9ydCB0eXBlIEFic3RyYWN0SW5zdGFuY2VUeXBlPFQ+ID0gVCBleHRlbmRzIHsgcHJvdG90eXBlOiBpbmZlciBVIH1cbiAgPyBVXG4gIDogbmV2ZXI7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZWRFbnRpdHk8VD4gPSBUIGV4dGVuZHMge1xuICBwcm90b3R5cGU6IGluZmVyIFU7XG4gIHNjaGVtYTogaW5mZXIgUztcbn1cbiAgPyB7IFtLIGluIEV4Y2x1ZGU8a2V5b2YgVSwga2V5b2YgUz5dOiBVW0tdIH0gJiB7IFtLIGluIGtleW9mIFNdOiBzdHJpbmcgfVxuICA6IG5ldmVyO1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gRGVub3JtYWxpemU8U1tLXT4gOiBTW0tdO1xufTtcblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVOdWxsYWJsZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gRGVub3JtYWxpemVOdWxsYWJsZTxTW0tdPiA6IFNbS107XG59O1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemVPYmplY3Q8UyBleHRlbmRzIFJlY29yZDxzdHJpbmcsIGFueT4+ID0ge1xuICBbSyBpbiBrZXlvZiBTXTogU1tLXSBleHRlbmRzIFNjaGVtYSA/IE5vcm1hbGl6ZTxTW0tdPiA6IFNbS107XG59O1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemVkTnVsbGFibGVPYmplY3Q8UyBleHRlbmRzIFJlY29yZDxzdHJpbmcsIGFueT4+ID0ge1xuICBbSyBpbiBrZXlvZiBTXTogU1tLXSBleHRlbmRzIFNjaGVtYSA/IE5vcm1hbGl6ZU51bGxhYmxlPFNbS10+IDogU1tLXTtcbn07XG5cbmludGVyZmFjZSBOZXN0ZWRTY2hlbWFDbGFzczxUID0gYW55PiB7XG4gIHNjaGVtYTogUmVjb3JkPHN0cmluZywgU2NoZW1hPjtcbiAgcHJvdG90eXBlOiBUO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFJlY29yZENsYXNzPFQgPSBhbnk+IGV4dGVuZHMgTmVzdGVkU2NoZW1hQ2xhc3M8VD4ge1xuICBmcm9tSlM6ICguLi5hcmdzOiBhbnkpID0+IEFic3RyYWN0SW5zdGFuY2VUeXBlPFQ+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERlbm9ybWFsaXplQ2FjaGUge1xuICBlbnRpdGllczoge1xuICAgIFtrZXk6IHN0cmluZ106IHtcbiAgICAgIFtwazogc3RyaW5nXTogV2Vha0xpc3RNYXA8b2JqZWN0LCBFbnRpdHlJbnRlcmZhY2U+O1xuICAgIH07XG4gIH07XG4gIHJlc3VsdHM6IHtcbiAgICBba2V5OiBzdHJpbmddOiBXZWFrTGlzdE1hcDxvYmplY3QsIGFueT47XG4gIH07XG59XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplTnVsbGFibGVOZXN0ZWRTY2hlbWE8UyBleHRlbmRzIE5lc3RlZFNjaGVtYUNsYXNzPiA9XG4gIGtleW9mIFNbJ3NjaGVtYSddIGV4dGVuZHMgbmV2ZXJcbiAgICA/IFNbJ3Byb3RvdHlwZSddIC8vIHRoaXMgaXMgdGhlIGNhc2Ugb2YgYSBub24tc2V0IHNjaGVtYSwgd2hpY2ggbWVhbnMgaXQgYWN0dWFsbHkgaGFzIG5vIG1lbWJlcnNcbiAgICA6IHN0cmluZyBleHRlbmRzIGtleW9mIFNbJ3NjaGVtYSddXG4gICAgPyBTWydwcm90b3R5cGUnXVxuICAgIDogU1sncHJvdG90eXBlJ10gLyomIHtcbiAgICAgICAgW0sgaW4ga2V5b2YgU1snc2NoZW1hJ11dOiBEZW5vcm1hbGl6ZU51bGxhYmxlPFNbJ3NjaGVtYSddW0tdPjtcbiAgICAgIH0qLztcblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVSZXR1cm5UeXBlPFQ+ID0gVCBleHRlbmRzIChcbiAgaW5wdXQ6IGFueSxcbiAgdW52aXNpdDogYW55LFxuKSA9PiBbaW5mZXIgUiwgYW55LCBhbnldXG4gID8gUlxuICA6IG5ldmVyO1xuZXhwb3J0IHR5cGUgTm9ybWFsaXplUmV0dXJuVHlwZTxUPiA9IFQgZXh0ZW5kcyAoLi4uYXJnczogYW55KSA9PiBpbmZlciBSXG4gID8gUlxuICA6IG5ldmVyO1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZTxTPiA9IFMgZXh0ZW5kcyBFbnRpdHlJbnRlcmZhY2U8aW5mZXIgVT5cbiAgPyBVXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gQWJzdHJhY3RJbnN0YW5jZVR5cGU8Uz5cbiAgOiBTIGV4dGVuZHMgU2NoZW1hQ2xhc3NcbiAgPyBEZW5vcm1hbGl6ZVJldHVyblR5cGU8U1snZGVub3JtYWxpemUnXT5cbiAgOiBTIGV4dGVuZHMgU2VyaWFsaXphYmxlPGluZmVyIFQ+XG4gID8gVFxuICA6IFMgZXh0ZW5kcyBBcnJheTxpbmZlciBGPlxuICA/IERlbm9ybWFsaXplPEY+W11cbiAgOiBTIGV4dGVuZHMgeyBbSzogc3RyaW5nXTogYW55IH1cbiAgPyBEZW5vcm1hbGl6ZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplTnVsbGFibGU8Uz4gPSBTIGV4dGVuZHMgRW50aXR5SW50ZXJmYWNlPGFueT5cbiAgPyBEZW5vcm1hbGl6ZU51bGxhYmxlTmVzdGVkU2NoZW1hPFM+IHwgdW5kZWZpbmVkXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gRGVub3JtYWxpemVOdWxsYWJsZU5lc3RlZFNjaGVtYTxTPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IERlbm9ybWFsaXplUmV0dXJuVHlwZTxTWydfZGVub3JtYWxpemVOdWxsYWJsZSddPlxuICA6IFMgZXh0ZW5kcyBTZXJpYWxpemFibGU8aW5mZXIgVD5cbiAgPyBUXG4gIDogUyBleHRlbmRzIEFycmF5PGluZmVyIEY+XG4gID8gRGVub3JtYWxpemU8Rj5bXSB8IHVuZGVmaW5lZFxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IERlbm9ybWFsaXplTnVsbGFibGVPYmplY3Q8Uz5cbiAgOiBTO1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemU8Uz4gPSBTIGV4dGVuZHMgRW50aXR5SW50ZXJmYWNlXG4gID8gc3RyaW5nXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gTm9ybWFsaXplT2JqZWN0PFNbJ3NjaGVtYSddPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IE5vcm1hbGl6ZVJldHVyblR5cGU8U1snbm9ybWFsaXplJ10+XG4gIDogUyBleHRlbmRzIFNlcmlhbGl6YWJsZTxpbmZlciBUPlxuICA/IFRcbiAgOiBTIGV4dGVuZHMgQXJyYXk8aW5mZXIgRj5cbiAgPyBOb3JtYWxpemU8Rj5bXVxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IE5vcm1hbGl6ZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZU51bGxhYmxlPFM+ID0gUyBleHRlbmRzIEVudGl0eUludGVyZmFjZVxuICA/IHN0cmluZyB8IHVuZGVmaW5lZFxuICA6IFMgZXh0ZW5kcyBSZWNvcmRDbGFzc1xuICA/IE5vcm1hbGl6ZWROdWxsYWJsZU9iamVjdDxTWydzY2hlbWEnXT5cbiAgOiBTIGV4dGVuZHMgU2NoZW1hQ2xhc3NcbiAgPyBOb3JtYWxpemVSZXR1cm5UeXBlPFNbJ19ub3JtYWxpemVOdWxsYWJsZSddPlxuICA6IFMgZXh0ZW5kcyBTZXJpYWxpemFibGU8aW5mZXIgVD5cbiAgPyBUXG4gIDogUyBleHRlbmRzIEFycmF5PGluZmVyIEY+XG4gID8gTm9ybWFsaXplPEY+W10gfCB1bmRlZmluZWRcbiAgOiBTIGV4dGVuZHMgeyBbSzogc3RyaW5nXTogYW55IH1cbiAgPyBOb3JtYWxpemVkTnVsbGFibGVPYmplY3Q8Uz5cbiAgOiBTO1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemVkU2NoZW1hPEUsIFI+ID0ge1xuICBlbnRpdGllczogRTtcbiAgcmVzdWx0OiBSO1xuICBpbmRleGVzOiBOb3JtYWxpemVkSW5kZXg7XG4gIGVudGl0eU1ldGE6IHtcbiAgICByZWFkb25seSBbZW50aXR5S2V5OiBzdHJpbmddOiB7XG4gICAgICByZWFkb25seSBbcGs6IHN0cmluZ106IHtcbiAgICAgICAgcmVhZG9ubHkgZGF0ZTogbnVtYmVyO1xuICAgICAgICByZWFkb25seSBleHBpcmVzQXQ6IG51bWJlcjtcbiAgICAgICAgcmVhZG9ubHkgZmV0Y2hlZEF0OiBudW1iZXI7XG4gICAgICB9O1xuICAgIH07XG4gIH07XG59O1xuXG5leHBvcnQgdHlwZSBFbnRpdHlNYXA8VCA9IGFueT4gPSBSZWNvcmQ8c3RyaW5nLCBFbnRpdHlJbnRlcmZhY2U8VD4+O1xuIl0sIm1hcHBpbmdzIjoiIn0= | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL3R5cGVzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgU2NoZW1hLFxuICBTZXJpYWxpemFibGUsXG4gIEVudGl0eUludGVyZmFjZSxcbiAgTm9ybWFsaXplZEluZGV4LFxuICBTY2hlbWFDbGFzcyxcbn0gZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuaW1wb3J0IHR5cGUgV2Vha0VudGl0eU1hcCBmcm9tICcuL1dlYWtFbnRpdHlNYXAuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBhdGgge1xuICBrZXk6IHN0cmluZztcbiAgcGs6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQWJzdHJhY3RJbnN0YW5jZVR5cGU8VD4gPSBUIGV4dGVuZHMgeyBwcm90b3R5cGU6IGluZmVyIFUgfVxuICA/IFVcbiAgOiBuZXZlcjtcblxuZXhwb3J0IHR5cGUgTm9ybWFsaXplZEVudGl0eTxUPiA9IFQgZXh0ZW5kcyB7XG4gIHByb3RvdHlwZTogaW5mZXIgVTtcbiAgc2NoZW1hOiBpbmZlciBTO1xufVxuICA/IHsgW0sgaW4gRXhjbHVkZTxrZXlvZiBVLCBrZXlvZiBTPl06IFVbS10gfSAmIHsgW0sgaW4ga2V5b2YgU106IHN0cmluZyB9XG4gIDogbmV2ZXI7XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplT2JqZWN0PFMgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBhbnk+PiA9IHtcbiAgW0sgaW4ga2V5b2YgU106IFNbS10gZXh0ZW5kcyBTY2hlbWEgPyBEZW5vcm1hbGl6ZTxTW0tdPiA6IFNbS107XG59O1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZU51bGxhYmxlT2JqZWN0PFMgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBhbnk+PiA9IHtcbiAgW0sgaW4ga2V5b2YgU106IFNbS10gZXh0ZW5kcyBTY2hlbWEgPyBEZW5vcm1hbGl6ZU51bGxhYmxlPFNbS10+IDogU1tLXTtcbn07XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gTm9ybWFsaXplPFNbS10+IDogU1tLXTtcbn07XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZWROdWxsYWJsZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gTm9ybWFsaXplTnVsbGFibGU8U1tLXT4gOiBTW0tdO1xufTtcblxuaW50ZXJmYWNlIE5lc3RlZFNjaGVtYUNsYXNzPFQgPSBhbnk+IHtcbiAgc2NoZW1hOiBSZWNvcmQ8c3RyaW5nLCBTY2hlbWE+O1xuICBwcm90b3R5cGU6IFQ7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVjb3JkQ2xhc3M8VCA9IGFueT4gZXh0ZW5kcyBOZXN0ZWRTY2hlbWFDbGFzczxUPiB7XG4gIGZyb21KUzogKC4uLmFyZ3M6IGFueSkgPT4gQWJzdHJhY3RJbnN0YW5jZVR5cGU8VD47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGVub3JtYWxpemVDYWNoZSB7XG4gIGVudGl0aWVzOiB7XG4gICAgW2tleTogc3RyaW5nXToge1xuICAgICAgW3BrOiBzdHJpbmddOiBXZWFrTWFwPEVudGl0eUludGVyZmFjZSwgV2Vha0VudGl0eU1hcDxvYmplY3QsIGFueT4+O1xuICAgIH07XG4gIH07XG4gIHJlc3VsdHM6IHtcbiAgICBba2V5OiBzdHJpbmddOiBXZWFrRW50aXR5TWFwPG9iamVjdCwgYW55PjtcbiAgfTtcbn1cblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVOdWxsYWJsZU5lc3RlZFNjaGVtYTxTIGV4dGVuZHMgTmVzdGVkU2NoZW1hQ2xhc3M+ID1cbiAga2V5b2YgU1snc2NoZW1hJ10gZXh0ZW5kcyBuZXZlclxuICAgID8gU1sncHJvdG90eXBlJ10gLy8gdGhpcyBpcyB0aGUgY2FzZSBvZiBhIG5vbi1zZXQgc2NoZW1hLCB3aGljaCBtZWFucyBpdCBhY3R1YWxseSBoYXMgbm8gbWVtYmVyc1xuICAgIDogc3RyaW5nIGV4dGVuZHMga2V5b2YgU1snc2NoZW1hJ11cbiAgICA/IFNbJ3Byb3RvdHlwZSddXG4gICAgOiBTWydwcm90b3R5cGUnXSAvKiYge1xuICAgICAgICBbSyBpbiBrZXlvZiBTWydzY2hlbWEnXV06IERlbm9ybWFsaXplTnVsbGFibGU8U1snc2NoZW1hJ11bS10+O1xuICAgICAgfSovO1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZVJldHVyblR5cGU8VD4gPSBUIGV4dGVuZHMgKFxuICBpbnB1dDogYW55LFxuICB1bnZpc2l0OiBhbnksXG4pID0+IFtpbmZlciBSLCBhbnksIGFueV1cbiAgPyBSXG4gIDogbmV2ZXI7XG5leHBvcnQgdHlwZSBOb3JtYWxpemVSZXR1cm5UeXBlPFQ+ID0gVCBleHRlbmRzICguLi5hcmdzOiBhbnkpID0+IGluZmVyIFJcbiAgPyBSXG4gIDogbmV2ZXI7XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplPFM+ID0gUyBleHRlbmRzIEVudGl0eUludGVyZmFjZTxpbmZlciBVPlxuICA/IFVcbiAgOiBTIGV4dGVuZHMgUmVjb3JkQ2xhc3NcbiAgPyBBYnN0cmFjdEluc3RhbmNlVHlwZTxTPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IERlbm9ybWFsaXplUmV0dXJuVHlwZTxTWydkZW5vcm1hbGl6ZSddPlxuICA6IFMgZXh0ZW5kcyBTZXJpYWxpemFibGU8aW5mZXIgVD5cbiAgPyBUXG4gIDogUyBleHRlbmRzIEFycmF5PGluZmVyIEY+XG4gID8gRGVub3JtYWxpemU8Rj5bXVxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IERlbm9ybWFsaXplT2JqZWN0PFM+XG4gIDogUztcblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVOdWxsYWJsZTxTPiA9IFMgZXh0ZW5kcyBFbnRpdHlJbnRlcmZhY2U8YW55PlxuICA/IERlbm9ybWFsaXplTnVsbGFibGVOZXN0ZWRTY2hlbWE8Uz4gfCB1bmRlZmluZWRcbiAgOiBTIGV4dGVuZHMgUmVjb3JkQ2xhc3NcbiAgPyBEZW5vcm1hbGl6ZU51bGxhYmxlTmVzdGVkU2NoZW1hPFM+XG4gIDogUyBleHRlbmRzIFNjaGVtYUNsYXNzXG4gID8gRGVub3JtYWxpemVSZXR1cm5UeXBlPFNbJ19kZW5vcm1hbGl6ZU51bGxhYmxlJ10+XG4gIDogUyBleHRlbmRzIFNlcmlhbGl6YWJsZTxpbmZlciBUPlxuICA/IFRcbiAgOiBTIGV4dGVuZHMgQXJyYXk8aW5mZXIgRj5cbiAgPyBEZW5vcm1hbGl6ZTxGPltdIHwgdW5kZWZpbmVkXG4gIDogUyBleHRlbmRzIHsgW0s6IHN0cmluZ106IGFueSB9XG4gID8gRGVub3JtYWxpemVOdWxsYWJsZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZTxTPiA9IFMgZXh0ZW5kcyBFbnRpdHlJbnRlcmZhY2VcbiAgPyBzdHJpbmdcbiAgOiBTIGV4dGVuZHMgUmVjb3JkQ2xhc3NcbiAgPyBOb3JtYWxpemVPYmplY3Q8U1snc2NoZW1hJ10+XG4gIDogUyBleHRlbmRzIFNjaGVtYUNsYXNzXG4gID8gTm9ybWFsaXplUmV0dXJuVHlwZTxTWydub3JtYWxpemUnXT5cbiAgOiBTIGV4dGVuZHMgU2VyaWFsaXphYmxlPGluZmVyIFQ+XG4gID8gVFxuICA6IFMgZXh0ZW5kcyBBcnJheTxpbmZlciBGPlxuICA/IE5vcm1hbGl6ZTxGPltdXG4gIDogUyBleHRlbmRzIHsgW0s6IHN0cmluZ106IGFueSB9XG4gID8gTm9ybWFsaXplT2JqZWN0PFM+XG4gIDogUztcblxuZXhwb3J0IHR5cGUgTm9ybWFsaXplTnVsbGFibGU8Uz4gPSBTIGV4dGVuZHMgRW50aXR5SW50ZXJmYWNlXG4gID8gc3RyaW5nIHwgdW5kZWZpbmVkXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gTm9ybWFsaXplZE51bGxhYmxlT2JqZWN0PFNbJ3NjaGVtYSddPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IE5vcm1hbGl6ZVJldHVyblR5cGU8U1snX25vcm1hbGl6ZU51bGxhYmxlJ10+XG4gIDogUyBleHRlbmRzIFNlcmlhbGl6YWJsZTxpbmZlciBUPlxuICA/IFRcbiAgOiBTIGV4dGVuZHMgQXJyYXk8aW5mZXIgRj5cbiAgPyBOb3JtYWxpemU8Rj5bXSB8IHVuZGVmaW5lZFxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IE5vcm1hbGl6ZWROdWxsYWJsZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZWRTY2hlbWE8RSwgUj4gPSB7XG4gIGVudGl0aWVzOiBFO1xuICByZXN1bHQ6IFI7XG4gIGluZGV4ZXM6IE5vcm1hbGl6ZWRJbmRleDtcbiAgZW50aXR5TWV0YToge1xuICAgIHJlYWRvbmx5IFtlbnRpdHlLZXk6IHN0cmluZ106IHtcbiAgICAgIHJlYWRvbmx5IFtwazogc3RyaW5nXToge1xuICAgICAgICByZWFkb25seSBkYXRlOiBudW1iZXI7XG4gICAgICAgIHJlYWRvbmx5IGV4cGlyZXNBdDogbnVtYmVyO1xuICAgICAgICByZWFkb25seSBmZXRjaGVkQXQ6IG51bWJlcjtcbiAgICAgIH07XG4gICAgfTtcbiAgfTtcbn07XG5cbmV4cG9ydCB0eXBlIEVudGl0eU1hcDxUID0gYW55PiA9IFJlY29yZDxzdHJpbmcsIEVudGl0eUludGVyZmFjZTxUPj47XG4iXSwibWFwcGluZ3MiOiIifQ== |
import type { Schema } from './interface.js'; | ||
import type { Denormalize, DenormalizeNullable, DenormalizeCache } from './types.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
import type { Denormalize, DenormalizeNullable, DenormalizeCache, Path } from './types.js'; | ||
type DenormalizeReturn<S extends Schema> = [ | ||
@@ -8,3 +7,3 @@ denormalized: Denormalize<S>, | ||
deleted: false, | ||
resolvedEntities: Record<string, Record<string, any>> | ||
entityPaths: Path[] | ||
] | [ | ||
@@ -14,3 +13,3 @@ denormalized: DenormalizeNullable<S>, | ||
deleted: true, | ||
resolvedEntities: Record<string, Record<string, any>> | ||
entityPaths: Path[] | ||
] | [ | ||
@@ -20,7 +19,7 @@ denormalized: DenormalizeNullable<S>, | ||
deleted: boolean, | ||
resolvedEntities: Record<string, Record<string, any>> | ||
entityPaths: Path[] | ||
]; | ||
export declare const denormalize: <S extends Schema>(input: unknown, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: WeakListMap<object, any>) => DenormalizeReturn<S>; | ||
export declare const denormalizeSimple: <S extends Schema>(input: any, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: WeakListMap<object, any>) => [denormalized: Denormalize<S>, found: true, deleted: false] | [denormalized: DenormalizeNullable<S>, found: boolean, deleted: true] | [denormalized: DenormalizeNullable<S>, found: false, deleted: boolean]; | ||
export declare const denormalize: <S extends Schema>(input: unknown, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: DenormalizeCache['results'][string]) => DenormalizeReturn<S>; | ||
export declare const denormalizeSimple: <S extends Schema>(input: any, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: DenormalizeCache['results'][string]) => [denormalized: Denormalize<S>, found: true, deleted: false] | [denormalized: DenormalizeNullable<S>, found: boolean, deleted: true] | [denormalized: DenormalizeNullable<S>, found: false, deleted: boolean]; | ||
export {}; | ||
//# sourceMappingURL=denormalize.d.ts.map |
@@ -5,5 +5,8 @@ import { isEntity } from './isEntity.js'; | ||
import { denormalize as objectDenormalize } from './schemas/Object.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex) => { | ||
const entity = getEntity(entityOrId, schema); | ||
import WeakEntityMap, { getEntities, depToPaths } from './WeakEntityMap.js'; | ||
const unvisitEntity = (entityOrId, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex) => { | ||
const entity = typeof entityOrId === 'object' ? entityOrId : getEntity({ | ||
pk: entityOrId, | ||
key: schema.key | ||
}); | ||
if (typeof entity === 'symbol' && entity.toString().includes('DELETED')) { | ||
@@ -36,45 +39,62 @@ return [undefined, true, true]; | ||
} | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
const localCacheKey = localCache[key]; | ||
const cycleCacheKey = cycleCache[key]; | ||
const entityCacheKey = entityCache[key]; | ||
let found = true; | ||
let deleted = false; | ||
// local cache lookup first | ||
if (!localCacheKey[pk]) { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push(entity); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk); | ||
const globalCache = getCache(pk, schema); | ||
const [cacheValue] = globalCache.get(entity, getEntity); | ||
// TODO: what if this just returned the deps - then we don't need to store them | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
if (!globalCacheEntry.has(localKey)) { | ||
globalCacheEntry.set(localKey, localCacheKey[pk]); | ||
} else { | ||
localCacheKey[pk] = globalCacheEntry.get(localKey); | ||
if (cacheValue) { | ||
localCacheKey[pk] = cacheValue.value[0]; | ||
// TODO: can we store the cache values instead of tracking *all* their sources? | ||
// this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it | ||
dependencies.push(...cacheValue.dependencies); | ||
return cacheValue.value; | ||
} | ||
// if we don't find in denormalize cache then do full denormalize | ||
else { | ||
const trackingIndex = dependencies.length; | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
let entityCopy; | ||
if (schema.createIfValid) { | ||
entityCopy = localCacheKey[pk] = isImmutable(entity) ? schema.createIfValid(entity.toObject()) : schema.createIfValid(entity); | ||
// TODO(breaking): remove once old verions no longer supported | ||
} else { | ||
entityCopy = entity; | ||
unvisit = withTrackedEntities(unvisit); | ||
unvisit.setLocal = entityCopy => localCacheKey[pk] = entityCopy; | ||
} | ||
cycleCacheKey[pk] = trackingIndex; | ||
if (entityCopy === undefined) { | ||
// undefined indicates we should suspense (perhaps failed validation) | ||
found = false; | ||
deleted = true; | ||
} else { | ||
[localCacheKey[pk], found, deleted] = schema.denormalize(entityCopy, unvisit); | ||
} | ||
delete cycleCacheKey[pk]; | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
// if in cycle, use the start of the cycle to track all deps | ||
// otherwise, we use our own trackingIndex | ||
const localKey = dependencies.slice(cycleIndex.i === -1 ? trackingIndex : cycleIndex.i); | ||
const cacheValue = { | ||
dependencies: localKey, | ||
value: [localCacheKey[pk], found, deleted] | ||
}; | ||
globalCache.set(localKey, cacheValue); | ||
// start of cycle - reset cycle detection | ||
if (cycleIndex.i === trackingIndex) { | ||
cycleIndex.i = -1; | ||
} | ||
} | ||
@@ -87,3 +107,9 @@ } else { | ||
// with no cycle, globalCacheEntry will have already been set | ||
dependencies.push(entity); | ||
dependencies.push({ | ||
entity, | ||
path: { | ||
key, | ||
pk | ||
} | ||
}); | ||
} | ||
@@ -93,4 +119,6 @@ } | ||
}; | ||
const getUnvisit = (entities, entityCache, resultCache, localCache) => { | ||
const getUnvisit = (entities, entityCache, resultCache) => { | ||
const getEntity = getEntities(entities); | ||
const getCache = getEntityCaches(entityCache); | ||
const localCache = {}; | ||
const dependencies = []; | ||
@@ -109,2 +137,4 @@ const cycleIndex = { | ||
const hasDenormalize = typeof schema.denormalize === 'function'; | ||
// deserialize fields (like Date) | ||
if (!hasDenormalize && typeof schema === 'function') { | ||
@@ -124,3 +154,3 @@ if (input instanceof schema) return [input, true, false]; | ||
if (isEntity(schema)) { | ||
return unvisitEntity(input, schema, unvisit, getEntity, localCache, cycleCache, entityCache, dependencies, cycleIndex); | ||
return unvisitEntity(input, schema, unvisit, getEntity, getCache, localCache, cycleCache, dependencies, cycleIndex); | ||
} | ||
@@ -132,51 +162,58 @@ if (hasDenormalize) { | ||
} | ||
//const wrappedUnvisit = withTrackedEntities(unvisit, globalKey); | ||
return (input, schema) => { | ||
const ret = unvisit(input, schema); | ||
// in the case where WeakMap cannot be used | ||
// this test ensures null is properly excluded from WeakMap | ||
if (Object(input) !== input) return ret; | ||
dependencies.push(input); | ||
if (!resultCache.has(dependencies)) { | ||
resultCache.set(dependencies, ret[0]); | ||
return ret; | ||
} else { | ||
return [resultCache.get(dependencies), ret[1], ret[2]]; | ||
const cachable = Object(input) === input && Object(schema) === schema; | ||
if (!cachable) { | ||
const ret = unvisit(input, schema); | ||
// this is faster than spread | ||
// https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples | ||
return [ret[0], ret[1], ret[2], depToPaths(dependencies)]; | ||
} | ||
let [ret, entityPaths] = resultCache.get(input, getEntity); | ||
if (ret === undefined) { | ||
ret = unvisit(input, schema); | ||
// we want to do this before we add our 'input' entry | ||
entityPaths = depToPaths(dependencies); | ||
// for the first entry, `path` is ignored so empty members is fine | ||
dependencies.unshift({ | ||
entity: input, | ||
path: { | ||
key: '', | ||
pk: '' | ||
} | ||
}); | ||
resultCache.set(dependencies, ret); | ||
} | ||
return [ret[0], ret[1], ret[2], entityPaths]; | ||
}; | ||
}; | ||
const getEntities = entities => { | ||
const entityIsImmutable = isImmutable(entities); | ||
return (entityOrId, schema) => { | ||
var _entities$schemaKey; | ||
const schemaKey = schema.key; | ||
if (typeof entityOrId === 'object') { | ||
return entityOrId; | ||
const getEntityCaches = entityCache => { | ||
return (pk, schema) => { | ||
const key = schema.key; | ||
if (!(key in entityCache)) { | ||
entityCache[key] = Object.create(null); | ||
} | ||
if (entityIsImmutable) { | ||
return entities.getIn([schemaKey, entityOrId]); | ||
const entityCacheKey = entityCache[key]; | ||
if (!entityCacheKey[pk]) entityCacheKey[pk] = new WeakMap(); | ||
let wem = entityCacheKey[pk].get(schema); | ||
if (!wem) { | ||
wem = new WeakEntityMap(); | ||
entityCacheKey[pk].set(schema, wem); | ||
} | ||
return (_entities$schemaKey = entities[schemaKey]) == null ? void 0 : _entities$schemaKey[entityOrId]; | ||
return wem; | ||
}; | ||
}; | ||
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types | ||
export const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => { | ||
export const denormalize = (input, schema, entities, entityCache = {}, resultCache = new WeakEntityMap()) => { | ||
// undefined mean don't do anything | ||
if (schema === undefined) { | ||
return [input, true, false, {}]; | ||
return [input, true, false, []]; | ||
} | ||
if (input === undefined) { | ||
return [undefined, false, false, {}]; | ||
return [undefined, false, false, []]; | ||
} | ||
const resolvedEntities = {}; | ||
const unvisit = getUnvisit(entities, entityCache, resultCache, resolvedEntities); | ||
return [...unvisit(input, schema), resolvedEntities]; | ||
return getUnvisit(entities, entityCache, resultCache)(input, schema); | ||
}; | ||
export const denormalizeSimple = (input, schema, entities, entityCache = {}, resultCache = new WeakListMap()) => denormalize(input, schema, entities, entityCache, resultCache).slice(0, 3); | ||
function getGlobalCacheEntry(entityCache, id) { | ||
if (!entityCache[id]) entityCache[id] = new WeakListMap(); | ||
return entityCache[id]; | ||
} | ||
export const denormalizeSimple = (input, schema, entities, entityCache, resultCache) => denormalize(input, schema, entities, entityCache, resultCache).slice(0, 3); | ||
@@ -192,2 +229,2 @@ // TODO(breaking): remove once unused | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["isEntity","denormalize","arrayDenormalize","isImmutable","objectDenormalize","WeakListMap","unvisitEntity","entityOrId","schema","unvisit","getEntity","localCache","cycleCache","entityCache","dependencies","cycleIndex","entity","toString","includes","undefined","pk","toJS","key","Object","create","localCacheKey","cycleCacheKey","entityCacheKey","found","deleted","trackingIndex","length","push","entityCopy","createIfValid","toObject","withTrackedEntities","setLocal","globalCacheEntry","getGlobalCacheEntry","localKey","slice","i","has","set","get","getUnvisit","entities","resultCache","getEntities","input","hasDenormalize","method","Array","isArray","ret","entityIsImmutable","schemaKey","getIn","resolvedEntities","denormalizeSimple","id","originalUnvisit","og","wrappedUnvisit"],"sources":["../src/denormalize.ts"],"sourcesContent":["import type { Schema, EntityInterface, UnvisitFunction } from './interface.js';\nimport { isEntity } from './isEntity.js';\nimport { denormalize as arrayDenormalize } from './schemas/Array.js';\nimport { isImmutable } from './schemas/ImmutableUtils.js';\nimport { denormalize as objectDenormalize } from './schemas/Object.js';\nimport type {\n  Denormalize,\n  DenormalizeNullable,\n  DenormalizeCache,\n} from './types.js';\nimport WeakListMap from './WeakListMap.js';\n\nconst unvisitEntity = (\n  entityOrId: Record<string, any> | string,\n  schema: EntityInterface,\n  unvisit: UnvisitFunction,\n  getEntity: (\n    entityOrId: Record<string, any> | string,\n    schema: EntityInterface,\n  ) => object | symbol,\n  localCache: Record<string, Record<string, any>>,\n  cycleCache: Record<string, Record<string, number>>,\n  entityCache: DenormalizeCache['entities'],\n  dependencies: object[],\n  cycleIndex: { i: number },\n): [denormalized: object | undefined, found: boolean, deleted: boolean] => {\n  const entity = getEntity(entityOrId, schema);\n  if (\n    typeof entity === 'symbol' &&\n    (entity as symbol).toString().includes('DELETED')\n  ) {\n    return [undefined, true, true];\n    // TODO: Change to this as breaking change once we only support newer entities\n    // also remove `(entity as symbol).toString().includes('DELETED')` and do this for all symbols\n    // return schema.denormalize(entity, unvisit);\n  }\n\n  if (typeof entity !== 'object' || entity === null) {\n    return [entity as any, false, false];\n  }\n\n  const pk =\n    // normalize must always place a string, because pk() return value is string | undefined\n    // therefore no need to check for numbers\n    typeof entityOrId === 'string'\n      ? entityOrId\n      : schema.pk(isImmutable(entity) ? (entity as any).toJS() : entity);\n  // if we can't generate a working pk; this is hopeless so let's give them what's already there\n  // otherwise, even when we aren't doing a lookup we want to turn the entityOrId object into the\n  // expected class, and cache that class for referential equality. PK is used for global equality lookup.\n  if (pk === undefined || pk === '' || pk === 'undefined') {\n    return [entity, false, false];\n  }\n\n  const key = schema.key;\n  if (!(key in localCache)) {\n    localCache[key] = Object.create(null);\n  }\n  if (!(key in cycleCache)) {\n    cycleCache[key] = Object.create(null);\n  }\n  if (!(key in entityCache)) {\n    entityCache[key] = Object.create(null);\n  }\n  const localCacheKey = localCache[key];\n  const cycleCacheKey = cycleCache[key];\n  const entityCacheKey = entityCache[key];\n\n  let found = true;\n  let deleted = false;\n\n  if (!localCacheKey[pk]) {\n    const trackingIndex = dependencies.length;\n    dependencies.push(entity);\n\n    let entityCopy: any;\n    if (schema.createIfValid) {\n      entityCopy = localCacheKey[pk] = isImmutable(entity)\n        ? schema.createIfValid(entity.toObject())\n        : schema.createIfValid(entity);\n      // TODO(breaking): remove once old verions no longer supported\n    } else {\n      entityCopy = entity;\n      unvisit = withTrackedEntities(unvisit);\n      unvisit.setLocal = entityCopy => (localCacheKey[pk] = entityCopy);\n    }\n\n    cycleCacheKey[pk] = trackingIndex;\n    if (entityCopy === undefined) {\n      // undefined indicates we should suspense (perhaps failed validation)\n      found = false;\n      deleted = true;\n    } else {\n      [localCacheKey[pk], found, deleted] = schema.denormalize(\n        entityCopy,\n        unvisit,\n      );\n    }\n    delete cycleCacheKey[pk];\n\n    const globalCacheEntry = getGlobalCacheEntry(entityCacheKey, pk);\n\n    // if in cycle, use the start of the cycle to track all deps\n    // otherwise, we use our own trackingIndex\n    const localKey = dependencies.slice(\n      cycleIndex.i === -1 ? trackingIndex : cycleIndex.i,\n    );\n\n    if (!globalCacheEntry.has(localKey)) {\n      globalCacheEntry.set(localKey, localCacheKey[pk]);\n    } else {\n      localCacheKey[pk] = globalCacheEntry.get(localKey);\n    }\n\n    // start of cycle - reset cycle detection\n    if (cycleIndex.i === trackingIndex) {\n      cycleIndex.i = -1;\n    }\n  } else {\n    // cycle detected\n    if (pk in cycleCacheKey) {\n      cycleIndex.i = cycleCacheKey[pk];\n    } else {\n      // with no cycle, globalCacheEntry will have already been set\n      dependencies.push(entity);\n    }\n  }\n\n  return [localCacheKey[pk], found, deleted];\n};\n\nconst getUnvisit = (\n  entities: Record<string, Record<string, any>>,\n  entityCache: DenormalizeCache['entities'],\n  resultCache: WeakListMap<object, any>,\n  localCache: Record<string, Record<string, any>>,\n) => {\n  const getEntity = getEntities(entities);\n  const dependencies: object[] = [];\n  const cycleIndex = { i: -1 };\n  const cycleCache = {};\n\n  function unvisit(\n    input: any,\n    schema: any,\n  ): [denormalized: any, found: boolean, deleted: boolean] {\n    if (!schema) return [input, true, false];\n\n    // null is considered intentional, thus always 'found' as true\n    if (input === null) {\n      return [input, true, false];\n    }\n\n    const hasDenormalize = typeof schema.denormalize === 'function';\n\n    if (!hasDenormalize && typeof schema === 'function') {\n      if (input instanceof schema) return [input, true, false];\n      // field deserialization should never count against 'found' (whether to used inferred results)\n      if (input === undefined) return [input, true, false];\n      return [new schema(input), true, false];\n    }\n\n    if (input === undefined) {\n      return [input, false, false];\n    }\n\n    if (!hasDenormalize && typeof schema === 'object') {\n      const method = Array.isArray(schema)\n        ? arrayDenormalize\n        : objectDenormalize;\n      return method(schema, input, unvisit);\n    }\n\n    if (isEntity(schema)) {\n      return unvisitEntity(\n        input,\n        schema,\n        unvisit,\n        getEntity,\n        localCache,\n        cycleCache,\n        entityCache,\n        dependencies,\n        cycleIndex,\n      );\n    }\n\n    if (hasDenormalize) {\n      return schema.denormalize(input, unvisit);\n    }\n\n    return [input, true, false];\n  }\n\n  //const wrappedUnvisit = withTrackedEntities(unvisit, globalKey);\n\n  return (\n    input: any,\n    schema: any,\n  ): [denormalized: any, found: boolean, deleted: boolean] => {\n    const ret = unvisit(input, schema);\n    // in the case where WeakMap cannot be used\n    // this test ensures null is properly excluded from WeakMap\n    if (Object(input) !== input) return ret;\n\n    dependencies.push(input);\n    if (!resultCache.has(dependencies)) {\n      resultCache.set(dependencies, ret[0]);\n      return ret;\n    } else {\n      return [resultCache.get(dependencies), ret[1], ret[2]];\n    }\n  };\n};\n\nconst getEntities = (entities: Record<string, any>) => {\n  const entityIsImmutable = isImmutable(entities);\n\n  return (\n    entityOrId: Record<string, any> | string,\n    schema: EntityInterface,\n  ) => {\n    const schemaKey = schema.key;\n\n    if (typeof entityOrId === 'object') {\n      return entityOrId;\n    }\n\n    if (entityIsImmutable) {\n      return entities.getIn([schemaKey, entityOrId]);\n    }\n\n    return entities[schemaKey]?.[entityOrId];\n  };\n};\n\ntype DenormalizeReturn<S extends Schema> =\n  | [\n      denormalized: Denormalize<S>,\n      found: true,\n      deleted: false,\n      resolvedEntities: Record<string, Record<string, any>>,\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: boolean,\n      deleted: true,\n      resolvedEntities: Record<string, Record<string, any>>,\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: false,\n      deleted: boolean,\n      resolvedEntities: Record<string, Record<string, any>>,\n    ];\n\n// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types\nexport const denormalize = <S extends Schema>(\n  input: unknown,\n  schema: S | undefined,\n  entities: any,\n  entityCache: DenormalizeCache['entities'] = {},\n  resultCache: WeakListMap<object, any> = new WeakListMap(),\n): DenormalizeReturn<S> => {\n  // undefined mean don't do anything\n  if (schema === undefined) {\n    return [input, true, false, {}] as [any, boolean, boolean, any];\n  }\n  if (input === undefined) {\n    return [undefined, false, false, {}] as [any, boolean, boolean, any];\n  }\n  const resolvedEntities: Record<string, Record<string, any>> = {};\n  const unvisit = getUnvisit(\n    entities,\n    entityCache,\n    resultCache,\n    resolvedEntities,\n  );\n  return [...unvisit(input, schema), resolvedEntities] as [\n    any,\n    boolean,\n    boolean,\n    any,\n  ];\n};\n\nexport const denormalizeSimple = <S extends Schema>(\n  input: any,\n  schema: S | undefined,\n  entities: any,\n  entityCache: DenormalizeCache['entities'] = {},\n  resultCache: WeakListMap<object, any> = new WeakListMap(),\n):\n  | [denormalized: Denormalize<S>, found: true, deleted: false]\n  | [denormalized: DenormalizeNullable<S>, found: boolean, deleted: true]\n  | [denormalized: DenormalizeNullable<S>, found: false, deleted: boolean] =>\n  denormalize(input, schema, entities, entityCache, resultCache).slice(\n    0,\n    3,\n  ) as any;\n\nfunction getGlobalCacheEntry(\n  entityCache: { [pk: string]: WeakListMap<object, EntityInterface<any>> },\n\n  id: any,\n) {\n  if (!entityCache[id]) entityCache[id] = new WeakListMap();\n  return entityCache[id];\n}\n\n// TODO(breaking): remove once unused\nfunction withTrackedEntities(unvisit: UnvisitFunction): UnvisitFunction {\n  // every time we nest, we want to unwrap back to the top.\n  // this is due to only needed the next level of nested entities for lookup\n  const originalUnvisit = unvisit.og || unvisit;\n  const wrappedUnvisit = (input: any, schema: any) =>\n    originalUnvisit(input, schema);\n  wrappedUnvisit.og = unvisit;\n  return wrappedUnvisit;\n}\n"],"mappings":"AACA,SAASA,QAAQ,QAAQ,eAAe;AACxC,SAASC,WAAW,IAAIC,gBAAgB,QAAQ,oBAAoB;AACpE,SAASC,WAAW,QAAQ,6BAA6B;AACzD,SAASF,WAAW,IAAIG,iBAAiB,QAAQ,qBAAqB;AAMtE,OAAOC,WAAW,MAAM,kBAAkB;AAE1C,MAAMC,aAAa,GAAG,CACpBC,UAAwC,EACxCC,MAAuB,EACvBC,OAAwB,EACxBC,SAGoB,EACpBC,UAA+C,EAC/CC,UAAkD,EAClDC,WAAyC,EACzCC,YAAsB,EACtBC,UAAyB,KACgD;EACzE,MAAMC,MAAM,GAAGN,SAAS,CAACH,UAAU,EAAEC,MAAM,CAAC;EAC5C,IACE,OAAOQ,MAAM,KAAK,QAAQ,IACzBA,MAAM,CAAYC,QAAQ,EAAE,CAACC,QAAQ,CAAC,SAAS,CAAC,EACjD;IACA,OAAO,CAACC,SAAS,EAAE,IAAI,EAAE,IAAI,CAAC;IAC9B;IACA;IACA;EACF;;EAEA,IAAI,OAAOH,MAAM,KAAK,QAAQ,IAAIA,MAAM,KAAK,IAAI,EAAE;IACjD,OAAO,CAACA,MAAM,EAAS,KAAK,EAAE,KAAK,CAAC;EACtC;EAEA,MAAMI,EAAE;EACN;EACA;EACA,OAAOb,UAAU,KAAK,QAAQ,GAC1BA,UAAU,GACVC,MAAM,CAACY,EAAE,CAACjB,WAAW,CAACa,MAAM,CAAC,GAAIA,MAAM,CAASK,IAAI,EAAE,GAAGL,MAAM,CAAC;EACtE;EACA;EACA;EACA,IAAII,EAAE,KAAKD,SAAS,IAAIC,EAAE,KAAK,EAAE,IAAIA,EAAE,KAAK,WAAW,EAAE;IACvD,OAAO,CAACJ,MAAM,EAAE,KAAK,EAAE,KAAK,CAAC;EAC/B;EAEA,MAAMM,GAAG,GAAGd,MAAM,CAACc,GAAG;EACtB,IAAI,EAAEA,GAAG,IAAIX,UAAU,CAAC,EAAE;IACxBA,UAAU,CAACW,GAAG,CAAC,GAAGC,MAAM,CAACC,MAAM,CAAC,IAAI,CAAC;EACvC;EACA,IAAI,EAAEF,GAAG,IAAIV,UAAU,CAAC,EAAE;IACxBA,UAAU,CAACU,GAAG,CAAC,GAAGC,MAAM,CAACC,MAAM,CAAC,IAAI,CAAC;EACvC;EACA,IAAI,EAAEF,GAAG,IAAIT,WAAW,CAAC,EAAE;IACzBA,WAAW,CAACS,GAAG,CAAC,GAAGC,MAAM,CAACC,MAAM,CAAC,IAAI,CAAC;EACxC;EACA,MAAMC,aAAa,GAAGd,UAAU,CAACW,GAAG,CAAC;EACrC,MAAMI,aAAa,GAAGd,UAAU,CAACU,GAAG,CAAC;EACrC,MAAMK,cAAc,GAAGd,WAAW,CAACS,GAAG,CAAC;EAEvC,IAAIM,KAAK,GAAG,IAAI;EAChB,IAAIC,OAAO,GAAG,KAAK;EAEnB,IAAI,CAACJ,aAAa,CAACL,EAAE,CAAC,EAAE;IACtB,MAAMU,aAAa,GAAGhB,YAAY,CAACiB,MAAM;IACzCjB,YAAY,CAACkB,IAAI,CAAChB,MAAM,CAAC;IAEzB,IAAIiB,UAAe;IACnB,IAAIzB,MAAM,CAAC0B,aAAa,EAAE;MACxBD,UAAU,GAAGR,aAAa,CAACL,EAAE,CAAC,GAAGjB,WAAW,CAACa,MAAM,CAAC,GAChDR,MAAM,CAAC0B,aAAa,CAAClB,MAAM,CAACmB,QAAQ,EAAE,CAAC,GACvC3B,MAAM,CAAC0B,aAAa,CAAClB,MAAM,CAAC;MAChC;IACF,CAAC,MAAM;MACLiB,UAAU,GAAGjB,MAAM;MACnBP,OAAO,GAAG2B,mBAAmB,CAAC3B,OAAO,CAAC;MACtCA,OAAO,CAAC4B,QAAQ,GAAGJ,UAAU,IAAKR,aAAa,CAACL,EAAE,CAAC,GAAGa,UAAW;IACnE;IAEAP,aAAa,CAACN,EAAE,CAAC,GAAGU,aAAa;IACjC,IAAIG,UAAU,KAAKd,SAAS,EAAE;MAC5B;MACAS,KAAK,GAAG,KAAK;MACbC,OAAO,GAAG,IAAI;IAChB,CAAC,MAAM;MACL,CAACJ,aAAa,CAACL,EAAE,CAAC,EAAEQ,KAAK,EAAEC,OAAO,CAAC,GAAGrB,MAAM,CAACP,WAAW,CACtDgC,UAAU,EACVxB,OAAO,CACR;IACH;IACA,OAAOiB,aAAa,CAACN,EAAE,CAAC;IAExB,MAAMkB,gBAAgB,GAAGC,mBAAmB,CAACZ,cAAc,EAAEP,EAAE,CAAC;;IAEhE;IACA;IACA,MAAMoB,QAAQ,GAAG1B,YAAY,CAAC2B,KAAK,CACjC1B,UAAU,CAAC2B,CAAC,KAAK,CAAC,CAAC,GAAGZ,aAAa,GAAGf,UAAU,CAAC2B,CAAC,CACnD;IAED,IAAI,CAACJ,gBAAgB,CAACK,GAAG,CAACH,QAAQ,CAAC,EAAE;MACnCF,gBAAgB,CAACM,GAAG,CAACJ,QAAQ,EAAEf,aAAa,CAACL,EAAE,CAAC,CAAC;IACnD,CAAC,MAAM;MACLK,aAAa,CAACL,EAAE,CAAC,GAAGkB,gBAAgB,CAACO,GAAG,CAACL,QAAQ,CAAC;IACpD;;IAEA;IACA,IAAIzB,UAAU,CAAC2B,CAAC,KAAKZ,aAAa,EAAE;MAClCf,UAAU,CAAC2B,CAAC,GAAG,CAAC,CAAC;IACnB;EACF,CAAC,MAAM;IACL;IACA,IAAItB,EAAE,IAAIM,aAAa,EAAE;MACvBX,UAAU,CAAC2B,CAAC,GAAGhB,aAAa,CAACN,EAAE,CAAC;IAClC,CAAC,MAAM;MACL;MACAN,YAAY,CAACkB,IAAI,CAAChB,MAAM,CAAC;IAC3B;EACF;EAEA,OAAO,CAACS,aAAa,CAACL,EAAE,CAAC,EAAEQ,KAAK,EAAEC,OAAO,CAAC;AAC5C,CAAC;AAED,MAAMiB,UAAU,GAAG,CACjBC,QAA6C,EAC7ClC,WAAyC,EACzCmC,WAAqC,EACrCrC,UAA+C,KAC5C;EACH,MAAMD,SAAS,GAAGuC,WAAW,CAACF,QAAQ,CAAC;EACvC,MAAMjC,YAAsB,GAAG,EAAE;EACjC,MAAMC,UAAU,GAAG;IAAE2B,CAAC,EAAE,CAAC;EAAE,CAAC;EAC5B,MAAM9B,UAAU,GAAG,CAAC,CAAC;EAErB,SAASH,OAAO,CACdyC,KAAU,EACV1C,MAAW,EAC4C;IACvD,IAAI,CAACA,MAAM,EAAE,OAAO,CAAC0C,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;;IAExC;IACA,IAAIA,KAAK,KAAK,IAAI,EAAE;MAClB,OAAO,CAACA,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;IAC7B;IAEA,MAAMC,cAAc,GAAG,OAAO3C,MAAM,CAACP,WAAW,KAAK,UAAU;IAE/D,IAAI,CAACkD,cAAc,IAAI,OAAO3C,MAAM,KAAK,UAAU,EAAE;MACnD,IAAI0C,KAAK,YAAY1C,MAAM,EAAE,OAAO,CAAC0C,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;MACxD;MACA,IAAIA,KAAK,KAAK/B,SAAS,EAAE,OAAO,CAAC+B,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;MACpD,OAAO,CAAC,IAAI1C,MAAM,CAAC0C,KAAK,CAAC,EAAE,IAAI,EAAE,KAAK,CAAC;IACzC;IAEA,IAAIA,KAAK,KAAK/B,SAAS,EAAE;MACvB,OAAO,CAAC+B,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC;IAC9B;IAEA,IAAI,CAACC,cAAc,IAAI,OAAO3C,MAAM,KAAK,QAAQ,EAAE;MACjD,MAAM4C,MAAM,GAAGC,KAAK,CAACC,OAAO,CAAC9C,MAAM,CAAC,GAChCN,gBAAgB,GAChBE,iBAAiB;MACrB,OAAOgD,MAAM,CAAC5C,MAAM,EAAE0C,KAAK,EAAEzC,OAAO,CAAC;IACvC;IAEA,IAAIT,QAAQ,CAACQ,MAAM,CAAC,EAAE;MACpB,OAAOF,aAAa,CAClB4C,KAAK,EACL1C,MAAM,EACNC,OAAO,EACPC,SAAS,EACTC,UAAU,EACVC,UAAU,EACVC,WAAW,EACXC,YAAY,EACZC,UAAU,CACX;IACH;IAEA,IAAIoC,cAAc,EAAE;MAClB,OAAO3C,MAAM,CAACP,WAAW,CAACiD,KAAK,EAAEzC,OAAO,CAAC;IAC3C;IAEA,OAAO,CAACyC,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC;EAC7B;;EAEA;;EAEA,OAAO,CACLA,KAAU,EACV1C,MAAW,KAC+C;IAC1D,MAAM+C,GAAG,GAAG9C,OAAO,CAACyC,KAAK,EAAE1C,MAAM,CAAC;IAClC;IACA;IACA,IAAIe,MAAM,CAAC2B,KAAK,CAAC,KAAKA,KAAK,EAAE,OAAOK,GAAG;IAEvCzC,YAAY,CAACkB,IAAI,CAACkB,KAAK,CAAC;IACxB,IAAI,CAACF,WAAW,CAACL,GAAG,CAAC7B,YAAY,CAAC,EAAE;MAClCkC,WAAW,CAACJ,GAAG,CAAC9B,YAAY,EAAEyC,GAAG,CAAC,CAAC,CAAC,CAAC;MACrC,OAAOA,GAAG;IACZ,CAAC,MAAM;MACL,OAAO,CAACP,WAAW,CAACH,GAAG,CAAC/B,YAAY,CAAC,EAAEyC,GAAG,CAAC,CAAC,CAAC,EAAEA,GAAG,CAAC,CAAC,CAAC,CAAC;IACxD;EACF,CAAC;AACH,CAAC;AAED,MAAMN,WAAW,GAAIF,QAA6B,IAAK;EACrD,MAAMS,iBAAiB,GAAGrD,WAAW,CAAC4C,QAAQ,CAAC;EAE/C,OAAO,CACLxC,UAAwC,EACxCC,MAAuB,KACpB;IAAA;IACH,MAAMiD,SAAS,GAAGjD,MAAM,CAACc,GAAG;IAE5B,IAAI,OAAOf,UAAU,KAAK,QAAQ,EAAE;MAClC,OAAOA,UAAU;IACnB;IAEA,IAAIiD,iBAAiB,EAAE;MACrB,OAAOT,QAAQ,CAACW,KAAK,CAAC,CAACD,SAAS,EAAElD,UAAU,CAAC,CAAC;IAChD;IAEA,8BAAOwC,QAAQ,CAACU,SAAS,CAAC,qBAAnB,oBAAsBlD,UAAU,CAAC;EAC1C,CAAC;AACH,CAAC;AAsBD;AACA,OAAO,MAAMN,WAAW,GAAG,CACzBiD,KAAc,EACd1C,MAAqB,EACrBuC,QAAa,EACblC,WAAyC,GAAG,CAAC,CAAC,EAC9CmC,WAAqC,GAAG,IAAI3C,WAAW,EAAE,KAChC;EACzB;EACA,IAAIG,MAAM,KAAKW,SAAS,EAAE;IACxB,OAAO,CAAC+B,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;EACjC;EACA,IAAIA,KAAK,KAAK/B,SAAS,EAAE;IACvB,OAAO,CAACA,SAAS,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;EACtC;EACA,MAAMwC,gBAAqD,GAAG,CAAC,CAAC;EAChE,MAAMlD,OAAO,GAAGqC,UAAU,CACxBC,QAAQ,EACRlC,WAAW,EACXmC,WAAW,EACXW,gBAAgB,CACjB;EACD,OAAO,CAAC,GAAGlD,OAAO,CAACyC,KAAK,EAAE1C,MAAM,CAAC,EAAEmD,gBAAgB,CAAC;AAMtD,CAAC;AAED,OAAO,MAAMC,iBAAiB,GAAG,CAC/BV,KAAU,EACV1C,MAAqB,EACrBuC,QAAa,EACblC,WAAyC,GAAG,CAAC,CAAC,EAC9CmC,WAAqC,GAAG,IAAI3C,WAAW,EAAE,KAKzDJ,WAAW,CAACiD,KAAK,EAAE1C,MAAM,EAAEuC,QAAQ,EAAElC,WAAW,EAAEmC,WAAW,CAAC,CAACP,KAAK,CAClE,CAAC,EACD,CAAC,CACK;AAEV,SAASF,mBAAmB,CAC1B1B,WAAwE,EAExEgD,EAAO,EACP;EACA,IAAI,CAAChD,WAAW,CAACgD,EAAE,CAAC,EAAEhD,WAAW,CAACgD,EAAE,CAAC,GAAG,IAAIxD,WAAW,EAAE;EACzD,OAAOQ,WAAW,CAACgD,EAAE,CAAC;AACxB;;AAEA;AACA,SAASzB,mBAAmB,CAAC3B,OAAwB,EAAmB;EACtE;EACA;EACA,MAAMqD,eAAe,GAAGrD,OAAO,CAACsD,EAAE,IAAItD,OAAO;EAC7C,MAAMuD,cAAc,GAAG,CAACd,KAAU,EAAE1C,MAAW,KAC7CsD,eAAe,CAACZ,KAAK,EAAE1C,MAAM,CAAC;EAChCwD,cAAc,CAACD,EAAE,GAAGtD,OAAO;EAC3B,OAAOuD,cAAc;AACvB"} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["isEntity","denormalize","arrayDenormalize","isImmutable","objectDenormalize","WeakEntityMap","getEntities","depToPaths","unvisitEntity","entityOrId","schema","unvisit","getEntity","getCache","localCache","cycleCache","dependencies","cycleIndex","entity","pk","key","toString","includes","undefined","toJS","Object","create","localCacheKey","cycleCacheKey","found","deleted","globalCache","cacheValue","get","value","push","trackingIndex","length","path","entityCopy","createIfValid","toObject","withTrackedEntities","setLocal","localKey","slice","i","set","getUnvisit","entities","entityCache","resultCache","getEntityCaches","input","hasDenormalize","method","Array","isArray","cachable","ret","entityPaths","unshift","entityCacheKey","WeakMap","wem","denormalizeSimple","originalUnvisit","og","wrappedUnvisit"],"sources":["../src/denormalize.ts"],"sourcesContent":["import type { Schema, EntityInterface, UnvisitFunction } from './interface.js';\nimport { isEntity } from './isEntity.js';\nimport { denormalize as arrayDenormalize } from './schemas/Array.js';\nimport { isImmutable } from './schemas/ImmutableUtils.js';\nimport { denormalize as objectDenormalize } from './schemas/Object.js';\nimport type {\n  Denormalize,\n  DenormalizeNullable,\n  DenormalizeCache,\n  Path,\n} from './types.js';\nimport WeakEntityMap, {\n  type Dep,\n  getEntities,\n  type GetEntity,\n  depToPaths,\n} from './WeakEntityMap.js';\n\ninterface EntityCacheValue {\n  dependencies: Dep[];\n  value: [any, boolean, boolean];\n}\nconst unvisitEntity = (\n  entityOrId: Record<string, any> | string,\n  schema: EntityInterface,\n  unvisit: UnvisitFunction,\n  getEntity: GetEntity,\n  getCache: ReturnType<typeof getEntityCaches>,\n  localCache: Record<string, Record<string, any>>,\n  cycleCache: Record<string, Record<string, number>>,\n  dependencies: Dep[],\n  cycleIndex: { i: number },\n): [denormalized: object | undefined, found: boolean, deleted: boolean] => {\n  const entity =\n    typeof entityOrId === 'object'\n      ? entityOrId\n      : getEntity({ pk: entityOrId, key: schema.key });\n  if (\n    typeof entity === 'symbol' &&\n    (entity as symbol).toString().includes('DELETED')\n  ) {\n    return [undefined, true, true];\n    // TODO: Change to this as breaking change once we only support newer entities\n    // also remove `(entity as symbol).toString().includes('DELETED')` and do this for all symbols\n    // return schema.denormalize(entity, unvisit);\n  }\n\n  if (typeof entity !== 'object' || entity === null) {\n    return [entity as any, false, false];\n  }\n\n  const pk =\n    // normalize must always place a string, because pk() return value is string | undefined\n    // therefore no need to check for numbers\n    typeof entityOrId === 'string'\n      ? entityOrId\n      : schema.pk(isImmutable(entity) ? (entity as any).toJS() : entity);\n  // if we can't generate a working pk; this is hopeless so let's give them what's already there\n  // otherwise, even when we aren't doing a lookup we want to turn the entityOrId object into the\n  // expected class, and cache that class for referential equality. PK is used for global equality lookup.\n  if (pk === undefined || pk === '' || pk === 'undefined') {\n    return [entity, false, false];\n  }\n\n  const key = schema.key;\n  if (!(key in localCache)) {\n    localCache[key] = Object.create(null);\n  }\n  if (!(key in cycleCache)) {\n    cycleCache[key] = Object.create(null);\n  }\n  const localCacheKey = localCache[key];\n  const cycleCacheKey = cycleCache[key];\n\n  let found = true;\n  let deleted = false;\n\n  // local cache lookup first\n  if (!localCacheKey[pk]) {\n    const globalCache: WeakEntityMap<object, EntityCacheValue> = getCache(\n      pk,\n      schema,\n    );\n    const [cacheValue] = globalCache.get(entity, getEntity);\n    // TODO: what if this just returned the deps - then we don't need to store them\n\n    if (cacheValue) {\n      localCacheKey[pk] = cacheValue.value[0];\n      // TODO: can we store the cache values instead of tracking *all* their sources?\n      // this is only used for setting results cache correctly. if we got this far we will def need to set as we would have already tried getting it\n      dependencies.push(...cacheValue.dependencies);\n      return cacheValue.value;\n    }\n    // if we don't find in denormalize cache then do full denormalize\n    else {\n      const trackingIndex = dependencies.length;\n      dependencies.push({ entity, path: { key, pk } });\n\n      let entityCopy: any;\n      if (schema.createIfValid) {\n        entityCopy = localCacheKey[pk] = isImmutable(entity)\n          ? schema.createIfValid(entity.toObject())\n          : schema.createIfValid(entity);\n        // TODO(breaking): remove once old verions no longer supported\n      } else {\n        entityCopy = entity;\n        unvisit = withTrackedEntities(unvisit);\n        unvisit.setLocal = entityCopy => (localCacheKey[pk] = entityCopy);\n      }\n\n      cycleCacheKey[pk] = trackingIndex;\n      if (entityCopy === undefined) {\n        // undefined indicates we should suspense (perhaps failed validation)\n        found = false;\n        deleted = true;\n      } else {\n        [localCacheKey[pk], found, deleted] = schema.denormalize(\n          entityCopy,\n          unvisit,\n        );\n      }\n      delete cycleCacheKey[pk];\n\n      // if in cycle, use the start of the cycle to track all deps\n      // otherwise, we use our own trackingIndex\n      const localKey = dependencies.slice(\n        cycleIndex.i === -1 ? trackingIndex : cycleIndex.i,\n      );\n      const cacheValue: EntityCacheValue = {\n        dependencies: localKey,\n        value: [localCacheKey[pk], found, deleted],\n      };\n      globalCache.set(localKey, cacheValue);\n\n      // start of cycle - reset cycle detection\n      if (cycleIndex.i === trackingIndex) {\n        cycleIndex.i = -1;\n      }\n    }\n  } else {\n    // cycle detected\n    if (pk in cycleCacheKey) {\n      cycleIndex.i = cycleCacheKey[pk];\n    } else {\n      // with no cycle, globalCacheEntry will have already been set\n      dependencies.push({ entity, path: { key, pk } });\n    }\n  }\n\n  return [localCacheKey[pk], found, deleted];\n};\n\nconst getUnvisit = (\n  entities: Record<string, Record<string, any>>,\n  entityCache: DenormalizeCache['entities'],\n  resultCache: DenormalizeCache['results'][string],\n) => {\n  const getEntity = getEntities(entities);\n  const getCache = getEntityCaches(entityCache);\n  const localCache: Record<string, Record<string, any>> = {};\n  const dependencies: Dep[] = [];\n  const cycleIndex = { i: -1 };\n  const cycleCache = {};\n\n  function unvisit(\n    input: any,\n    schema: any,\n  ): [denormalized: any, found: boolean, deleted: boolean] {\n    if (!schema) return [input, true, false];\n\n    // null is considered intentional, thus always 'found' as true\n    if (input === null) {\n      return [input, true, false];\n    }\n\n    const hasDenormalize = typeof schema.denormalize === 'function';\n\n    // deserialize fields (like Date)\n    if (!hasDenormalize && typeof schema === 'function') {\n      if (input instanceof schema) return [input, true, false];\n      // field deserialization should never count against 'found' (whether to used inferred results)\n      if (input === undefined) return [input, true, false];\n      return [new schema(input), true, false];\n    }\n\n    if (input === undefined) {\n      return [input, false, false];\n    }\n\n    if (!hasDenormalize && typeof schema === 'object') {\n      const method = Array.isArray(schema)\n        ? arrayDenormalize\n        : objectDenormalize;\n      return method(schema, input, unvisit);\n    }\n\n    if (isEntity(schema)) {\n      return unvisitEntity(\n        input,\n        schema,\n        unvisit,\n        getEntity,\n        getCache,\n        localCache,\n        cycleCache,\n        dependencies,\n        cycleIndex,\n      );\n    }\n\n    if (hasDenormalize) {\n      return schema.denormalize(input, unvisit);\n    }\n\n    return [input, true, false];\n  }\n\n  return (\n    input: any,\n    schema: any,\n  ): [\n    denormalized: any,\n    found: boolean,\n    deleted: boolean,\n    entityPaths: Path[],\n  ] => {\n    // in the case where WeakMap cannot be used\n    // this test ensures null is properly excluded from WeakMap\n    const cachable = Object(input) === input && Object(schema) === schema;\n    if (!cachable) {\n      const ret = unvisit(input, schema);\n      // this is faster than spread\n      // https://www.measurethat.net/Benchmarks/Show/23636/0/spread-with-tuples\n      return [ret[0], ret[1], ret[2], depToPaths(dependencies)];\n    }\n\n    let [ret, entityPaths] = resultCache.get(input, getEntity);\n\n    if (ret === undefined) {\n      ret = unvisit(input, schema);\n      // we want to do this before we add our 'input' entry\n      entityPaths = depToPaths(dependencies);\n      // for the first entry, `path` is ignored so empty members is fine\n      dependencies.unshift({ entity: input, path: { key: '', pk: '' } });\n      resultCache.set(dependencies, ret);\n    }\n\n    return [ret[0], ret[1], ret[2], entityPaths as Path[]];\n  };\n};\n\nconst getEntityCaches = (entityCache: DenormalizeCache['entities']) => {\n  return (pk: string, schema: EntityInterface) => {\n    const key = schema.key;\n\n    if (!(key in entityCache)) {\n      entityCache[key] = Object.create(null);\n    }\n    const entityCacheKey = entityCache[key];\n    if (!entityCacheKey[pk])\n      entityCacheKey[pk] = new WeakMap<\n        EntityInterface,\n        WeakEntityMap<object, any>\n      >();\n\n    let wem: WeakEntityMap<object, any> = entityCacheKey[pk].get(schema) as any;\n    if (!wem) {\n      wem = new WeakEntityMap<object, any>();\n      entityCacheKey[pk].set(schema, wem);\n    }\n\n    return wem;\n  };\n};\n\ntype DenormalizeReturn<S extends Schema> =\n  | [\n      denormalized: Denormalize<S>,\n      found: true,\n      deleted: false,\n      entityPaths: Path[],\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: boolean,\n      deleted: true,\n      entityPaths: Path[],\n    ]\n  | [\n      denormalized: DenormalizeNullable<S>,\n      found: false,\n      deleted: boolean,\n      entityPaths: Path[],\n    ];\n\n// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types\nexport const denormalize = <S extends Schema>(\n  input: unknown,\n  schema: S | undefined,\n  entities: any,\n  entityCache: DenormalizeCache['entities'] = {},\n  resultCache: DenormalizeCache['results'][string] = new WeakEntityMap(),\n): DenormalizeReturn<S> => {\n  // undefined mean don't do anything\n  if (schema === undefined) {\n    return [input, true, false, []] as [any, boolean, boolean, any];\n  }\n  if (input === undefined) {\n    return [undefined, false, false, []] as [any, boolean, boolean, any];\n  }\n  return getUnvisit(entities, entityCache, resultCache)(input, schema);\n};\n\nexport const denormalizeSimple = <S extends Schema>(\n  input: any,\n  schema: S | undefined,\n  entities: any,\n  entityCache?: DenormalizeCache['entities'],\n  resultCache?: DenormalizeCache['results'][string],\n):\n  | [denormalized: Denormalize<S>, found: true, deleted: false]\n  | [denormalized: DenormalizeNullable<S>, found: boolean, deleted: true]\n  | [denormalized: DenormalizeNullable<S>, found: false, deleted: boolean] =>\n  denormalize(input, schema, entities, entityCache, resultCache).slice(\n    0,\n    3,\n  ) as any;\n\n// TODO(breaking): remove once unused\nfunction withTrackedEntities(unvisit: UnvisitFunction): UnvisitFunction {\n  // every time we nest, we want to unwrap back to the top.\n  // this is due to only needed the next level of nested entities for lookup\n  const originalUnvisit = unvisit.og || unvisit;\n  const wrappedUnvisit = (input: any, schema: any) =>\n    originalUnvisit(input, schema);\n  wrappedUnvisit.og = unvisit;\n  return wrappedUnvisit;\n}\n"],"mappings":"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"} |
import { denormalize } from './denormalize.js'; | ||
import { isEntity } from './isEntity.js'; | ||
import { normalize } from './normalize.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
import WeakEntityMap from './WeakEntityMap.js'; | ||
export { default as inferResults } from './inferResults.js'; | ||
export { DELETED } from './special.js'; | ||
export type { AbstractInstanceType, NormalizeReturnType, NormalizedSchema, DenormalizeReturnType, DenormalizeCache, } from './types.js'; | ||
export type { AbstractInstanceType, NormalizeReturnType, NormalizedSchema, DenormalizeReturnType, DenormalizeCache, Path, } from './types.js'; | ||
export * from './endpoint/types.js'; | ||
@@ -12,3 +12,3 @@ export * from './interface.js'; | ||
export * from './normal.js'; | ||
export { denormalize, normalize, isEntity, WeakListMap }; | ||
export { denormalize, normalize, isEntity, WeakEntityMap }; | ||
//# sourceMappingURL=index.d.ts.map |
@@ -7,3 +7,3 @@ Object.hasOwn = Object.hasOwn || /* istanbul ignore next */function hasOwn(it, key) { | ||
import { normalize } from './normalize.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
import WeakEntityMap from './WeakEntityMap.js'; | ||
export { default as inferResults } from './inferResults.js'; | ||
@@ -15,3 +15,3 @@ export { DELETED } from './special.js'; | ||
export * from './normal.js'; | ||
export { denormalize, normalize, isEntity, WeakListMap }; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJPYmplY3QiLCJoYXNPd24iLCJpdCIsImtleSIsInByb3RvdHlwZSIsImhhc093blByb3BlcnR5IiwiY2FsbCIsImRlbm9ybWFsaXplIiwiaXNFbnRpdHkiLCJub3JtYWxpemUiLCJXZWFrTGlzdE1hcCIsImRlZmF1bHQiLCJpbmZlclJlc3VsdHMiLCJERUxFVEVEIl0sInNvdXJjZXMiOlsiLi4vc3JjL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbIk9iamVjdC5oYXNPd24gPVxuICBPYmplY3QuaGFzT3duIHx8XG4gIC8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICovIGZ1bmN0aW9uIGhhc093bihpdCwga2V5KSB7XG4gICAgcmV0dXJuIE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbChpdCwga2V5KTtcbiAgfTtcbmltcG9ydCB7IGRlbm9ybWFsaXplIH0gZnJvbSAnLi9kZW5vcm1hbGl6ZS5qcyc7XG5pbXBvcnQgeyBpc0VudGl0eSB9IGZyb20gJy4vaXNFbnRpdHkuanMnO1xuaW1wb3J0IHsgbm9ybWFsaXplIH0gZnJvbSAnLi9ub3JtYWxpemUuanMnO1xuaW1wb3J0IFdlYWtMaXN0TWFwIGZyb20gJy4vV2Vha0xpc3RNYXAuanMnO1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBpbmZlclJlc3VsdHMgfSBmcm9tICcuL2luZmVyUmVzdWx0cy5qcyc7XG5leHBvcnQgeyBERUxFVEVEIH0gZnJvbSAnLi9zcGVjaWFsLmpzJztcblxuZXhwb3J0IHR5cGUge1xuICBBYnN0cmFjdEluc3RhbmNlVHlwZSxcbiAgTm9ybWFsaXplUmV0dXJuVHlwZSxcbiAgTm9ybWFsaXplZFNjaGVtYSxcbiAgRGVub3JtYWxpemVSZXR1cm5UeXBlLFxuICBEZW5vcm1hbGl6ZUNhY2hlLFxufSBmcm9tICcuL3R5cGVzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vZW5kcG9pbnQvdHlwZXMuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9FeHBpcnkuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9ub3JtYWwuanMnO1xuXG5leHBvcnQgeyBkZW5vcm1hbGl6ZSwgbm9ybWFsaXplLCBpc0VudGl0eSwgV2Vha0xpc3RNYXAgfTtcbiJdLCJtYXBwaW5ncyI6IkFBQUFBLE1BQU0sQ0FBQ0MsTUFBTSxHQUNYRCxNQUFNLENBQUNDLE1BQU0sSUFDYiwwQkFBMkIsU0FBU0EsTUFBTSxDQUFDQyxFQUFFLEVBQUVDLEdBQUcsRUFBRTtFQUNsRCxPQUFPSCxNQUFNLENBQUNJLFNBQVMsQ0FBQ0MsY0FBYyxDQUFDQyxJQUFJLENBQUNKLEVBQUUsRUFBRUMsR0FBRyxDQUFDO0FBQ3RELENBQUM7QUFDSCxTQUFTSSxXQUFXLFFBQVEsa0JBQWtCO0FBQzlDLFNBQVNDLFFBQVEsUUFBUSxlQUFlO0FBQ3hDLFNBQVNDLFNBQVMsUUFBUSxnQkFBZ0I7QUFDMUMsT0FBT0MsV0FBVyxNQUFNLGtCQUFrQjtBQUMxQyxTQUFTQyxPQUFPLElBQUlDLFlBQVksUUFBUSxtQkFBbUI7QUFDM0QsU0FBU0MsT0FBTyxRQUFRLGNBQWM7QUFTdEMsY0FBYyxxQkFBcUI7QUFDbkMsY0FBYyxnQkFBZ0I7QUFDOUIsY0FBYyxhQUFhO0FBQzNCLGNBQWMsYUFBYTtBQUUzQixTQUFTTixXQUFXLEVBQUVFLFNBQVMsRUFBRUQsUUFBUSxFQUFFRSxXQUFXIn0= | ||
export { denormalize, normalize, isEntity, WeakEntityMap }; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJPYmplY3QiLCJoYXNPd24iLCJpdCIsImtleSIsInByb3RvdHlwZSIsImhhc093blByb3BlcnR5IiwiY2FsbCIsImRlbm9ybWFsaXplIiwiaXNFbnRpdHkiLCJub3JtYWxpemUiLCJXZWFrRW50aXR5TWFwIiwiZGVmYXVsdCIsImluZmVyUmVzdWx0cyIsIkRFTEVURUQiXSwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiT2JqZWN0Lmhhc093biA9XG4gIE9iamVjdC5oYXNPd24gfHxcbiAgLyogaXN0YW5idWwgaWdub3JlIG5leHQgKi8gZnVuY3Rpb24gaGFzT3duKGl0LCBrZXkpIHtcbiAgICByZXR1cm4gT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eS5jYWxsKGl0LCBrZXkpO1xuICB9O1xuaW1wb3J0IHsgZGVub3JtYWxpemUgfSBmcm9tICcuL2Rlbm9ybWFsaXplLmpzJztcbmltcG9ydCB7IGlzRW50aXR5IH0gZnJvbSAnLi9pc0VudGl0eS5qcyc7XG5pbXBvcnQgeyBub3JtYWxpemUgfSBmcm9tICcuL25vcm1hbGl6ZS5qcyc7XG5pbXBvcnQgV2Vha0VudGl0eU1hcCBmcm9tICcuL1dlYWtFbnRpdHlNYXAuanMnO1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBpbmZlclJlc3VsdHMgfSBmcm9tICcuL2luZmVyUmVzdWx0cy5qcyc7XG5leHBvcnQgeyBERUxFVEVEIH0gZnJvbSAnLi9zcGVjaWFsLmpzJztcblxuZXhwb3J0IHR5cGUge1xuICBBYnN0cmFjdEluc3RhbmNlVHlwZSxcbiAgTm9ybWFsaXplUmV0dXJuVHlwZSxcbiAgTm9ybWFsaXplZFNjaGVtYSxcbiAgRGVub3JtYWxpemVSZXR1cm5UeXBlLFxuICBEZW5vcm1hbGl6ZUNhY2hlLFxuICBQYXRoLFxufSBmcm9tICcuL3R5cGVzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vZW5kcG9pbnQvdHlwZXMuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9FeHBpcnkuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9ub3JtYWwuanMnO1xuXG5leHBvcnQgeyBkZW5vcm1hbGl6ZSwgbm9ybWFsaXplLCBpc0VudGl0eSwgV2Vha0VudGl0eU1hcCB9O1xuIl0sIm1hcHBpbmdzIjoiQUFBQUEsTUFBTSxDQUFDQyxNQUFNLEdBQ1hELE1BQU0sQ0FBQ0MsTUFBTSxJQUNiLDBCQUEyQixTQUFTQSxNQUFNLENBQUNDLEVBQUUsRUFBRUMsR0FBRyxFQUFFO0VBQ2xELE9BQU9ILE1BQU0sQ0FBQ0ksU0FBUyxDQUFDQyxjQUFjLENBQUNDLElBQUksQ0FBQ0osRUFBRSxFQUFFQyxHQUFHLENBQUM7QUFDdEQsQ0FBQztBQUNILFNBQVNJLFdBQVcsUUFBUSxrQkFBa0I7QUFDOUMsU0FBU0MsUUFBUSxRQUFRLGVBQWU7QUFDeEMsU0FBU0MsU0FBUyxRQUFRLGdCQUFnQjtBQUMxQyxPQUFPQyxhQUFhLE1BQU0sb0JBQW9CO0FBQzlDLFNBQVNDLE9BQU8sSUFBSUMsWUFBWSxRQUFRLG1CQUFtQjtBQUMzRCxTQUFTQyxPQUFPLFFBQVEsY0FBYztBQVV0QyxjQUFjLHFCQUFxQjtBQUNuQyxjQUFjLGdCQUFnQjtBQUM5QixjQUFjLGFBQWE7QUFDM0IsY0FBYyxhQUFhO0FBRTNCLFNBQVNOLFdBQVcsRUFBRUUsU0FBUyxFQUFFRCxRQUFRLEVBQUVFLGFBQWEifQ== |
import type { Schema, Serializable, EntityInterface, NormalizedIndex, SchemaClass } from './interface.js'; | ||
import type WeakListMap from './WeakListMap.js'; | ||
import type WeakEntityMap from './WeakEntityMap.js'; | ||
export interface Path { | ||
key: string; | ||
pk: string; | ||
} | ||
export type AbstractInstanceType<T> = T extends { | ||
@@ -36,7 +40,7 @@ prototype: infer U; | ||
[key: string]: { | ||
[pk: string]: WeakListMap<object, EntityInterface>; | ||
[pk: string]: WeakMap<EntityInterface, WeakEntityMap<object, any>>; | ||
}; | ||
}; | ||
results: { | ||
[key: string]: WeakListMap<object, any>; | ||
[key: string]: WeakEntityMap<object, any>; | ||
}; | ||
@@ -43,0 +47,0 @@ } |
export {}; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL3R5cGVzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgU2NoZW1hLFxuICBTZXJpYWxpemFibGUsXG4gIEVudGl0eUludGVyZmFjZSxcbiAgTm9ybWFsaXplZEluZGV4LFxuICBTY2hlbWFDbGFzcyxcbn0gZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuaW1wb3J0IHR5cGUgV2Vha0xpc3RNYXAgZnJvbSAnLi9XZWFrTGlzdE1hcC5qcyc7XG5cbmV4cG9ydCB0eXBlIEFic3RyYWN0SW5zdGFuY2VUeXBlPFQ+ID0gVCBleHRlbmRzIHsgcHJvdG90eXBlOiBpbmZlciBVIH1cbiAgPyBVXG4gIDogbmV2ZXI7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZWRFbnRpdHk8VD4gPSBUIGV4dGVuZHMge1xuICBwcm90b3R5cGU6IGluZmVyIFU7XG4gIHNjaGVtYTogaW5mZXIgUztcbn1cbiAgPyB7IFtLIGluIEV4Y2x1ZGU8a2V5b2YgVSwga2V5b2YgUz5dOiBVW0tdIH0gJiB7IFtLIGluIGtleW9mIFNdOiBzdHJpbmcgfVxuICA6IG5ldmVyO1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gRGVub3JtYWxpemU8U1tLXT4gOiBTW0tdO1xufTtcblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVOdWxsYWJsZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gRGVub3JtYWxpemVOdWxsYWJsZTxTW0tdPiA6IFNbS107XG59O1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemVPYmplY3Q8UyBleHRlbmRzIFJlY29yZDxzdHJpbmcsIGFueT4+ID0ge1xuICBbSyBpbiBrZXlvZiBTXTogU1tLXSBleHRlbmRzIFNjaGVtYSA/IE5vcm1hbGl6ZTxTW0tdPiA6IFNbS107XG59O1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemVkTnVsbGFibGVPYmplY3Q8UyBleHRlbmRzIFJlY29yZDxzdHJpbmcsIGFueT4+ID0ge1xuICBbSyBpbiBrZXlvZiBTXTogU1tLXSBleHRlbmRzIFNjaGVtYSA/IE5vcm1hbGl6ZU51bGxhYmxlPFNbS10+IDogU1tLXTtcbn07XG5cbmludGVyZmFjZSBOZXN0ZWRTY2hlbWFDbGFzczxUID0gYW55PiB7XG4gIHNjaGVtYTogUmVjb3JkPHN0cmluZywgU2NoZW1hPjtcbiAgcHJvdG90eXBlOiBUO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFJlY29yZENsYXNzPFQgPSBhbnk+IGV4dGVuZHMgTmVzdGVkU2NoZW1hQ2xhc3M8VD4ge1xuICBmcm9tSlM6ICguLi5hcmdzOiBhbnkpID0+IEFic3RyYWN0SW5zdGFuY2VUeXBlPFQ+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERlbm9ybWFsaXplQ2FjaGUge1xuICBlbnRpdGllczoge1xuICAgIFtrZXk6IHN0cmluZ106IHtcbiAgICAgIFtwazogc3RyaW5nXTogV2Vha0xpc3RNYXA8b2JqZWN0LCBFbnRpdHlJbnRlcmZhY2U+O1xuICAgIH07XG4gIH07XG4gIHJlc3VsdHM6IHtcbiAgICBba2V5OiBzdHJpbmddOiBXZWFrTGlzdE1hcDxvYmplY3QsIGFueT47XG4gIH07XG59XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplTnVsbGFibGVOZXN0ZWRTY2hlbWE8UyBleHRlbmRzIE5lc3RlZFNjaGVtYUNsYXNzPiA9XG4gIGtleW9mIFNbJ3NjaGVtYSddIGV4dGVuZHMgbmV2ZXJcbiAgICA/IFNbJ3Byb3RvdHlwZSddIC8vIHRoaXMgaXMgdGhlIGNhc2Ugb2YgYSBub24tc2V0IHNjaGVtYSwgd2hpY2ggbWVhbnMgaXQgYWN0dWFsbHkgaGFzIG5vIG1lbWJlcnNcbiAgICA6IHN0cmluZyBleHRlbmRzIGtleW9mIFNbJ3NjaGVtYSddXG4gICAgPyBTWydwcm90b3R5cGUnXVxuICAgIDogU1sncHJvdG90eXBlJ10gLyomIHtcbiAgICAgICAgW0sgaW4ga2V5b2YgU1snc2NoZW1hJ11dOiBEZW5vcm1hbGl6ZU51bGxhYmxlPFNbJ3NjaGVtYSddW0tdPjtcbiAgICAgIH0qLztcblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVSZXR1cm5UeXBlPFQ+ID0gVCBleHRlbmRzIChcbiAgaW5wdXQ6IGFueSxcbiAgdW52aXNpdDogYW55LFxuKSA9PiBbaW5mZXIgUiwgYW55LCBhbnldXG4gID8gUlxuICA6IG5ldmVyO1xuZXhwb3J0IHR5cGUgTm9ybWFsaXplUmV0dXJuVHlwZTxUPiA9IFQgZXh0ZW5kcyAoLi4uYXJnczogYW55KSA9PiBpbmZlciBSXG4gID8gUlxuICA6IG5ldmVyO1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZTxTPiA9IFMgZXh0ZW5kcyBFbnRpdHlJbnRlcmZhY2U8aW5mZXIgVT5cbiAgPyBVXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gQWJzdHJhY3RJbnN0YW5jZVR5cGU8Uz5cbiAgOiBTIGV4dGVuZHMgU2NoZW1hQ2xhc3NcbiAgPyBEZW5vcm1hbGl6ZVJldHVyblR5cGU8U1snZGVub3JtYWxpemUnXT5cbiAgOiBTIGV4dGVuZHMgU2VyaWFsaXphYmxlPGluZmVyIFQ+XG4gID8gVFxuICA6IFMgZXh0ZW5kcyBBcnJheTxpbmZlciBGPlxuICA/IERlbm9ybWFsaXplPEY+W11cbiAgOiBTIGV4dGVuZHMgeyBbSzogc3RyaW5nXTogYW55IH1cbiAgPyBEZW5vcm1hbGl6ZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplTnVsbGFibGU8Uz4gPSBTIGV4dGVuZHMgRW50aXR5SW50ZXJmYWNlPGFueT5cbiAgPyBEZW5vcm1hbGl6ZU51bGxhYmxlTmVzdGVkU2NoZW1hPFM+IHwgdW5kZWZpbmVkXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gRGVub3JtYWxpemVOdWxsYWJsZU5lc3RlZFNjaGVtYTxTPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IERlbm9ybWFsaXplUmV0dXJuVHlwZTxTWydfZGVub3JtYWxpemVOdWxsYWJsZSddPlxuICA6IFMgZXh0ZW5kcyBTZXJpYWxpemFibGU8aW5mZXIgVD5cbiAgPyBUXG4gIDogUyBleHRlbmRzIEFycmF5PGluZmVyIEY+XG4gID8gRGVub3JtYWxpemU8Rj5bXSB8IHVuZGVmaW5lZFxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IERlbm9ybWFsaXplTnVsbGFibGVPYmplY3Q8Uz5cbiAgOiBTO1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemU8Uz4gPSBTIGV4dGVuZHMgRW50aXR5SW50ZXJmYWNlXG4gID8gc3RyaW5nXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gTm9ybWFsaXplT2JqZWN0PFNbJ3NjaGVtYSddPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IE5vcm1hbGl6ZVJldHVyblR5cGU8U1snbm9ybWFsaXplJ10+XG4gIDogUyBleHRlbmRzIFNlcmlhbGl6YWJsZTxpbmZlciBUPlxuICA/IFRcbiAgOiBTIGV4dGVuZHMgQXJyYXk8aW5mZXIgRj5cbiAgPyBOb3JtYWxpemU8Rj5bXVxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IE5vcm1hbGl6ZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZU51bGxhYmxlPFM+ID0gUyBleHRlbmRzIEVudGl0eUludGVyZmFjZVxuICA/IHN0cmluZyB8IHVuZGVmaW5lZFxuICA6IFMgZXh0ZW5kcyBSZWNvcmRDbGFzc1xuICA/IE5vcm1hbGl6ZWROdWxsYWJsZU9iamVjdDxTWydzY2hlbWEnXT5cbiAgOiBTIGV4dGVuZHMgU2NoZW1hQ2xhc3NcbiAgPyBOb3JtYWxpemVSZXR1cm5UeXBlPFNbJ19ub3JtYWxpemVOdWxsYWJsZSddPlxuICA6IFMgZXh0ZW5kcyBTZXJpYWxpemFibGU8aW5mZXIgVD5cbiAgPyBUXG4gIDogUyBleHRlbmRzIEFycmF5PGluZmVyIEY+XG4gID8gTm9ybWFsaXplPEY+W10gfCB1bmRlZmluZWRcbiAgOiBTIGV4dGVuZHMgeyBbSzogc3RyaW5nXTogYW55IH1cbiAgPyBOb3JtYWxpemVkTnVsbGFibGVPYmplY3Q8Uz5cbiAgOiBTO1xuXG5leHBvcnQgdHlwZSBOb3JtYWxpemVkU2NoZW1hPEUsIFI+ID0ge1xuICBlbnRpdGllczogRTtcbiAgcmVzdWx0OiBSO1xuICBpbmRleGVzOiBOb3JtYWxpemVkSW5kZXg7XG4gIGVudGl0eU1ldGE6IHtcbiAgICByZWFkb25seSBbZW50aXR5S2V5OiBzdHJpbmddOiB7XG4gICAgICByZWFkb25seSBbcGs6IHN0cmluZ106IHtcbiAgICAgICAgcmVhZG9ubHkgZGF0ZTogbnVtYmVyO1xuICAgICAgICByZWFkb25seSBleHBpcmVzQXQ6IG51bWJlcjtcbiAgICAgICAgcmVhZG9ubHkgZmV0Y2hlZEF0OiBudW1iZXI7XG4gICAgICB9O1xuICAgIH07XG4gIH07XG59O1xuXG5leHBvcnQgdHlwZSBFbnRpdHlNYXA8VCA9IGFueT4gPSBSZWNvcmQ8c3RyaW5nLCBFbnRpdHlJbnRlcmZhY2U8VD4+O1xuIl0sIm1hcHBpbmdzIjoiIn0= | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vc3JjL3R5cGVzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgU2NoZW1hLFxuICBTZXJpYWxpemFibGUsXG4gIEVudGl0eUludGVyZmFjZSxcbiAgTm9ybWFsaXplZEluZGV4LFxuICBTY2hlbWFDbGFzcyxcbn0gZnJvbSAnLi9pbnRlcmZhY2UuanMnO1xuaW1wb3J0IHR5cGUgV2Vha0VudGl0eU1hcCBmcm9tICcuL1dlYWtFbnRpdHlNYXAuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBhdGgge1xuICBrZXk6IHN0cmluZztcbiAgcGs6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQWJzdHJhY3RJbnN0YW5jZVR5cGU8VD4gPSBUIGV4dGVuZHMgeyBwcm90b3R5cGU6IGluZmVyIFUgfVxuICA/IFVcbiAgOiBuZXZlcjtcblxuZXhwb3J0IHR5cGUgTm9ybWFsaXplZEVudGl0eTxUPiA9IFQgZXh0ZW5kcyB7XG4gIHByb3RvdHlwZTogaW5mZXIgVTtcbiAgc2NoZW1hOiBpbmZlciBTO1xufVxuICA/IHsgW0sgaW4gRXhjbHVkZTxrZXlvZiBVLCBrZXlvZiBTPl06IFVbS10gfSAmIHsgW0sgaW4ga2V5b2YgU106IHN0cmluZyB9XG4gIDogbmV2ZXI7XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplT2JqZWN0PFMgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBhbnk+PiA9IHtcbiAgW0sgaW4ga2V5b2YgU106IFNbS10gZXh0ZW5kcyBTY2hlbWEgPyBEZW5vcm1hbGl6ZTxTW0tdPiA6IFNbS107XG59O1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZU51bGxhYmxlT2JqZWN0PFMgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBhbnk+PiA9IHtcbiAgW0sgaW4ga2V5b2YgU106IFNbS10gZXh0ZW5kcyBTY2hlbWEgPyBEZW5vcm1hbGl6ZU51bGxhYmxlPFNbS10+IDogU1tLXTtcbn07XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gTm9ybWFsaXplPFNbS10+IDogU1tLXTtcbn07XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZWROdWxsYWJsZU9iamVjdDxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55Pj4gPSB7XG4gIFtLIGluIGtleW9mIFNdOiBTW0tdIGV4dGVuZHMgU2NoZW1hID8gTm9ybWFsaXplTnVsbGFibGU8U1tLXT4gOiBTW0tdO1xufTtcblxuaW50ZXJmYWNlIE5lc3RlZFNjaGVtYUNsYXNzPFQgPSBhbnk+IHtcbiAgc2NoZW1hOiBSZWNvcmQ8c3RyaW5nLCBTY2hlbWE+O1xuICBwcm90b3R5cGU6IFQ7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVjb3JkQ2xhc3M8VCA9IGFueT4gZXh0ZW5kcyBOZXN0ZWRTY2hlbWFDbGFzczxUPiB7XG4gIGZyb21KUzogKC4uLmFyZ3M6IGFueSkgPT4gQWJzdHJhY3RJbnN0YW5jZVR5cGU8VD47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGVub3JtYWxpemVDYWNoZSB7XG4gIGVudGl0aWVzOiB7XG4gICAgW2tleTogc3RyaW5nXToge1xuICAgICAgW3BrOiBzdHJpbmddOiBXZWFrTWFwPEVudGl0eUludGVyZmFjZSwgV2Vha0VudGl0eU1hcDxvYmplY3QsIGFueT4+O1xuICAgIH07XG4gIH07XG4gIHJlc3VsdHM6IHtcbiAgICBba2V5OiBzdHJpbmddOiBXZWFrRW50aXR5TWFwPG9iamVjdCwgYW55PjtcbiAgfTtcbn1cblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVOdWxsYWJsZU5lc3RlZFNjaGVtYTxTIGV4dGVuZHMgTmVzdGVkU2NoZW1hQ2xhc3M+ID1cbiAga2V5b2YgU1snc2NoZW1hJ10gZXh0ZW5kcyBuZXZlclxuICAgID8gU1sncHJvdG90eXBlJ10gLy8gdGhpcyBpcyB0aGUgY2FzZSBvZiBhIG5vbi1zZXQgc2NoZW1hLCB3aGljaCBtZWFucyBpdCBhY3R1YWxseSBoYXMgbm8gbWVtYmVyc1xuICAgIDogc3RyaW5nIGV4dGVuZHMga2V5b2YgU1snc2NoZW1hJ11cbiAgICA/IFNbJ3Byb3RvdHlwZSddXG4gICAgOiBTWydwcm90b3R5cGUnXSAvKiYge1xuICAgICAgICBbSyBpbiBrZXlvZiBTWydzY2hlbWEnXV06IERlbm9ybWFsaXplTnVsbGFibGU8U1snc2NoZW1hJ11bS10+O1xuICAgICAgfSovO1xuXG5leHBvcnQgdHlwZSBEZW5vcm1hbGl6ZVJldHVyblR5cGU8VD4gPSBUIGV4dGVuZHMgKFxuICBpbnB1dDogYW55LFxuICB1bnZpc2l0OiBhbnksXG4pID0+IFtpbmZlciBSLCBhbnksIGFueV1cbiAgPyBSXG4gIDogbmV2ZXI7XG5leHBvcnQgdHlwZSBOb3JtYWxpemVSZXR1cm5UeXBlPFQ+ID0gVCBleHRlbmRzICguLi5hcmdzOiBhbnkpID0+IGluZmVyIFJcbiAgPyBSXG4gIDogbmV2ZXI7XG5cbmV4cG9ydCB0eXBlIERlbm9ybWFsaXplPFM+ID0gUyBleHRlbmRzIEVudGl0eUludGVyZmFjZTxpbmZlciBVPlxuICA/IFVcbiAgOiBTIGV4dGVuZHMgUmVjb3JkQ2xhc3NcbiAgPyBBYnN0cmFjdEluc3RhbmNlVHlwZTxTPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IERlbm9ybWFsaXplUmV0dXJuVHlwZTxTWydkZW5vcm1hbGl6ZSddPlxuICA6IFMgZXh0ZW5kcyBTZXJpYWxpemFibGU8aW5mZXIgVD5cbiAgPyBUXG4gIDogUyBleHRlbmRzIEFycmF5PGluZmVyIEY+XG4gID8gRGVub3JtYWxpemU8Rj5bXVxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IERlbm9ybWFsaXplT2JqZWN0PFM+XG4gIDogUztcblxuZXhwb3J0IHR5cGUgRGVub3JtYWxpemVOdWxsYWJsZTxTPiA9IFMgZXh0ZW5kcyBFbnRpdHlJbnRlcmZhY2U8YW55PlxuICA/IERlbm9ybWFsaXplTnVsbGFibGVOZXN0ZWRTY2hlbWE8Uz4gfCB1bmRlZmluZWRcbiAgOiBTIGV4dGVuZHMgUmVjb3JkQ2xhc3NcbiAgPyBEZW5vcm1hbGl6ZU51bGxhYmxlTmVzdGVkU2NoZW1hPFM+XG4gIDogUyBleHRlbmRzIFNjaGVtYUNsYXNzXG4gID8gRGVub3JtYWxpemVSZXR1cm5UeXBlPFNbJ19kZW5vcm1hbGl6ZU51bGxhYmxlJ10+XG4gIDogUyBleHRlbmRzIFNlcmlhbGl6YWJsZTxpbmZlciBUPlxuICA/IFRcbiAgOiBTIGV4dGVuZHMgQXJyYXk8aW5mZXIgRj5cbiAgPyBEZW5vcm1hbGl6ZTxGPltdIHwgdW5kZWZpbmVkXG4gIDogUyBleHRlbmRzIHsgW0s6IHN0cmluZ106IGFueSB9XG4gID8gRGVub3JtYWxpemVOdWxsYWJsZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZTxTPiA9IFMgZXh0ZW5kcyBFbnRpdHlJbnRlcmZhY2VcbiAgPyBzdHJpbmdcbiAgOiBTIGV4dGVuZHMgUmVjb3JkQ2xhc3NcbiAgPyBOb3JtYWxpemVPYmplY3Q8U1snc2NoZW1hJ10+XG4gIDogUyBleHRlbmRzIFNjaGVtYUNsYXNzXG4gID8gTm9ybWFsaXplUmV0dXJuVHlwZTxTWydub3JtYWxpemUnXT5cbiAgOiBTIGV4dGVuZHMgU2VyaWFsaXphYmxlPGluZmVyIFQ+XG4gID8gVFxuICA6IFMgZXh0ZW5kcyBBcnJheTxpbmZlciBGPlxuICA/IE5vcm1hbGl6ZTxGPltdXG4gIDogUyBleHRlbmRzIHsgW0s6IHN0cmluZ106IGFueSB9XG4gID8gTm9ybWFsaXplT2JqZWN0PFM+XG4gIDogUztcblxuZXhwb3J0IHR5cGUgTm9ybWFsaXplTnVsbGFibGU8Uz4gPSBTIGV4dGVuZHMgRW50aXR5SW50ZXJmYWNlXG4gID8gc3RyaW5nIHwgdW5kZWZpbmVkXG4gIDogUyBleHRlbmRzIFJlY29yZENsYXNzXG4gID8gTm9ybWFsaXplZE51bGxhYmxlT2JqZWN0PFNbJ3NjaGVtYSddPlxuICA6IFMgZXh0ZW5kcyBTY2hlbWFDbGFzc1xuICA/IE5vcm1hbGl6ZVJldHVyblR5cGU8U1snX25vcm1hbGl6ZU51bGxhYmxlJ10+XG4gIDogUyBleHRlbmRzIFNlcmlhbGl6YWJsZTxpbmZlciBUPlxuICA/IFRcbiAgOiBTIGV4dGVuZHMgQXJyYXk8aW5mZXIgRj5cbiAgPyBOb3JtYWxpemU8Rj5bXSB8IHVuZGVmaW5lZFxuICA6IFMgZXh0ZW5kcyB7IFtLOiBzdHJpbmddOiBhbnkgfVxuICA/IE5vcm1hbGl6ZWROdWxsYWJsZU9iamVjdDxTPlxuICA6IFM7XG5cbmV4cG9ydCB0eXBlIE5vcm1hbGl6ZWRTY2hlbWE8RSwgUj4gPSB7XG4gIGVudGl0aWVzOiBFO1xuICByZXN1bHQ6IFI7XG4gIGluZGV4ZXM6IE5vcm1hbGl6ZWRJbmRleDtcbiAgZW50aXR5TWV0YToge1xuICAgIHJlYWRvbmx5IFtlbnRpdHlLZXk6IHN0cmluZ106IHtcbiAgICAgIHJlYWRvbmx5IFtwazogc3RyaW5nXToge1xuICAgICAgICByZWFkb25seSBkYXRlOiBudW1iZXI7XG4gICAgICAgIHJlYWRvbmx5IGV4cGlyZXNBdDogbnVtYmVyO1xuICAgICAgICByZWFkb25seSBmZXRjaGVkQXQ6IG51bWJlcjtcbiAgICAgIH07XG4gICAgfTtcbiAgfTtcbn07XG5cbmV4cG9ydCB0eXBlIEVudGl0eU1hcDxUID0gYW55PiA9IFJlY29yZDxzdHJpbmcsIEVudGl0eUludGVyZmFjZTxUPj47XG4iXSwibWFwcGluZ3MiOiIifQ== |
{ | ||
"name": "@rest-hooks/normalizr", | ||
"version": "9.5.1", | ||
"version": "10.0.0", | ||
"description": "Normalizes and denormalizes JSON according to schema for Redux and Flux applications", | ||
@@ -118,3 +118,3 @@ "homepage": "https://resthooks.io/docs/concepts/normalization", | ||
}, | ||
"gitHead": "8d3be93d44408c4b2b565486df25a30df543f21f" | ||
"gitHead": "8b0eda4d5e129ba0a02d704a65cdfeb402bafc4f" | ||
} |
import { Schema } from './interface.js'; | ||
import { Denormalize, DenormalizeNullable, DenormalizeCache } from './types.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
import { Denormalize, DenormalizeNullable, DenormalizeCache, Path } from './types.js'; | ||
type DenormalizeReturn<S extends Schema> = [ | ||
@@ -8,3 +7,3 @@ /*denormalized*/ Denormalize<S>, | ||
/*deleted*/ false, | ||
/*resolvedEntities*/ Record<string, Record<string, any>> | ||
/*entityPaths*/ Path[] | ||
] | [ | ||
@@ -14,3 +13,3 @@ /*denormalized*/ DenormalizeNullable<S>, | ||
/*deleted*/ true, | ||
/*resolvedEntities*/ Record<string, Record<string, any>> | ||
/*entityPaths*/ Path[] | ||
] | [ | ||
@@ -20,6 +19,6 @@ /*denormalized*/ DenormalizeNullable<S>, | ||
/*deleted*/ boolean, | ||
/*resolvedEntities*/ Record<string, Record<string, any>> | ||
/*entityPaths*/ Path[] | ||
]; | ||
export declare const denormalize: <S extends Schema>(input: unknown, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: WeakListMap<object, any>) => DenormalizeReturn<S>; | ||
export declare const denormalizeSimple: <S extends Schema>(input: any, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: WeakListMap<object, any>) => [ | ||
export declare const denormalize: <S extends Schema>(input: unknown, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: DenormalizeCache['results'][string]) => DenormalizeReturn<S>; | ||
export declare const denormalizeSimple: <S extends Schema>(input: any, schema: S | undefined, entities: any, entityCache?: DenormalizeCache['entities'], resultCache?: DenormalizeCache['results'][string]) => [ | ||
/*denormalized*/ Denormalize<S>, | ||
@@ -26,0 +25,0 @@ /*found*/ true, |
import { denormalize } from './denormalize.js'; | ||
import { isEntity } from './isEntity.js'; | ||
import { normalize } from './normalize.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
import WeakEntityMap from './WeakEntityMap.js'; | ||
export { default as inferResults } from './inferResults.js'; | ||
export { DELETED } from './special.js'; | ||
export { AbstractInstanceType, NormalizeReturnType, NormalizedSchema, DenormalizeReturnType, DenormalizeCache, } from './types.js'; | ||
export { AbstractInstanceType, NormalizeReturnType, NormalizedSchema, DenormalizeReturnType, DenormalizeCache, Path, } from './types.js'; | ||
export * from './endpoint/types.js'; | ||
@@ -12,3 +12,3 @@ export * from './interface.js'; | ||
export * from './normal.js'; | ||
export { denormalize, normalize, isEntity, WeakListMap }; | ||
export { denormalize, normalize, isEntity, WeakEntityMap }; | ||
//# sourceMappingURL=index.d.ts.map |
import { Schema, Serializable, EntityInterface, NormalizedIndex, SchemaClass } from './interface.js'; | ||
import WeakListMap from './WeakListMap.js'; | ||
import WeakEntityMap from './WeakEntityMap.js'; | ||
export interface Path { | ||
key: string; | ||
pk: string; | ||
} | ||
export type AbstractInstanceType<T> = T extends { | ||
@@ -36,7 +40,7 @@ prototype: infer U; | ||
[key: string]: { | ||
[pk: string]: WeakListMap<object, EntityInterface>; | ||
[pk: string]: WeakMap<EntityInterface, WeakEntityMap<object, any>>; | ||
}; | ||
}; | ||
results: { | ||
[key: string]: WeakListMap<object, any>; | ||
[key: string]: WeakEntityMap<object, any>; | ||
}; | ||
@@ -43,0 +47,0 @@ } |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
404816
4514