@data-client/endpoint
Version:
Declarative Network Interface Definitions
214 lines (204 loc) • 33.8 kB
JavaScript
import { consistentSerialize } from './consistentSerialize.js';
import { Values, Array as ArraySchema } from '../schema.js';
const pushMerge = (existing, incoming) => {
return [...existing, ...incoming];
};
const unshiftMerge = (existing, incoming) => {
return [...incoming, ...existing];
};
const valuesMerge = (existing, incoming) => {
return {
...existing,
...incoming
};
};
const createArray = value => [...value];
const createValue = value => ({
...value
});
/**
* Entities but for Arrays instead of classes
* @see https://dataclient.io/rest/api/Collection
*/
export default class CollectionSchema {
addWith(merge, createCollectionFilter) {
return CreateAdder(this, merge, createCollectionFilter);
}
// this adds to any list *in store* that has same members as the urlParams
// so fetch(create, { userId: 'bob', completed: true }, data)
// would possibly add to {}, {userId: 'bob'}, {completed: true}, {userId: 'bob', completed: true } - but only those already in the store
// it ignores keys that start with sort as those are presumed to not filter results
createCollectionFilter(...args) {
return collectionKey => Object.entries(collectionKey).every(([key, value]) => {
var _args$;
return this.nonFilterArgumentKeys(key) ||
// strings are canonical form. See pk() above for value transformation
`${args[0][key]}` === value || `${(_args$ = args[1]) == null ? void 0 : _args$[key]}` === value;
});
}
nonFilterArgumentKeys(key) {
return key.startsWith('order');
}
constructor(schema, options) {
this.schema = Array.isArray(schema) ? new ArraySchema(schema[0]) : schema;
if (!options) {
this.argsKey = params => ({
...params
});
} else {
if ('nestKey' in options) {
this.nestKey = options.nestKey;
} else if ('argsKey' in options) {
this.argsKey = options.argsKey;
} else {
this.argsKey = params => ({
...params
});
}
}
this.key = keyFromSchema(this.schema);
if (options != null && options.nonFilterArgumentKeys) {
const {
nonFilterArgumentKeys
} = options;
if (typeof nonFilterArgumentKeys === 'function') {
this.nonFilterArgumentKeys = nonFilterArgumentKeys;
} else if (nonFilterArgumentKeys instanceof RegExp) {
this.nonFilterArgumentKeys = key => nonFilterArgumentKeys.test(key);
} else {
this.nonFilterArgumentKeys = key => nonFilterArgumentKeys.includes(key);
}
} else if (options != null && options.createCollectionFilter)
// TODO(breaking): rename to filterCollections
this.createCollectionFilter = options.createCollectionFilter.bind(this);
// >>>>>>>>>>>>>>CREATION<<<<<<<<<<<<<<
if (this.schema instanceof ArraySchema) {
this.createIfValid = createArray;
this.push = CreateAdder(this, pushMerge);
this.unshift = CreateAdder(this, unshiftMerge);
} else if (schema instanceof Values) {
this.createIfValid = createValue;
this.assign = CreateAdder(this, valuesMerge);
}
}
get cacheWith() {
return this.schema.schema;
}
toString() {
return this.key;
}
toJSON() {
return {
key: this.key,
schema: this.schema.schema.toJSON()
};
}
pk(value, parent, key, args) {
const obj = this.argsKey ? this.argsKey(...args) : this.nestKey(parent, key);
for (const key in obj) {
if (['number', 'boolean'].includes(typeof obj[key])) obj[key] = `${obj[key]}`;
}
return consistentSerialize(obj);
}
// >>>>>>>>>>>>>>NORMALIZE<<<<<<<<<<<<<<
normalize(input, parent, key, args, visit, addEntity, getEntity, checkLoop) {
const normalizedValue = this.schema.normalize(input, parent, key, args, visit, addEntity, getEntity, checkLoop);
const id = this.pk(normalizedValue, parent, key, args);
addEntity(this, normalizedValue, id);
return id;
}
// always replace
merge(existing, incoming) {
return incoming;
}
shouldReorder(existingMeta, incomingMeta, existing, incoming) {
return incomingMeta.fetchedAt < existingMeta.fetchedAt;
}
mergeWithStore(existingMeta, incomingMeta, existing, incoming) {
return this.shouldReorder(existingMeta, incomingMeta, existing, incoming) ? this.merge(incoming, existing) : this.merge(existing, incoming);
}
mergeMetaWithStore(existingMeta, incomingMeta, existing, incoming) {
return this.shouldReorder(existingMeta, incomingMeta, existing, incoming) ? existingMeta : incomingMeta;
}
// >>>>>>>>>>>>>>DENORMALIZE<<<<<<<<<<<<<<
queryKey(args, queryKey, getEntity, getIndex) {
if (this.argsKey) {
const id = this.pk(undefined, undefined, '', args);
// ensure this actually has entity or we shouldn't try to use it in our query
if (getEntity(this.key, id)) return id;
}
}
denormalize(input, args, unvisit) {
return this.schema.denormalize(input, args, unvisit);
}
}
function CreateAdder(collection, merge, createCollectionFilter) {
const properties = {
merge: {
value: merge
},
normalize: {
value: normalizeCreate
},
queryKey: {
value: queryKeyCreate
}
};
if (collection.schema instanceof ArraySchema) {
properties.createIfValid = {
value: createIfValid
};
properties.denormalize = {
value: denormalize
};
}
if (createCollectionFilter) {
properties.createCollectionFilter = {
value: createCollectionFilter
};
}
return Object.create(collection, properties);
}
function queryKeyCreate() {}
function normalizeCreate(input, parent, key, args, visit, addEntity, getEntity, checkLoop) {
if (process.env.NODE_ENV !== 'production') {
// means 'this is a creation endpoint' - so real PKs are not required
// this is used by Entity.normalize() to determine whether to allow empty pks
// visit instances are created on each normalize call so this will safely be reset
visit.creating = true;
}
const normalizedValue = this.schema.normalize(!(this.schema instanceof ArraySchema) || Array.isArray(input) ? input : [input], parent, key, args, visit, addEntity, getEntity, checkLoop);
// parent is args when not nested
const filterCollections = this.createCollectionFilter(...args);
// add to any collections that match this
const entities = getEntity(this.key);
if (entities) Object.keys(entities).forEach(collectionPk => {
if (!filterCollections(JSON.parse(collectionPk))) return;
addEntity(this, normalizedValue, collectionPk);
});
return normalizedValue;
}
function createIfValid(value) {
return Array.isArray(value) ? [...value] : {
...value
};
}
// only for arrays
function denormalize(input, args, unvisit) {
return Array.isArray(input) ? this.schema.denormalize(input, args, unvisit) : this.schema.denormalize([input], args, unvisit)[0];
}
/**
* We call schema.denormalize and schema.normalize directly
* instead of visit/unvisit as we are not operating on new data
* so the additional checks in those methods are redundant
*/
function keyFromSchema(schema) {
if (schema instanceof ArraySchema) {
// this assumes the definition of Array/Values is Entity
return `[${schema.schemaKey()}]`;
} else if (schema instanceof Values) {
return `{${schema.schemaKey()}}`;
}
return `(${schema.schemaKey()})`;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["consistentSerialize","Values","Array","ArraySchema","pushMerge","existing","incoming","unshiftMerge","valuesMerge","createArray","value","createValue","CollectionSchema","addWith","merge","createCollectionFilter","CreateAdder","args","collectionKey","Object","entries","every","key","_args$","nonFilterArgumentKeys","startsWith","constructor","schema","options","isArray","argsKey","params","nestKey","keyFromSchema","RegExp","test","includes","bind","createIfValid","push","unshift","assign","cacheWith","toString","toJSON","pk","parent","obj","normalize","input","visit","addEntity","getEntity","checkLoop","normalizedValue","id","shouldReorder","existingMeta","incomingMeta","fetchedAt","mergeWithStore","mergeMetaWithStore","queryKey","getIndex","undefined","denormalize","unvisit","collection","properties","normalizeCreate","queryKeyCreate","create","process","env","NODE_ENV","creating","filterCollections","entities","keys","forEach","collectionPk","JSON","parse","schemaKey"],"sources":["../../src/schemas/Collection.ts"],"sourcesContent":["import { consistentSerialize } from './consistentSerialize.js';\nimport { CheckLoop, GetEntity, PolymorphicInterface } from '../interface.js';\nimport { Values, Array as ArraySchema } from '../schema.js';\nimport type { DefaultArgs, EntityInterface } from '../schemaTypes.js';\n\nconst pushMerge = (existing: any, incoming: any) => {\n  return [...existing, ...incoming];\n};\nconst unshiftMerge = (existing: any, incoming: any) => {\n  return [...incoming, ...existing];\n};\nconst valuesMerge = (existing: any, incoming: any) => {\n  return { ...existing, ...incoming };\n};\nconst createArray = (value: any) => [...value];\nconst createValue = (value: any) => ({ ...value });\n\n/**\n * Entities but for Arrays instead of classes\n * @see https://dataclient.io/rest/api/Collection\n */\nexport default class CollectionSchema<\n  S extends PolymorphicInterface = any,\n  Args extends any[] = DefaultArgs,\n  Parent = any,\n> {\n  declare protected nestKey: (parent: any, key: string) => Record<string, any>;\n\n  declare protected argsKey?: (...args: any) => Record<string, any>;\n\n  declare readonly schema: S;\n\n  declare readonly key: string;\n\n  declare push: S extends ArraySchema<any> ? CollectionSchema<S, Args, Parent>\n  : undefined;\n\n  declare unshift: S extends ArraySchema<any> ?\n    CollectionSchema<S, Args, Parent>\n  : undefined;\n\n  declare assign: S extends Values<any> ? CollectionSchema<S, Args, Parent>\n  : undefined;\n\n  addWith<P extends any[] = Args>(\n    merge: (existing: any, incoming: any) => any,\n    createCollectionFilter?: (\n      ...args: P\n    ) => (collectionKey: Record<string, string>) => boolean,\n  ): CollectionSchema<S, P> {\n    return CreateAdder(this, merge, createCollectionFilter);\n  }\n\n  // this adds to any list *in store* that has same members as the urlParams\n  // so fetch(create, { userId: 'bob', completed: true }, data)\n  // would possibly add to {}, {userId: 'bob'}, {completed: true}, {userId: 'bob', completed: true } - but only those already in the store\n  // it ignores keys that start with sort as those are presumed to not filter results\n  protected createCollectionFilter(...args: Args) {\n    return (collectionKey: Record<string, string>) =>\n      Object.entries(collectionKey).every(\n        ([key, value]) =>\n          this.nonFilterArgumentKeys(key) ||\n          // strings are canonical form. See pk() above for value transformation\n          `${args[0][key]}` === value ||\n          `${args[1]?.[key]}` === value,\n      );\n  }\n\n  protected nonFilterArgumentKeys(key: string) {\n    return key.startsWith('order');\n  }\n\n  constructor(schema: S, options?: CollectionOptions<Args, Parent>) {\n    this.schema =\n      Array.isArray(schema) ? (new ArraySchema(schema[0]) as any) : schema;\n    if (!options) {\n      this.argsKey = params => ({ ...params });\n    } else {\n      if ('nestKey' in options) {\n        (this as any).nestKey = options.nestKey;\n      } else if ('argsKey' in options) {\n        this.argsKey = options.argsKey;\n      } else {\n        this.argsKey = params => ({ ...params });\n      }\n    }\n    this.key = keyFromSchema(this.schema);\n    if ((options as any)?.nonFilterArgumentKeys) {\n      const { nonFilterArgumentKeys } = options as {\n        nonFilterArgumentKeys: ((key: string) => boolean) | string[] | RegExp;\n      };\n      if (typeof nonFilterArgumentKeys === 'function') {\n        this.nonFilterArgumentKeys = nonFilterArgumentKeys;\n      } else if (nonFilterArgumentKeys instanceof RegExp) {\n        this.nonFilterArgumentKeys = key => nonFilterArgumentKeys.test(key);\n      } else {\n        this.nonFilterArgumentKeys = key => nonFilterArgumentKeys.includes(key);\n      }\n    } else if ((options as any)?.createCollectionFilter)\n      // TODO(breaking): rename to filterCollections\n      this.createCollectionFilter = (\n        options as any as {\n          createCollectionFilter: (\n            ...args: Args\n          ) => (collectionKey: Record<string, string>) => boolean;\n        }\n      ).createCollectionFilter.bind(this) as any;\n\n    // >>>>>>>>>>>>>>CREATION<<<<<<<<<<<<<<\n    if (this.schema instanceof ArraySchema) {\n      this.createIfValid = createArray;\n      this.push = CreateAdder(this, pushMerge);\n      this.unshift = CreateAdder(this, unshiftMerge);\n    } else if (schema instanceof Values) {\n      this.createIfValid = createValue;\n      this.assign = CreateAdder(this, valuesMerge);\n    }\n  }\n\n  get cacheWith(): object {\n    return this.schema.schema;\n  }\n\n  toString() {\n    return this.key;\n  }\n\n  toJSON() {\n    return {\n      key: this.key,\n      schema: this.schema.schema.toJSON(),\n    };\n  }\n\n  pk(value: any, parent: any, key: string, args: readonly any[]) {\n    const obj =\n      this.argsKey ? this.argsKey(...args) : this.nestKey(parent, key);\n    for (const key in obj) {\n      if (['number', 'boolean'].includes(typeof obj[key]))\n        obj[key] = `${obj[key]}`;\n    }\n    return consistentSerialize(obj);\n  }\n\n  // >>>>>>>>>>>>>>NORMALIZE<<<<<<<<<<<<<<\n\n  normalize(\n    input: any,\n    parent: Parent,\n    key: string,\n    args: any[],\n    visit: (...args: any) => any,\n    addEntity: (...args: any) => any,\n    getEntity: any,\n    checkLoop: any,\n  ): string {\n    const normalizedValue = this.schema.normalize(\n      input,\n      parent,\n      key,\n      args,\n      visit,\n      addEntity,\n      getEntity,\n      checkLoop,\n    );\n    const id = this.pk(normalizedValue, parent, key, args);\n\n    addEntity(this, normalizedValue, id);\n    return id;\n  }\n\n  // always replace\n  merge(existing: any, incoming: any) {\n    return incoming;\n  }\n\n  shouldReorder(\n    existingMeta: { date: number; fetchedAt: number },\n    incomingMeta: { date: number; fetchedAt: number },\n    existing: any,\n    incoming: any,\n  ) {\n    return incomingMeta.fetchedAt < existingMeta.fetchedAt;\n  }\n\n  mergeWithStore(\n    existingMeta: {\n      date: number;\n      fetchedAt: number;\n    },\n    incomingMeta: { date: number; fetchedAt: number },\n    existing: any,\n    incoming: any,\n  ) {\n    return this.shouldReorder(existingMeta, incomingMeta, existing, incoming) ?\n        this.merge(incoming, existing)\n      : this.merge(existing, incoming);\n  }\n\n  mergeMetaWithStore(\n    existingMeta: {\n      fetchedAt: number;\n      date: number;\n      expiresAt: number;\n    },\n    incomingMeta: { fetchedAt: number; date: number; expiresAt: number },\n    existing: any,\n    incoming: any,\n  ) {\n    return this.shouldReorder(existingMeta, incomingMeta, existing, incoming) ?\n        existingMeta\n      : incomingMeta;\n  }\n\n  // >>>>>>>>>>>>>>DENORMALIZE<<<<<<<<<<<<<<\n\n  queryKey(\n    args: Args,\n    queryKey: unknown,\n    getEntity: GetEntity,\n    getIndex: unknown,\n  ): any {\n    if (this.argsKey) {\n      const id = this.pk(undefined, undefined, '', args);\n      // ensure this actually has entity or we shouldn't try to use it in our query\n      if (getEntity(this.key, id)) return id;\n    }\n  }\n\n  declare createIfValid: (value: any) => any | undefined;\n\n  denormalize(\n    input: any,\n    args: readonly any[],\n    unvisit: (schema: any, input: any) => any,\n  ): ReturnType<S['denormalize']> {\n    return this.schema.denormalize(input, args, unvisit) as any;\n  }\n}\n\nexport type CollectionOptions<\n  Args extends any[] = DefaultArgs,\n  Parent = any,\n> = (\n  | {\n      /** Defines lookups for Collections nested in other schemas.\n       *\n       * @see https://dataclient.io/rest/api/Collection#nestKey\n       */\n      nestKey?: (parent: Parent, key: string) => Record<string, any>;\n    }\n  | {\n      /** Defines lookups top-level Collections using ...args.\n       *\n       * @see https://dataclient.io/rest/api/Collection#argsKey\n       */\n      argsKey?: (...args: Args) => Record<string, any>;\n    }\n) &\n  (\n    | {\n        /** Sets a default createCollectionFilter for addWith(), push, unshift, and assign.\n         *\n         * @see https://dataclient.io/rest/api/Collection#createcollectionfilter\n         */\n        createCollectionFilter?: (\n          ...args: Args\n        ) => (collectionKey: Record<string, string>) => boolean;\n      }\n    | {\n        /** Test to determine which arg keys should **not** be used for filtering results.\n         *\n         * @see https://dataclient.io/rest/api/Collection#nonfilterargumentkeys\n         */\n        nonFilterArgumentKeys?: ((key: string) => boolean) | string[] | RegExp;\n      }\n  );\n\nfunction CreateAdder<C extends CollectionSchema<any, any>, P extends any[]>(\n  collection: C,\n  merge: (existing: any, incoming: any) => any[],\n  createCollectionFilter?: (\n    ...args: P\n  ) => (collectionKey: Record<string, string>) => boolean,\n) {\n  const properties: PropertyDescriptorMap = {\n    merge: { value: merge },\n    normalize: { value: normalizeCreate },\n    queryKey: { value: queryKeyCreate },\n  };\n  if (collection.schema instanceof ArraySchema) {\n    properties.createIfValid = { value: createIfValid };\n    properties.denormalize = { value: denormalize };\n  }\n  if (createCollectionFilter) {\n    properties.createCollectionFilter = { value: createCollectionFilter };\n  }\n  return Object.create(collection, properties);\n}\n\nfunction queryKeyCreate() {}\n\nfunction normalizeCreate(\n  this: CollectionSchema<any, any>,\n  input: any,\n  parent: any,\n  key: string,\n  args: readonly any[],\n  visit: ((...args: any) => any) & { creating?: boolean },\n  addEntity: (schema: any, processedEntity: any, id: string) => void,\n  getEntity: GetEntity,\n  checkLoop: CheckLoop,\n): any {\n  if (process.env.NODE_ENV !== 'production') {\n    // means 'this is a creation endpoint' - so real PKs are not required\n    // this is used by Entity.normalize() to determine whether to allow empty pks\n    // visit instances are created on each normalize call so this will safely be reset\n    visit.creating = true;\n  }\n  const normalizedValue = this.schema.normalize(\n    !(this.schema instanceof ArraySchema) || Array.isArray(input) ?\n      input\n    : [input],\n    parent,\n    key,\n    args,\n    visit,\n    addEntity,\n    getEntity,\n    checkLoop,\n  );\n  // parent is args when not nested\n  const filterCollections = (this.createCollectionFilter as any)(...args);\n  // add to any collections that match this\n  const entities = getEntity(this.key);\n  if (entities)\n    Object.keys(entities).forEach(collectionPk => {\n      if (!filterCollections(JSON.parse(collectionPk))) return;\n      addEntity(this, normalizedValue, collectionPk);\n    });\n  return normalizedValue as any;\n}\n\nfunction createIfValid(value: object): any | undefined {\n  return Array.isArray(value) ? [...value] : { ...value };\n}\n\n// only for arrays\nfunction denormalize(\n  this: CollectionSchema<any, any>,\n  input: any,\n  args: readonly any[],\n  unvisit: (schema: any, input: any) => any,\n): any {\n  return Array.isArray(input) ?\n      (this.schema.denormalize(input, args, unvisit) as any)\n    : (this.schema.denormalize([input], args, unvisit)[0] as any);\n}\n/**\n * We call schema.denormalize and schema.normalize directly\n * instead of visit/unvisit as we are not operating on new data\n * so the additional checks in those methods are redundant\n */\n\nfunction keyFromSchema(schema: PolymorphicInterface) {\n  if (schema instanceof ArraySchema) {\n    // this assumes the definition of Array/Values is Entity\n    return `[${schema.schemaKey()}]`;\n  } else if (schema instanceof Values) {\n    return `{${schema.schemaKey()}}`;\n  }\n  return `(${schema.schemaKey()})`;\n}\n"],"mappings":"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","ignoreList":[]}