UNPKG

@data-client/endpoint

Version:

Declarative Network Interface Definitions

204 lines (194 loc) 33.9 kB
import _extends from "@babel/runtime/helpers/esm/extends"; 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 _extends({}, existing, incoming); }; const createArray = value => [...value]; const createValue = value => _extends({}, 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 => _extends({}, params); } else { if ('nestKey' in options) { this.nestKey = options.nestKey; } else if ('argsKey' in options) { this.argsKey = options.argsKey; } else { this.argsKey = params => _extends({}, 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] : _extends({}, 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","_extends","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":[]}