UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

145 lines (144 loc) • 18.4 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); Object.defineProperty(exports, "validateSearchParam", { enumerable: true, get: function() { return validateSearchParam; } }); const _types = require("../../fields/config/types"); const _getEntityPolicies = require("../../utilities/getEntityPolicies"); const _getLocalizedPaths = require("../getLocalizedPaths"); const _validateQueryPaths = require("./validateQueryPaths"); async function validateSearchParam({ collectionConfig, errors, fields, globalConfig, operator, overrideAccess, path: incomingPath, policies, req, val, versionFields }) { // Replace GraphQL nested field double underscore formatting let sanitizedPath; if (incomingPath === '_id') { sanitizedPath = 'id'; } else { sanitizedPath = incomingPath.replace(/__/g, '.'); } let paths = []; const { slug } = collectionConfig || globalConfig; if (globalConfig && !policies.globals[slug]) { // eslint-disable-next-line no-param-reassign globalConfig.fields = fields; // eslint-disable-next-line no-param-reassign policies.globals[slug] = await (0, _getEntityPolicies.getEntityPolicies)({ entity: globalConfig, operations: [ 'read' ], req, type: 'global' }); } if (sanitizedPath !== 'id') { paths = await (0, _getLocalizedPaths.getLocalizedPaths)({ collectionSlug: collectionConfig?.slug, fields, globalSlug: globalConfig?.slug, incomingPath: sanitizedPath, locale: req.locale, overrideAccess, payload: req.payload }); } const promises = []; promises.push(...paths.map(async ({ collectionSlug, field, invalid, path }, i)=>{ if (invalid) { errors.push({ path }); return; } if (!overrideAccess && (0, _types.fieldAffectsData)(field)) { if (collectionSlug) { if (!policies.collections[collectionSlug]) { // eslint-disable-next-line no-param-reassign policies.collections[collectionSlug] = await (0, _getEntityPolicies.getEntityPolicies)({ entity: req.payload.collections[collectionSlug].config, operations: [ 'read' ], req, type: 'collection' }); } if ([ 'hash', 'salt' ].includes(incomingPath) && collectionConfig.auth && !collectionConfig.auth?.disableLocalStrategy) { errors.push({ path: incomingPath }); } } let fieldPath = path; // remove locale from end of path if (path.endsWith(`.${req.locale}`)) { fieldPath = path.slice(0, -(req.locale.length + 1)); } // remove ".value" from ends of polymorphic relationship paths if (field.type === 'relationship' && Array.isArray(field.relationTo)) { fieldPath = fieldPath.replace('.value', ''); } const entityType = globalConfig ? 'globals' : 'collections'; const entitySlug = collectionSlug || globalConfig.slug; const segments = fieldPath.split('.'); let fieldAccess; if (versionFields) { fieldAccess = policies[entityType][entitySlug]; if (segments[0] === 'parent' || segments[0] === 'version') { segments.shift(); } } else { fieldAccess = policies[entityType][entitySlug].fields; } segments.forEach((segment)=>{ if (fieldAccess[segment]) { if ('fields' in fieldAccess[segment]) { fieldAccess = fieldAccess[segment].fields; } else if ('blocks' in fieldAccess[segment]) { fieldAccess = fieldAccess[segment]; } else { fieldAccess = fieldAccess[segment]; } } }); if (!fieldAccess?.read?.permission) { errors.push({ path: fieldPath }); } } if (i > 1) { // Remove top collection and reverse array // to work backwards from top const pathsToQuery = paths.slice(1).reverse(); pathsToQuery.forEach(({ collectionSlug: pathCollectionSlug, path: subPath }, pathToQueryIndex)=>{ // On the "deepest" collection, // validate query of the relationship if (pathToQueryIndex === 0) { promises.push((0, _validateQueryPaths.validateQueryPaths)({ collectionConfig: req.payload.collections[pathCollectionSlug].config, errors, globalConfig: undefined, overrideAccess, policies, req, where: { [subPath]: { [operator]: val } } })); } }); } })); await Promise.all(promises); } //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/database/queryValidation/validateSearchParams.ts"],"sourcesContent":["import type { SanitizedCollectionConfig } from '../../collections/config/types'\nimport type { PayloadRequest } from '../../express/types'\nimport type { Field } from '../../fields/config/types'\nimport type { SanitizedGlobalConfig } from '../../globals/config/types'\nimport type { EntityPolicies, PathToQuery } from './types'\n\nimport { fieldAffectsData } from '../../fields/config/types'\nimport { getEntityPolicies } from '../../utilities/getEntityPolicies'\nimport { getLocalizedPaths } from '../getLocalizedPaths'\nimport { validateQueryPaths } from './validateQueryPaths'\n\ntype Args = {\n  collectionConfig?: SanitizedCollectionConfig\n  errors: { path: string }[]\n  fields: Field[]\n  globalConfig?: SanitizedGlobalConfig\n  operator: string\n  overrideAccess: boolean\n  path: string\n  policies: EntityPolicies\n  req: PayloadRequest\n  val: unknown\n  versionFields?: Field[]\n}\n\n/**\n * Validate the Payload key / value / operator\n */\nexport async function validateSearchParam({\n  collectionConfig,\n  errors,\n  fields,\n  globalConfig,\n  operator,\n  overrideAccess,\n  path: incomingPath,\n  policies,\n  req,\n  val,\n  versionFields,\n}: Args): Promise<void> {\n  // Replace GraphQL nested field double underscore formatting\n  let sanitizedPath\n  if (incomingPath === '_id') {\n    sanitizedPath = 'id'\n  } else {\n    sanitizedPath = incomingPath.replace(/__/g, '.')\n  }\n  let paths: PathToQuery[] = []\n  const { slug } = collectionConfig || globalConfig\n\n  if (globalConfig && !policies.globals[slug]) {\n    // eslint-disable-next-line no-param-reassign\n    globalConfig.fields = fields\n\n    // eslint-disable-next-line no-param-reassign\n    policies.globals[slug] = await getEntityPolicies({\n      entity: globalConfig,\n      operations: ['read'],\n      req,\n      type: 'global',\n    })\n  }\n\n  if (sanitizedPath !== 'id') {\n    paths = await getLocalizedPaths({\n      collectionSlug: collectionConfig?.slug,\n      fields,\n      globalSlug: globalConfig?.slug,\n      incomingPath: sanitizedPath,\n      locale: req.locale,\n      overrideAccess,\n      payload: req.payload,\n    })\n  }\n  const promises = []\n  promises.push(\n    ...paths.map(async ({ collectionSlug, field, invalid, path }, i) => {\n      if (invalid) {\n        errors.push({ path })\n        return\n      }\n\n      if (!overrideAccess && fieldAffectsData(field)) {\n        if (collectionSlug) {\n          if (!policies.collections[collectionSlug]) {\n            // eslint-disable-next-line no-param-reassign\n            policies.collections[collectionSlug] = await getEntityPolicies({\n              entity: req.payload.collections[collectionSlug].config,\n              operations: ['read'],\n              req,\n              type: 'collection',\n            })\n          }\n\n          if (\n            ['hash', 'salt'].includes(incomingPath) &&\n            collectionConfig.auth &&\n            !collectionConfig.auth?.disableLocalStrategy\n          ) {\n            errors.push({ path: incomingPath })\n          }\n        }\n        let fieldPath = path\n        // remove locale from end of path\n        if (path.endsWith(`.${req.locale}`)) {\n          fieldPath = path.slice(0, -(req.locale.length + 1))\n        }\n        // remove \".value\" from ends of polymorphic relationship paths\n        if (field.type === 'relationship' && Array.isArray(field.relationTo)) {\n          fieldPath = fieldPath.replace('.value', '')\n        }\n        const entityType: 'collections' | 'globals' = globalConfig ? 'globals' : 'collections'\n        const entitySlug = collectionSlug || globalConfig.slug\n        const segments = fieldPath.split('.')\n\n        let fieldAccess\n        if (versionFields) {\n          fieldAccess = policies[entityType][entitySlug]\n          if (segments[0] === 'parent' || segments[0] === 'version') {\n            segments.shift()\n          }\n        } else {\n          fieldAccess = policies[entityType][entitySlug].fields\n        }\n\n        segments.forEach((segment) => {\n          if (fieldAccess[segment]) {\n            if ('fields' in fieldAccess[segment]) {\n              fieldAccess = fieldAccess[segment].fields\n            } else if ('blocks' in fieldAccess[segment]) {\n              fieldAccess = fieldAccess[segment]\n            } else {\n              fieldAccess = fieldAccess[segment]\n            }\n          }\n        })\n\n        if (!fieldAccess?.read?.permission) {\n          errors.push({ path: fieldPath })\n        }\n      }\n\n      if (i > 1) {\n        // Remove top collection and reverse array\n        // to work backwards from top\n        const pathsToQuery = paths.slice(1).reverse()\n\n        pathsToQuery.forEach(\n          ({ collectionSlug: pathCollectionSlug, path: subPath }, pathToQueryIndex) => {\n            // On the \"deepest\" collection,\n            // validate query of the relationship\n            if (pathToQueryIndex === 0) {\n              promises.push(\n                validateQueryPaths({\n                  collectionConfig: req.payload.collections[pathCollectionSlug].config,\n                  errors,\n                  globalConfig: undefined,\n                  overrideAccess,\n                  policies,\n                  req,\n                  where: {\n                    [subPath]: {\n                      [operator]: val,\n                    },\n                  },\n                }),\n              )\n            }\n          },\n        )\n      }\n    }),\n  )\n  await Promise.all(promises)\n}\n"],"names":["validateSearchParam","collectionConfig","errors","fields","globalConfig","operator","overrideAccess","path","incomingPath","policies","req","val","versionFields","sanitizedPath","replace","paths","slug","globals","getEntityPolicies","entity","operations","type","getLocalizedPaths","collectionSlug","globalSlug","locale","payload","promises","push","map","field","invalid","i","fieldAffectsData","collections","config","includes","auth","disableLocalStrategy","fieldPath","endsWith","slice","length","Array","isArray","relationTo","entityType","entitySlug","segments","split","fieldAccess","shift","forEach","segment","read","permission","pathsToQuery","reverse","pathCollectionSlug","subPath","pathToQueryIndex","validateQueryPaths","undefined","where","Promise","all"],"mappings":";;;;+BA4BsBA;;;eAAAA;;;uBAtBW;mCACC;mCACA;oCACC;AAmB5B,eAAeA,oBAAoB,EACxCC,gBAAgB,EAChBC,MAAM,EACNC,MAAM,EACNC,YAAY,EACZC,QAAQ,EACRC,cAAc,EACdC,MAAMC,YAAY,EAClBC,QAAQ,EACRC,GAAG,EACHC,GAAG,EACHC,aAAa,EACR;IACL,4DAA4D;IAC5D,IAAIC;IACJ,IAAIL,iBAAiB,OAAO;QAC1BK,gBAAgB;IAClB,OAAO;QACLA,gBAAgBL,aAAaM,OAAO,CAAC,OAAO;IAC9C;IACA,IAAIC,QAAuB,EAAE;IAC7B,MAAM,EAAEC,IAAI,EAAE,GAAGf,oBAAoBG;IAErC,IAAIA,gBAAgB,CAACK,SAASQ,OAAO,CAACD,KAAK,EAAE;QAC3C,6CAA6C;QAC7CZ,aAAaD,MAAM,GAAGA;QAEtB,6CAA6C;QAC7CM,SAASQ,OAAO,CAACD,KAAK,GAAG,MAAME,IAAAA,oCAAiB,EAAC;YAC/CC,QAAQf;YACRgB,YAAY;gBAAC;aAAO;YACpBV;YACAW,MAAM;QACR;IACF;IAEA,IAAIR,kBAAkB,MAAM;QAC1BE,QAAQ,MAAMO,IAAAA,oCAAiB,EAAC;YAC9BC,gBAAgBtB,kBAAkBe;YAClCb;YACAqB,YAAYpB,cAAcY;YAC1BR,cAAcK;YACdY,QAAQf,IAAIe,MAAM;YAClBnB;YACAoB,SAAShB,IAAIgB,OAAO;QACtB;IACF;IACA,MAAMC,WAAW,EAAE;IACnBA,SAASC,IAAI,IACRb,MAAMc,GAAG,CAAC,OAAO,EAAEN,cAAc,EAAEO,KAAK,EAAEC,OAAO,EAAExB,IAAI,EAAE,EAAEyB;QAC5D,IAAID,SAAS;YACX7B,OAAO0B,IAAI,CAAC;gBAAErB;YAAK;YACnB;QACF;QAEA,IAAI,CAACD,kBAAkB2B,IAAAA,uBAAgB,EAACH,QAAQ;YAC9C,IAAIP,gBAAgB;gBAClB,IAAI,CAACd,SAASyB,WAAW,CAACX,eAAe,EAAE;oBACzC,6CAA6C;oBAC7Cd,SAASyB,WAAW,CAACX,eAAe,GAAG,MAAML,IAAAA,oCAAiB,EAAC;wBAC7DC,QAAQT,IAAIgB,OAAO,CAACQ,WAAW,CAACX,eAAe,CAACY,MAAM;wBACtDf,YAAY;4BAAC;yBAAO;wBACpBV;wBACAW,MAAM;oBACR;gBACF;gBAEA,IACE;oBAAC;oBAAQ;iBAAO,CAACe,QAAQ,CAAC5B,iBAC1BP,iBAAiBoC,IAAI,IACrB,CAACpC,iBAAiBoC,IAAI,EAAEC,sBACxB;oBACApC,OAAO0B,IAAI,CAAC;wBAAErB,MAAMC;oBAAa;gBACnC;YACF;YACA,IAAI+B,YAAYhC;YAChB,iCAAiC;YACjC,IAAIA,KAAKiC,QAAQ,CAAC,CAAC,CAAC,EAAE9B,IAAIe,MAAM,CAAC,CAAC,GAAG;gBACnCc,YAAYhC,KAAKkC,KAAK,CAAC,GAAG,CAAE/B,CAAAA,IAAIe,MAAM,CAACiB,MAAM,GAAG,CAAA;YAClD;YACA,8DAA8D;YAC9D,IAAIZ,MAAMT,IAAI,KAAK,kBAAkBsB,MAAMC,OAAO,CAACd,MAAMe,UAAU,GAAG;gBACpEN,YAAYA,UAAUzB,OAAO,CAAC,UAAU;YAC1C;YACA,MAAMgC,aAAwC1C,eAAe,YAAY;YACzE,MAAM2C,aAAaxB,kBAAkBnB,aAAaY,IAAI;YACtD,MAAMgC,WAAWT,UAAUU,KAAK,CAAC;YAEjC,IAAIC;YACJ,IAAItC,eAAe;gBACjBsC,cAAczC,QAAQ,CAACqC,WAAW,CAACC,WAAW;gBAC9C,IAAIC,QAAQ,CAAC,EAAE,KAAK,YAAYA,QAAQ,CAAC,EAAE,KAAK,WAAW;oBACzDA,SAASG,KAAK;gBAChB;YACF,OAAO;gBACLD,cAAczC,QAAQ,CAACqC,WAAW,CAACC,WAAW,CAAC5C,MAAM;YACvD;YAEA6C,SAASI,OAAO,CAAC,CAACC;gBAChB,IAAIH,WAAW,CAACG,QAAQ,EAAE;oBACxB,IAAI,YAAYH,WAAW,CAACG,QAAQ,EAAE;wBACpCH,cAAcA,WAAW,CAACG,QAAQ,CAAClD,MAAM;oBAC3C,OAAO,IAAI,YAAY+C,WAAW,CAACG,QAAQ,EAAE;wBAC3CH,cAAcA,WAAW,CAACG,QAAQ;oBACpC,OAAO;wBACLH,cAAcA,WAAW,CAACG,QAAQ;oBACpC;gBACF;YACF;YAEA,IAAI,CAACH,aAAaI,MAAMC,YAAY;gBAClCrD,OAAO0B,IAAI,CAAC;oBAAErB,MAAMgC;gBAAU;YAChC;QACF;QAEA,IAAIP,IAAI,GAAG;YACT,0CAA0C;YAC1C,6BAA6B;YAC7B,MAAMwB,eAAezC,MAAM0B,KAAK,CAAC,GAAGgB,OAAO;YAE3CD,aAAaJ,OAAO,CAClB,CAAC,EAAE7B,gBAAgBmC,kBAAkB,EAAEnD,MAAMoD,OAAO,EAAE,EAAEC;gBACtD,+BAA+B;gBAC/B,qCAAqC;gBACrC,IAAIA,qBAAqB,GAAG;oBAC1BjC,SAASC,IAAI,CACXiC,IAAAA,sCAAkB,EAAC;wBACjB5D,kBAAkBS,IAAIgB,OAAO,CAACQ,WAAW,CAACwB,mBAAmB,CAACvB,MAAM;wBACpEjC;wBACAE,cAAc0D;wBACdxD;wBACAG;wBACAC;wBACAqD,OAAO;4BACL,CAACJ,QAAQ,EAAE;gCACT,CAACtD,SAAS,EAAEM;4BACd;wBACF;oBACF;gBAEJ;YACF;QAEJ;IACF;IAEF,MAAMqD,QAAQC,GAAG,CAACtC;AACpB"}