@data-client/normalizr
Version:
Normalizes and denormalizes JSON according to schema for Redux and Flux applications
74 lines (71 loc) • 11.1 kB
JavaScript
import _extends from "@babel/runtime/helpers/esm/extends";
import { addEntities } from './addEntities.js';
import { getVisit } from './getVisit.js';
import { createGetEntity } from '../memo/MemoCache.js';
export const normalize = (schema, input, args = [], {
entities,
indexes,
entityMeta
} = emptyStore, meta = {
fetchedAt: 0,
date: Date.now(),
expiresAt: Infinity
}) => {
// no schema means we don't process at all
if (schema === undefined || schema === null) return {
result: input,
entities,
indexes,
entityMeta
};
const schemaType = expectedSchemaType(schema);
if (input === null || typeof input !== schemaType &&
// we will allow a Delete schema to be a string or object
!(schema.key !== undefined && schema.pk === undefined && typeof input === 'string')) {
/* istanbul ignore else */
if (process.env.NODE_ENV !== 'production') {
const parseWorks = input => {
try {
return typeof JSON.parse(input) !== 'string';
} catch (e) {
return false;
}
};
if (typeof input === 'string' && parseWorks(input)) {
throw new Error(`Normalizing a string, but this does match schema.
Parsing this input string as JSON worked. This likely indicates fetch function did not parse
the JSON. By default, this only happens if "content-type" header includes "json".
See https://dataclient.io/rest/api/RestEndpoint#parseResponse for more information
Schema: ${JSON.stringify(schema, undefined, 2)}
Input: "${input}"`);
} else {
throw new Error(`Unexpected input given to normalize. Expected type to be "${schemaType}", found "${input === null ? 'null' : typeof input}".
Schema: ${JSON.stringify(schema, undefined, 2)}
Input: "${input}"`);
}
} else {
throw new Error(`Unexpected input given to normalize. Expected type to be "${schemaType}", found "${input === null ? 'null' : typeof input}".`);
}
}
const newEntities = new Map();
const newIndexes = new Map();
const ret = {
result: '',
entities: _extends({}, entities),
indexes: _extends({}, indexes),
entityMeta: _extends({}, entityMeta)
};
const addEntity = addEntities(newEntities, newIndexes, ret.entities, ret.indexes, ret.entityMeta, meta);
const visit = getVisit(addEntity, createGetEntity(entities));
ret.result = visit(schema, input, input, undefined, args);
return ret;
};
function expectedSchemaType(schema) {
return ['object', 'function'].includes(typeof schema) ? 'object' : typeof schema;
}
const emptyStore = {
entities: {},
indexes: {},
entityMeta: {}
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["addEntities","getVisit","createGetEntity","normalize","schema","input","args","entities","indexes","entityMeta","emptyStore","meta","fetchedAt","date","Date","now","expiresAt","Infinity","undefined","result","schemaType","expectedSchemaType","key","pk","process","env","NODE_ENV","parseWorks","JSON","parse","e","Error","stringify","newEntities","Map","newIndexes","ret","_extends","addEntity","visit","includes"],"sources":["../../src/normalizr/normalize.ts"],"sourcesContent":["import { addEntities } from './addEntities.js';\nimport { getVisit } from './getVisit.js';\nimport type { Schema, NormalizedIndex } from '../interface.js';\nimport { createGetEntity } from '../memo/MemoCache.js';\nimport type {\n  NormalizeMeta,\n  NormalizeNullable,\n  NormalizedSchema,\n  StoreData,\n} from '../types.js';\n\nexport const normalize = <\n  S extends Schema = Schema,\n  E extends Record<string, Record<string, any> | undefined> = Record<\n    string,\n    Record<string, any>\n  >,\n  R = NormalizeNullable<S>,\n>(\n  schema: S | undefined,\n  input: any,\n  args: readonly any[] = [],\n  { entities, indexes, entityMeta }: StoreData<E> = emptyStore,\n  meta: NormalizeMeta = { fetchedAt: 0, date: Date.now(), expiresAt: Infinity },\n): NormalizedSchema<E, R> => {\n  // no schema means we don't process at all\n  if (schema === undefined || schema === null)\n    return {\n      result: input,\n      entities,\n      indexes,\n      entityMeta,\n    };\n\n  const schemaType = expectedSchemaType(schema);\n  if (\n    input === null ||\n    (typeof input !== schemaType &&\n      // we will allow a Delete schema to be a string or object\n      !(\n        (schema as any).key !== undefined &&\n        (schema as any).pk === undefined &&\n        typeof input === 'string'\n      ))\n  ) {\n    /* istanbul ignore else */\n    if (process.env.NODE_ENV !== 'production') {\n      const parseWorks = (input: string) => {\n        try {\n          return typeof JSON.parse(input) !== 'string';\n        } catch (e) {\n          return false;\n        }\n      };\n      if (typeof input === 'string' && parseWorks(input)) {\n        throw 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://dataclient.io/rest/api/RestEndpoint#parseResponse for more information\n\n  Schema: ${JSON.stringify(schema, undefined, 2)}\n  Input: \"${input}\"`);\n      } else {\n        throw new Error(\n          `Unexpected input given to normalize. Expected type to be \"${schemaType}\", found \"${\n            input === null ? 'null' : typeof input\n          }\".\n\n          Schema: ${JSON.stringify(schema, undefined, 2)}\n          Input: \"${input}\"`,\n        );\n      }\n    } else {\n      throw new Error(\n        `Unexpected input given to normalize. Expected type to be \"${schemaType}\", found \"${\n          input === null ? 'null' : typeof input\n        }\".`,\n      );\n    }\n  }\n\n  const newEntities = new Map<string, Map<string, any>>();\n  const newIndexes = new Map<string, Map<string, any>>();\n  const ret: NormalizedSchema<E, R> = {\n    result: '' as any,\n    entities: { ...entities },\n    indexes: { ...indexes },\n    entityMeta: { ...entityMeta },\n  };\n  const addEntity = addEntities(\n    newEntities,\n    newIndexes,\n    ret.entities,\n    ret.indexes,\n    ret.entityMeta,\n    meta,\n  );\n\n  const visit = getVisit(addEntity, createGetEntity(entities));\n  ret.result = visit(schema, input, input, undefined, args);\n  return ret;\n};\n\nfunction expectedSchemaType(schema: Schema) {\n  return ['object', 'function'].includes(typeof schema) ? 'object' : (\n      typeof schema\n    );\n}\n\nconst emptyStore: StoreData<any> = {\n  entities: {},\n  indexes: {},\n  entityMeta: {},\n};\n"],"mappings":";AAAA,SAASA,WAAW,QAAQ,kBAAkB;AAC9C,SAASC,QAAQ,QAAQ,eAAe;AAExC,SAASC,eAAe,QAAQ,sBAAsB;AAQtD,OAAO,MAAMC,SAAS,GAAGA,CAQvBC,MAAqB,EACrBC,KAAU,EACVC,IAAoB,GAAG,EAAE,EACzB;EAAEC,QAAQ;EAAEC,OAAO;EAAEC;AAAyB,CAAC,GAAGC,UAAU,EAC5DC,IAAmB,GAAG;EAAEC,SAAS,EAAE,CAAC;EAAEC,IAAI,EAAEC,IAAI,CAACC,GAAG,CAAC,CAAC;EAAEC,SAAS,EAAEC;AAAS,CAAC,KAClD;EAC3B;EACA,IAAIb,MAAM,KAAKc,SAAS,IAAId,MAAM,KAAK,IAAI,EACzC,OAAO;IACLe,MAAM,EAAEd,KAAK;IACbE,QAAQ;IACRC,OAAO;IACPC;EACF,CAAC;EAEH,MAAMW,UAAU,GAAGC,kBAAkB,CAACjB,MAAM,CAAC;EAC7C,IACEC,KAAK,KAAK,IAAI,IACb,OAAOA,KAAK,KAAKe,UAAU;EAC1B;EACA,EACGhB,MAAM,CAASkB,GAAG,KAAKJ,SAAS,IAChCd,MAAM,CAASmB,EAAE,KAAKL,SAAS,IAChC,OAAOb,KAAK,KAAK,QAAQ,CACzB,EACJ;IACA;IACA,IAAImB,OAAO,CAACC,GAAG,CAACC,QAAQ,KAAK,YAAY,EAAE;MACzC,MAAMC,UAAU,GAAItB,KAAa,IAAK;QACpC,IAAI;UACF,OAAO,OAAOuB,IAAI,CAACC,KAAK,CAACxB,KAAK,CAAC,KAAK,QAAQ;QAC9C,CAAC,CAAC,OAAOyB,CAAC,EAAE;UACV,OAAO,KAAK;QACd;MACF,CAAC;MACD,IAAI,OAAOzB,KAAK,KAAK,QAAQ,IAAIsB,UAAU,CAACtB,KAAK,CAAC,EAAE;QAClD,MAAM,IAAI0B,KAAK,CAAC;AACxB;AACA;AACA;AACA;AACA;AACA,YAAYH,IAAI,CAACI,SAAS,CAAC5B,MAAM,EAAEc,SAAS,EAAE,CAAC,CAAC;AAChD,YAAYb,KAAK,GAAG,CAAC;MACf,CAAC,MAAM;QACL,MAAM,IAAI0B,KAAK,CACb,6DAA6DX,UAAU,aACrEf,KAAK,KAAK,IAAI,GAAG,MAAM,GAAG,OAAOA,KAAK;AAClD;AACA,oBACoBuB,IAAI,CAACI,SAAS,CAAC5B,MAAM,EAAEc,SAAS,EAAE,CAAC,CAAC;AACxD,oBAAoBb,KAAK,GACjB,CAAC;MACH;IACF,CAAC,MAAM;MACL,MAAM,IAAI0B,KAAK,CACb,6DAA6DX,UAAU,aACrEf,KAAK,KAAK,IAAI,GAAG,MAAM,GAAG,OAAOA,KAAK,IAE1C,CAAC;IACH;EACF;EAEA,MAAM4B,WAAW,GAAG,IAAIC,GAAG,CAA2B,CAAC;EACvD,MAAMC,UAAU,GAAG,IAAID,GAAG,CAA2B,CAAC;EACtD,MAAME,GAA2B,GAAG;IAClCjB,MAAM,EAAE,EAAS;IACjBZ,QAAQ,EAAA8B,QAAA,KAAO9B,QAAQ,CAAE;IACzBC,OAAO,EAAA6B,QAAA,KAAO7B,OAAO,CAAE;IACvBC,UAAU,EAAA4B,QAAA,KAAO5B,UAAU;EAC7B,CAAC;EACD,MAAM6B,SAAS,GAAGtC,WAAW,CAC3BiC,WAAW,EACXE,UAAU,EACVC,GAAG,CAAC7B,QAAQ,EACZ6B,GAAG,CAAC5B,OAAO,EACX4B,GAAG,CAAC3B,UAAU,EACdE,IACF,CAAC;EAED,MAAM4B,KAAK,GAAGtC,QAAQ,CAACqC,SAAS,EAAEpC,eAAe,CAACK,QAAQ,CAAC,CAAC;EAC5D6B,GAAG,CAACjB,MAAM,GAAGoB,KAAK,CAACnC,MAAM,EAAEC,KAAK,EAAEA,KAAK,EAAEa,SAAS,EAAEZ,IAAI,CAAC;EACzD,OAAO8B,GAAG;AACZ,CAAC;AAED,SAASf,kBAAkBA,CAACjB,MAAc,EAAE;EAC1C,OAAO,CAAC,QAAQ,EAAE,UAAU,CAAC,CAACoC,QAAQ,CAAC,OAAOpC,MAAM,CAAC,GAAG,QAAQ,GAC5D,OAAOA,MACR;AACL;AAEA,MAAMM,UAA0B,GAAG;EACjCH,QAAQ,EAAE,CAAC,CAAC;EACZC,OAAO,EAAE,CAAC,CAAC;EACXC,UAAU,EAAE,CAAC;AACf,CAAC","ignoreList":[]}