UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

319 lines (318 loc) • 38.6 kB
/* eslint-disable no-param-reassign */ "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); Object.defineProperty(exports, "promise", { enumerable: true, get: function() { return promise; } }); const _deepmerge = /*#__PURE__*/ _interop_require_default(require("deepmerge")); const _types = require("../../config/types"); const _getExistingRowDoc = require("./getExistingRowDoc"); const _traverseFields = require("./traverseFields"); function _interop_require_default(obj) { return obj && obj.__esModule ? obj : { default: obj }; } const promise = async ({ id, collection, context, data, doc, docWithLocales, errors, field, global, mergeLocaleActions, operation, path, req, siblingData, siblingDoc, siblingDocWithLocales, skipValidation })=>{ const passesCondition = field.admin?.condition ? Boolean(field.admin.condition(data, siblingData, { user: req.user })) : true; let skipValidationFromHere = skipValidation || !passesCondition; const defaultLocale = req.payload.config?.localization ? req.payload.config.localization?.defaultLocale : 'en'; const operationLocale = req.locale || defaultLocale; if ((0, _types.fieldAffectsData)(field)) { // skip validation if the field is localized and the incoming data is null if (field.localized && operationLocale !== defaultLocale) { if ([ 'array', 'blocks' ].includes(field.type) && siblingData[field.name] === null) { skipValidationFromHere = true; } } // Execute hooks if (field.hooks?.beforeChange) { await field.hooks.beforeChange.reduce(async (priorHook, currentHook)=>{ await priorHook; const hookedValue = await currentHook({ collection, context, data, field, global, operation, originalDoc: doc, previousSiblingDoc: siblingDoc, previousValue: siblingDoc[field.name], req, siblingData, value: siblingData[field.name] }); if (hookedValue !== undefined) { siblingData[field.name] = hookedValue; } }, Promise.resolve()); } // Validate if (!skipValidationFromHere && field.validate) { const valueToValidate = siblingData[field.name]; let jsonError; if (field.type === 'json' && typeof siblingData[field.name] === 'string') { try { JSON.parse(siblingData[field.name]); } catch (e) { jsonError = e; } } const validationResult = await field.validate(valueToValidate, { ...field, id, config: req.payload.config, data: (0, _deepmerge.default)(doc, data, { arrayMerge: (_, source)=>source }), jsonError, operation, payload: req.payload, req, siblingData: (0, _deepmerge.default)(siblingDoc, siblingData, { arrayMerge: (_, source)=>source }), t: req.t, user: req.user }); if (typeof validationResult === 'string') { errors.push({ field: `${path}${field.name}`, message: validationResult }); } } // Push merge locale action if applicable if (field.localized) { mergeLocaleActions.push(()=>{ if (req.payload.config.localization) { const { localization } = req.payload.config; const localeData = localization.localeCodes.reduce((localizedValues, locale)=>{ const fieldValue = locale === req.locale ? siblingData[field.name] : siblingDocWithLocales?.[field.name]?.[locale]; // update locale value if it's not undefined if (typeof fieldValue !== 'undefined') { return { ...localizedValues, [locale]: fieldValue }; } return localizedValues; }, {}); // If there are locales with data, set the data if (Object.keys(localeData).length > 0) { siblingData[field.name] = localeData; } } }); } } switch(field.type){ case 'point': { // Transform point data for storage if (Array.isArray(siblingData[field.name]) && siblingData[field.name][0] !== null && siblingData[field.name][1] !== null) { siblingData[field.name] = { coordinates: [ parseFloat(siblingData[field.name][0]), parseFloat(siblingData[field.name][1]) ], type: 'Point' }; } break; } case 'group': { if (typeof siblingData[field.name] !== 'object') siblingData[field.name] = {}; if (typeof siblingDoc[field.name] !== 'object') siblingDoc[field.name] = {}; if (typeof siblingDocWithLocales[field.name] !== 'object') siblingDocWithLocales[field.name] = {}; await (0, _traverseFields.traverseFields)({ id, collection, context, data, doc, docWithLocales, errors, fields: field.fields, global, mergeLocaleActions, operation, path: `${path}${field.name}.`, req, siblingData: siblingData[field.name], siblingDoc: siblingDoc[field.name], siblingDocWithLocales: siblingDocWithLocales[field.name], skipValidation: skipValidationFromHere }); break; } case 'array': { const rows = siblingData[field.name]; if (Array.isArray(rows)) { const promises = []; rows.forEach((row, i)=>{ promises.push((0, _traverseFields.traverseFields)({ id, collection, context, data, doc, docWithLocales, errors, fields: field.fields, global, mergeLocaleActions, operation, path: `${path}${field.name}.${i}.`, req, siblingData: row, siblingDoc: (0, _getExistingRowDoc.getExistingRowDoc)(row, siblingDoc[field.name]), siblingDocWithLocales: (0, _getExistingRowDoc.getExistingRowDoc)(row, siblingDocWithLocales[field.name]), skipValidation: skipValidationFromHere })); }); await Promise.all(promises); } break; } case 'blocks': { const rows = siblingData[field.name]; if (Array.isArray(rows)) { const promises = []; rows.forEach((row, i)=>{ const rowSiblingDoc = (0, _getExistingRowDoc.getExistingRowDoc)(row, siblingDoc[field.name]); const rowSiblingDocWithLocales = (0, _getExistingRowDoc.getExistingRowDoc)(row, siblingDocWithLocales[field.name]); const blockTypeToMatch = row.blockType || rowSiblingDoc.blockType; const block = field.blocks.find((blockType)=>blockType.slug === blockTypeToMatch); if (block) { promises.push((0, _traverseFields.traverseFields)({ id, collection, context, data, doc, docWithLocales, errors, fields: block.fields, global, mergeLocaleActions, operation, path: `${path}${field.name}.${i}.`, req, siblingData: row, siblingDoc: rowSiblingDoc, siblingDocWithLocales: rowSiblingDocWithLocales, skipValidation: skipValidationFromHere })); } }); await Promise.all(promises); } break; } case 'row': case 'collapsible': { await (0, _traverseFields.traverseFields)({ id, collection, context, data, doc, docWithLocales, errors, fields: field.fields, global, mergeLocaleActions, operation, path, req, siblingData, siblingDoc, siblingDocWithLocales, skipValidation: skipValidationFromHere }); break; } case 'tab': { let tabPath = path; let tabSiblingData = siblingData; let tabSiblingDoc = siblingDoc; let tabSiblingDocWithLocales = siblingDocWithLocales; if ((0, _types.tabHasName)(field)) { tabPath = `${path}${field.name}.`; if (typeof siblingData[field.name] !== 'object') siblingData[field.name] = {}; if (typeof siblingDoc[field.name] !== 'object') siblingDoc[field.name] = {}; if (typeof siblingDocWithLocales[field.name] !== 'object') siblingDocWithLocales[field.name] = {}; tabSiblingData = siblingData[field.name]; tabSiblingDoc = siblingDoc[field.name]; tabSiblingDocWithLocales = siblingDocWithLocales[field.name]; } await (0, _traverseFields.traverseFields)({ id, collection, context, data, doc, docWithLocales, errors, fields: field.fields, global, mergeLocaleActions, operation, path: tabPath, req, siblingData: tabSiblingData, siblingDoc: tabSiblingDoc, siblingDocWithLocales: tabSiblingDocWithLocales, skipValidation: skipValidationFromHere }); break; } case 'tabs': { await (0, _traverseFields.traverseFields)({ id, collection, context, data, doc, docWithLocales, errors, fields: field.tabs.map((tab)=>({ ...tab, type: 'tab' })), global, mergeLocaleActions, operation, path, req, siblingData, siblingDoc, siblingDocWithLocales, skipValidation: skipValidationFromHere }); break; } default: { break; } } }; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../src/fields/hooks/beforeChange/promise.ts"],"sourcesContent":["/* eslint-disable no-param-reassign */\nimport merge from 'deepmerge'\n\nimport type { SanitizedCollectionConfig } from '../../../collections/config/types'\nimport type { PayloadRequest, RequestContext } from '../../../express/types'\nimport type { SanitizedGlobalConfig } from '../../../globals/config/types'\nimport type { Operation } from '../../../types'\nimport type { Field, TabAsField } from '../../config/types'\n\nimport { fieldAffectsData, tabHasName } from '../../config/types'\nimport { getExistingRowDoc } from './getExistingRowDoc'\nimport { traverseFields } from './traverseFields'\n\ntype Args = {\n  collection: SanitizedCollectionConfig | null\n  context: RequestContext\n  data: Record<string, unknown>\n  doc: Record<string, unknown>\n  docWithLocales: Record<string, unknown>\n  errors: { field: string; message: string }[]\n  field: Field | TabAsField\n  global: SanitizedGlobalConfig | null\n  id?: number | string\n  mergeLocaleActions: (() => void)[]\n  operation: Operation\n  path: string\n  req: PayloadRequest\n  siblingData: Record<string, unknown>\n  siblingDoc: Record<string, unknown>\n  siblingDocWithLocales?: Record<string, unknown>\n  skipValidation: boolean\n}\n\n// This function is responsible for the following actions, in order:\n// - Run condition\n// - Execute field hooks\n// - Validate data\n// - Transform data for storage\n// - Unflatten locales\n\nexport const promise = async ({\n  id,\n  collection,\n  context,\n  data,\n  doc,\n  docWithLocales,\n  errors,\n  field,\n  global,\n  mergeLocaleActions,\n  operation,\n  path,\n  req,\n  siblingData,\n  siblingDoc,\n  siblingDocWithLocales,\n  skipValidation,\n}: Args): Promise<void> => {\n  const passesCondition = field.admin?.condition\n    ? Boolean(field.admin.condition(data, siblingData, { user: req.user }))\n    : true\n  let skipValidationFromHere = skipValidation || !passesCondition\n\n  const defaultLocale = req.payload.config?.localization\n    ? req.payload.config.localization?.defaultLocale\n    : 'en'\n  const operationLocale = req.locale || defaultLocale\n\n  if (fieldAffectsData(field)) {\n    // skip validation if the field is localized and the incoming data is null\n    if (field.localized && operationLocale !== defaultLocale) {\n      if (['array', 'blocks'].includes(field.type) && siblingData[field.name] === null) {\n        skipValidationFromHere = true\n      }\n    }\n\n    // Execute hooks\n    if (field.hooks?.beforeChange) {\n      await field.hooks.beforeChange.reduce(async (priorHook, currentHook) => {\n        await priorHook\n\n        const hookedValue = await currentHook({\n          collection,\n          context,\n          data,\n          field,\n          global,\n          operation,\n          originalDoc: doc,\n          previousSiblingDoc: siblingDoc,\n          previousValue: siblingDoc[field.name],\n          req,\n          siblingData,\n          value: siblingData[field.name],\n        })\n\n        if (hookedValue !== undefined) {\n          siblingData[field.name] = hookedValue\n        }\n      }, Promise.resolve())\n    }\n\n    // Validate\n    if (!skipValidationFromHere && field.validate) {\n      const valueToValidate = siblingData[field.name]\n      let jsonError\n\n      if (field.type === 'json' && typeof siblingData[field.name] === 'string') {\n        try {\n          JSON.parse(siblingData[field.name] as string)\n        } catch (e) {\n          jsonError = e\n        }\n      }\n\n      const validationResult = await field.validate(valueToValidate, {\n        ...field,\n        id,\n        config: req.payload.config,\n        data: merge(doc, data, { arrayMerge: (_, source) => source }),\n        jsonError,\n        operation,\n        payload: req.payload,\n        req,\n        siblingData: merge(siblingDoc, siblingData, { arrayMerge: (_, source) => source }),\n        t: req.t,\n        user: req.user,\n      })\n\n      if (typeof validationResult === 'string') {\n        errors.push({\n          field: `${path}${field.name}`,\n          message: validationResult,\n        })\n      }\n    }\n\n    // Push merge locale action if applicable\n    if (field.localized) {\n      mergeLocaleActions.push(() => {\n        if (req.payload.config.localization) {\n          const { localization } = req.payload.config\n          const localeData = localization.localeCodes.reduce((localizedValues, locale) => {\n            const fieldValue =\n              locale === req.locale\n                ? siblingData[field.name]\n                : siblingDocWithLocales?.[field.name]?.[locale]\n\n            // update locale value if it's not undefined\n            if (typeof fieldValue !== 'undefined') {\n              return {\n                ...localizedValues,\n                [locale]: fieldValue,\n              }\n            }\n\n            return localizedValues\n          }, {})\n\n          // If there are locales with data, set the data\n          if (Object.keys(localeData).length > 0) {\n            siblingData[field.name] = localeData\n          }\n        }\n      })\n    }\n  }\n\n  switch (field.type) {\n    case 'point': {\n      // Transform point data for storage\n      if (\n        Array.isArray(siblingData[field.name]) &&\n        siblingData[field.name][0] !== null &&\n        siblingData[field.name][1] !== null\n      ) {\n        siblingData[field.name] = {\n          coordinates: [\n            parseFloat(siblingData[field.name][0]),\n            parseFloat(siblingData[field.name][1]),\n          ],\n          type: 'Point',\n        }\n      }\n\n      break\n    }\n\n    case 'group': {\n      if (typeof siblingData[field.name] !== 'object') siblingData[field.name] = {}\n      if (typeof siblingDoc[field.name] !== 'object') siblingDoc[field.name] = {}\n      if (typeof siblingDocWithLocales[field.name] !== 'object')\n        siblingDocWithLocales[field.name] = {}\n\n      await traverseFields({\n        id,\n        collection,\n        context,\n        data,\n        doc,\n        docWithLocales,\n        errors,\n        fields: field.fields,\n        global,\n        mergeLocaleActions,\n        operation,\n        path: `${path}${field.name}.`,\n        req,\n        siblingData: siblingData[field.name] as Record<string, unknown>,\n        siblingDoc: siblingDoc[field.name] as Record<string, unknown>,\n        siblingDocWithLocales: siblingDocWithLocales[field.name] as Record<string, unknown>,\n        skipValidation: skipValidationFromHere,\n      })\n\n      break\n    }\n\n    case 'array': {\n      const rows = siblingData[field.name]\n\n      if (Array.isArray(rows)) {\n        const promises = []\n        rows.forEach((row, i) => {\n          promises.push(\n            traverseFields({\n              id,\n              collection,\n              context,\n              data,\n              doc,\n              docWithLocales,\n              errors,\n              fields: field.fields,\n              global,\n              mergeLocaleActions,\n              operation,\n              path: `${path}${field.name}.${i}.`,\n              req,\n              siblingData: row,\n              siblingDoc: getExistingRowDoc(row, siblingDoc[field.name]),\n              siblingDocWithLocales: getExistingRowDoc(row, siblingDocWithLocales[field.name]),\n              skipValidation: skipValidationFromHere,\n            }),\n          )\n        })\n\n        await Promise.all(promises)\n      }\n\n      break\n    }\n\n    case 'blocks': {\n      const rows = siblingData[field.name]\n\n      if (Array.isArray(rows)) {\n        const promises = []\n        rows.forEach((row, i) => {\n          const rowSiblingDoc = getExistingRowDoc(row, siblingDoc[field.name])\n          const rowSiblingDocWithLocales = getExistingRowDoc(row, siblingDocWithLocales[field.name])\n\n          const blockTypeToMatch = row.blockType || rowSiblingDoc.blockType\n          const block = field.blocks.find((blockType) => blockType.slug === blockTypeToMatch)\n\n          if (block) {\n            promises.push(\n              traverseFields({\n                id,\n                collection,\n                context,\n                data,\n                doc,\n                docWithLocales,\n                errors,\n                fields: block.fields,\n                global,\n                mergeLocaleActions,\n                operation,\n                path: `${path}${field.name}.${i}.`,\n                req,\n                siblingData: row,\n                siblingDoc: rowSiblingDoc,\n                siblingDocWithLocales: rowSiblingDocWithLocales,\n                skipValidation: skipValidationFromHere,\n              }),\n            )\n          }\n        })\n\n        await Promise.all(promises)\n      }\n\n      break\n    }\n\n    case 'row':\n    case 'collapsible': {\n      await traverseFields({\n        id,\n        collection,\n        context,\n        data,\n        doc,\n        docWithLocales,\n        errors,\n        fields: field.fields,\n        global,\n        mergeLocaleActions,\n        operation,\n        path,\n        req,\n        siblingData,\n        siblingDoc,\n        siblingDocWithLocales,\n        skipValidation: skipValidationFromHere,\n      })\n\n      break\n    }\n\n    case 'tab': {\n      let tabPath = path\n      let tabSiblingData = siblingData\n      let tabSiblingDoc = siblingDoc\n      let tabSiblingDocWithLocales = siblingDocWithLocales\n\n      if (tabHasName(field)) {\n        tabPath = `${path}${field.name}.`\n        if (typeof siblingData[field.name] !== 'object') siblingData[field.name] = {}\n        if (typeof siblingDoc[field.name] !== 'object') siblingDoc[field.name] = {}\n        if (typeof siblingDocWithLocales[field.name] !== 'object')\n          siblingDocWithLocales[field.name] = {}\n\n        tabSiblingData = siblingData[field.name] as Record<string, unknown>\n        tabSiblingDoc = siblingDoc[field.name] as Record<string, unknown>\n        tabSiblingDocWithLocales = siblingDocWithLocales[field.name] as Record<string, unknown>\n      }\n\n      await traverseFields({\n        id,\n        collection,\n        context,\n        data,\n        doc,\n        docWithLocales,\n        errors,\n        fields: field.fields,\n        global,\n        mergeLocaleActions,\n        operation,\n        path: tabPath,\n        req,\n        siblingData: tabSiblingData,\n        siblingDoc: tabSiblingDoc,\n        siblingDocWithLocales: tabSiblingDocWithLocales,\n        skipValidation: skipValidationFromHere,\n      })\n\n      break\n    }\n\n    case 'tabs': {\n      await traverseFields({\n        id,\n        collection,\n        context,\n        data,\n        doc,\n        docWithLocales,\n        errors,\n        fields: field.tabs.map((tab) => ({ ...tab, type: 'tab' })),\n        global,\n        mergeLocaleActions,\n        operation,\n        path,\n        req,\n        siblingData,\n        siblingDoc,\n        siblingDocWithLocales,\n        skipValidation: skipValidationFromHere,\n      })\n\n      break\n    }\n\n    default: {\n      break\n    }\n  }\n}\n"],"names":["promise","id","collection","context","data","doc","docWithLocales","errors","field","global","mergeLocaleActions","operation","path","req","siblingData","siblingDoc","siblingDocWithLocales","skipValidation","passesCondition","admin","condition","Boolean","user","skipValidationFromHere","defaultLocale","payload","config","localization","operationLocale","locale","fieldAffectsData","localized","includes","type","name","hooks","beforeChange","reduce","priorHook","currentHook","hookedValue","originalDoc","previousSiblingDoc","previousValue","value","undefined","Promise","resolve","validate","valueToValidate","jsonError","JSON","parse","e","validationResult","merge","arrayMerge","_","source","t","push","message","localeData","localeCodes","localizedValues","fieldValue","Object","keys","length","Array","isArray","coordinates","parseFloat","traverseFields","fields","rows","promises","forEach","row","i","getExistingRowDoc","all","rowSiblingDoc","rowSiblingDocWithLocales","blockTypeToMatch","blockType","block","blocks","find","slug","tabPath","tabSiblingData","tabSiblingDoc","tabSiblingDocWithLocales","tabHasName","tabs","map","tab"],"mappings":"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"}