UNPKG

jsonpath-mapper

Version:

A json to json transformation utility with a few nice features to use when translating for example API responses into a domain object for use in your domain-driven JavaScript applications. Can be used in React applications with the 'useMapper' hook.

169 lines (148 loc) 4.8 kB
import { fromEntries, isArray, isWireFunction, isNullOrUndefined, isMapperFunctions, isString, isUndefined, jpath, tryMultiple, isObject, } from './util.js'; import { MappingTemplate, MappingElement } from './models/Template.js'; import { FormatResult, HandleMapperFunctions, MapArrayFunction, MapElement, MapJson, MapObject, } from './models/MapperFunctions.js'; const formatResult: FormatResult = (value, $formatting, $root) => isWireFunction($formatting) ? $formatting(value, $root) : isObject($formatting) ? mapObject(value, $formatting, $root) : value; const handleMapperFunctions: HandleMapperFunctions = ([k, v], json, $root) => { // store results from various stages of the evaluations // eventually reducing this to a finalVal let val, formatted, finalVal: any; // if (v) { // evaluate a given string as a jsonpath expression if (isString(v.$path)) { val = jpath(v.$path, json); } if (isArray(v.$path)) { val = tryMultiple(json, v.$path, $root, findMultiple); } if (!isUndefined(v.$formatting) && !isNullOrUndefined(val)) { if (isArray(val)) { formatted = val.map( (inner) => !isUndefined(v.$formatting) && formatResult(inner, v.$formatting, $root) ); } else { formatted = formatResult(val, v.$formatting, $root); } val = formatted; } if (v.$return) { finalVal = formatResult(val, v.$return, $root); } else { finalVal = val; } if (!isUndefined(v.$default) && isNullOrUndefined(finalVal)) { if (isWireFunction(v.$default)) { finalVal = v.$default(json, $root); } else { finalVal = v.$default; } } if (v.$disable && isWireFunction(v.$disable) && v.$disable(finalVal, $root)) { return null; } return finalVal; }; const mapElement: MapElement = ([k, v], json, $root) => { if (isNullOrUndefined(v) || v === '') return [k, undefined as any]; // evaluate a given string as a jsonpath expression if (isString(v)) { return [k, jpath(v, json)]; } // execute a given function if (isWireFunction(v)) { return [k, v(json, $root)]; } // evaluate all the array strings as jsonpath // and then return the first match if (isArray(v)) { return [k, tryMultiple(json, v, $root, findMultiple)]; } // if typeof Object, this could be a template object // look for reserved key $path and it can be combined with // any of $formatting, $return, $default, $disable // if not we will treat as plain object and call self recursively if (isMapperFunctions(v)) { return [k, handleMapperFunctions([k, v], json, $root)]; } // Otherwise we have ourselves a nested object if (typeof v === 'object') return [k, mapObject(json, v, $root)]; return [k, v]; }; const mapObject: MapObject = (json, obj, $root) => { // For each key value entry in the object, evaluate each entry // and make the mapping from the provided json according to the // supplied object template that follows a few simple principles const objectAsEntries = Object.entries(obj) .map(([k, v]) => mapElement([k, v], json, $root)) .filter((obj) => isArray(obj) && obj[1] !== null); return fromEntries(objectAsEntries); }; const mapArray: MapArrayFunction = <S, T>( json: S, arr: MappingTemplate<S>[], $root: S ) => { return (arr.map((v) => mapObject(json, v, $root)) as unknown) as T; }; const findMultiple = <S>( json: S, arr: MappingElement<S>[], $root: S ): any[] => { const results = arr.map((inner, i) => { // evaluate any value supplied as string if (isString(inner)) return jpath(inner, json); // if typeof func if (isWireFunction(inner)) return inner(json, $root); // if typeof array if (isArray(inner)) return tryMultiple(json, inner, $root, findMultiple); // if typeof mapper functions if (isMapperFunctions(inner)) return handleMapperFunctions([i.toString(), inner], json, $root); // if typeof plain object if (typeof inner === 'object') return mapObject(json, inner as MappingTemplate<S>, $root); }); return results; }; const mapJson: MapJson = (json, template) => { const $root = json; // console.log(template); if (typeof template === 'function') { return template(json); } if (isArray(template)) { // console.log(mapArray(json, template)); return mapArray(json, template, $root); } // console.log(mapObject(json, template)); return mapObject(json, template, $root); }; export const useMapper: MapJson = (json, template) => { return mapJson(json, template); }; export default mapJson;