UNPKG

@grnsft/if

Version:

Impact Framework

98 lines 13 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.CSVLookup = void 0; /* eslint-disable eqeqeq */ const zod_1 = require("zod"); const interfaces_1 = require("@grnsft/if-core/interfaces"); const utils_1 = require("@grnsft/if-core/utils"); const config_1 = require("../../config"); const csv_helpers_1 = require("../util/csv-helpers"); const { MISSING_CONFIG, NO_QUERY_DATA } = config_1.STRINGS; const { QueryDataNotFoundError, ConfigError } = utils_1.ERRORS; exports.CSVLookup = (0, interfaces_1.PluginFactory)({ configValidation: (config) => { if (!config || !Object.keys(config)?.length) { throw new ConfigError(MISSING_CONFIG); } const configSchema = zod_1.z.object({ filepath: zod_1.z.string(), query: zod_1.z.record(zod_1.z.string(), zod_1.z.string()), output: zod_1.z .string() .or(zod_1.z.array(zod_1.z.string())) .or(zod_1.z.array(zod_1.z.array(zod_1.z.string()))), }); return (0, utils_1.validate)(configSchema, config); }, implementation: async (inputs, config) => { /** * 1. Tries to retrieve given file (with url or local path). * 2. Parses given CSV. * 3. Filters requested information from CSV. */ const { filepath, query, output } = config; const file = await (0, csv_helpers_1.retrieveFile)(filepath); const parsedCSV = (0, csv_helpers_1.parseCSVFile)(file); return inputs.map(input => { /** Collects query values from input. */ const queryData = {}; const queryKeys = Object.keys(query); queryKeys.forEach(queryKey => { const queryValue = query[queryKey]; queryData[queryKey] = input[queryValue]; }); /** Gets related data from CSV. */ const relatedData = parsedCSV.find(withCriteria(queryData)); if (!relatedData) { throw new QueryDataNotFoundError(NO_QUERY_DATA); } return { ...input, ...filterOutput(relatedData, { output, query }), }; }); }, }); /** * 1. If output is anything, then removes query data from csv record to escape duplicates. * 2. Otherwise checks if it's a miltidimensional array, then grabs multiple fields (). * 3. If not, then returns single field. * 4. In case if it's string, then */ const filterOutput = (dataFromCSV, params) => { const { output, query } = params; if (output === '*') { const keys = Object.keys(query); keys.forEach(key => { delete dataFromCSV[key]; }); return (0, csv_helpers_1.nanifyEmptyValues)(dataFromCSV); } if (Array.isArray(output)) { /** Check if it's a multidimensional array. */ if (Array.isArray(output[0])) { const result = {}; output.forEach(outputField => { /** Check if there is no renaming request, then export as is */ const outputTitle = outputField[1] || outputField[0]; result[outputTitle] = (0, csv_helpers_1.fieldAccessor)(outputField[0], dataFromCSV); }); return result; } const outputTitle = output[1] || output[0]; return { [outputTitle]: (0, csv_helpers_1.fieldAccessor)(output[0], dataFromCSV), }; } return { [output]: (0, csv_helpers_1.fieldAccessor)(output, dataFromCSV), }; }; /** * Asserts CSV record with query data. */ const withCriteria = (queryData) => (csvRecord) => { const ifMatchesCriteria = Object.keys(queryData).map((key) => csvRecord[key] == queryData[key]); return ifMatchesCriteria.every(value => value === true); }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/if-run/builtins/csv-lookup/index.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable eqeqeq */\nimport {z} from 'zod';\n\nimport {ConfigParams, PluginParams} from '@grnsft/if-core/types';\nimport {PluginFactory} from '@grnsft/if-core/interfaces';\nimport {ERRORS, validate} from '@grnsft/if-core/utils';\n\nimport {STRINGS} from '../../config';\nimport {\n  fieldAccessor,\n  nanifyEmptyValues,\n  parseCSVFile,\n  retrieveFile,\n} from '../util/csv-helpers';\n\nconst {MISSING_CONFIG, NO_QUERY_DATA} = STRINGS;\n\nconst {QueryDataNotFoundError, ConfigError} = ERRORS;\n\nexport const CSVLookup = PluginFactory({\n  configValidation: (config: ConfigParams) => {\n    if (!config || !Object.keys(config)?.length) {\n      throw new ConfigError(MISSING_CONFIG);\n    }\n\n    const configSchema = z.object({\n      filepath: z.string(),\n      query: z.record(z.string(), z.string()),\n      output: z\n        .string()\n        .or(z.array(z.string()))\n        .or(z.array(z.array(z.string()))),\n    });\n\n    return validate<z.infer<typeof configSchema>>(configSchema, config);\n  },\n  implementation: async (inputs: PluginParams[], config: ConfigParams) => {\n    /**\n     * 1. Tries to retrieve given file (with url or local path).\n     * 2. Parses given CSV.\n     * 3. Filters requested information from CSV.\n     */\n    const {filepath, query, output} = config;\n    const file = await retrieveFile(filepath);\n    const parsedCSV = parseCSVFile(file);\n\n    return inputs.map(input => {\n      /** Collects query values from input. */\n      const queryData: any = {};\n      const queryKeys = Object.keys(query);\n      queryKeys.forEach(queryKey => {\n        const queryValue = query[queryKey];\n        queryData[queryKey] = input[queryValue];\n      });\n\n      /** Gets related data from CSV. */\n      const relatedData = parsedCSV.find(withCriteria(queryData));\n\n      if (!relatedData) {\n        throw new QueryDataNotFoundError(NO_QUERY_DATA);\n      }\n\n      return {\n        ...input,\n        ...filterOutput(relatedData, {output, query}),\n      };\n    });\n  },\n});\n\n/**\n * 1. If output is anything, then removes query data from csv record to escape duplicates.\n * 2. Otherwise checks if it's a miltidimensional array, then grabs multiple fields ().\n * 3. If not, then returns single field.\n * 4. In case if it's string, then\n */\nconst filterOutput = (\n  dataFromCSV: any,\n  params: {\n    output: string | string[] | string[][];\n    query: Record<string, any>;\n  }\n) => {\n  const {output, query} = params;\n\n  if (output === '*') {\n    const keys = Object.keys(query);\n\n    keys.forEach(key => {\n      delete dataFromCSV[key];\n    });\n\n    return nanifyEmptyValues(dataFromCSV);\n  }\n\n  if (Array.isArray(output)) {\n    /** Check if it's a multidimensional array. */\n    if (Array.isArray(output[0])) {\n      const result: any = {};\n\n      output.forEach(outputField => {\n        /** Check if there is no renaming request, then export as is */\n        const outputTitle = outputField[1] || outputField[0];\n        result[outputTitle] = fieldAccessor(outputField[0], dataFromCSV);\n      });\n\n      return result;\n    }\n\n    const outputTitle = output[1] || output[0];\n\n    return {\n      [outputTitle as string]: fieldAccessor(output[0], dataFromCSV),\n    };\n  }\n\n  return {\n    [output]: fieldAccessor(output, dataFromCSV),\n  };\n};\n\n/**\n * Asserts CSV record with query data.\n */\nconst withCriteria = (queryData: Record<string, any>) => (csvRecord: any) => {\n  const ifMatchesCriteria = Object.keys(queryData).map(\n    (key: string) => csvRecord[key] == queryData[key]\n  );\n\n  return ifMatchesCriteria.every(value => value === true);\n};\n"]}