payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
319 lines (318 loc) • 38.6 kB
JavaScript
/* 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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"}