UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

205 lines (204 loc) • 25.6 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); Object.defineProperty(exports, "default", { enumerable: true, get: function() { return _default; } }); const _jsonwebtoken = /*#__PURE__*/ _interop_require_default(require("jsonwebtoken")); const _utils = require("../../collections/operations/utils"); const _errors = require("../../errors"); const _afterRead = require("../../fields/hooks/afterRead"); const _commitTransaction = require("../../utilities/commitTransaction"); const _getCookieExpiration = /*#__PURE__*/ _interop_require_default(require("../../utilities/getCookieExpiration")); const _initTransaction = require("../../utilities/initTransaction"); const _killTransaction = require("../../utilities/killTransaction"); const _sanitizeInternalFields = /*#__PURE__*/ _interop_require_default(require("../../utilities/sanitizeInternalFields")); const _isLocked = /*#__PURE__*/ _interop_require_default(require("../isLocked")); const _authenticate = require("../strategies/local/authenticate"); const _incrementLoginAttempts = require("../strategies/local/incrementLoginAttempts"); const _resetLoginAttempts = require("../strategies/local/resetLoginAttempts"); const _getFieldsToSign = require("./getFieldsToSign"); function _interop_require_default(obj) { return obj && obj.__esModule ? obj : { default: obj }; } async function login(incomingArgs) { let args = incomingArgs; try { const shouldCommit = await (0, _initTransaction.initTransaction)(args.req); // ///////////////////////////////////// // beforeOperation - Collection // ///////////////////////////////////// await args.collection.config.hooks.beforeOperation.reduce(async (priorHook, hook)=>{ await priorHook; args = await hook({ args, collection: args.collection?.config, context: args.req.context, operation: 'login', req: args.req }) || args; }, Promise.resolve()); const { collection: { config: collectionConfig }, data, depth, overrideAccess, req, req: { fallbackLocale, locale, payload, payload: { config, secret } }, showHiddenFields } = args; // ///////////////////////////////////// // Login // ///////////////////////////////////// const { email: unsanitizedEmail, password } = data; const email = unsanitizedEmail ? unsanitizedEmail.toLowerCase().trim() : null; let user = await payload.db.findOne({ collection: collectionConfig.slug, req, where: { email: { equals: email.toLowerCase() } } }); if (!user || args.collection.config.auth.verify && user._verified === false) { throw new _errors.AuthenticationError(req.t); } if (user && (0, _isLocked.default)(user.lockUntil)) { throw new _errors.LockedAuth(req.t); } const authResult = await (0, _authenticate.authenticateLocalStrategy)({ doc: user, password }); user = (0, _sanitizeInternalFields.default)(user); const maxLoginAttemptsEnabled = args.collection.config.auth.maxLoginAttempts > 0; if (!authResult) { if (maxLoginAttemptsEnabled) { await (0, _incrementLoginAttempts.incrementLoginAttempts)({ collection: collectionConfig, doc: user, payload: req.payload, req }); } if (shouldCommit) await (0, _commitTransaction.commitTransaction)(req); throw new _errors.AuthenticationError(req.t); } if (maxLoginAttemptsEnabled) { await (0, _resetLoginAttempts.resetLoginAttempts)({ collection: collectionConfig, doc: user, payload: req.payload, req }); } const fieldsToSign = (0, _getFieldsToSign.getFieldsToSign)({ collectionConfig, email, user }); await collectionConfig.hooks.beforeLogin.reduce(async (priorHook, hook)=>{ await priorHook; user = await hook({ collection: args.collection?.config, context: args.req.context, req: args.req, user }) || user; }, Promise.resolve()); const token = _jsonwebtoken.default.sign(fieldsToSign, secret, { expiresIn: collectionConfig.auth.tokenExpiration }); if (args.res) { const cookieOptions = { domain: undefined, expires: (0, _getCookieExpiration.default)(collectionConfig.auth.tokenExpiration), httpOnly: true, path: '/', sameSite: collectionConfig.auth.cookies.sameSite, secure: collectionConfig.auth.cookies.secure }; if (collectionConfig.auth.cookies.domain) cookieOptions.domain = collectionConfig.auth.cookies.domain; args.res.cookie(`${config.cookiePrefix}-token`, token, cookieOptions); } req.user = user; // ///////////////////////////////////// // afterLogin - Collection // ///////////////////////////////////// await collectionConfig.hooks.afterLogin.reduce(async (priorHook, hook)=>{ await priorHook; user = await hook({ collection: args.collection?.config, context: args.req.context, req: args.req, token, user }) || user; }, Promise.resolve()); // ///////////////////////////////////// // afterRead - Fields // ///////////////////////////////////// user = await (0, _afterRead.afterRead)({ collection: collectionConfig, context: req.context, depth, doc: user, fallbackLocale, global: null, locale, overrideAccess, req, showHiddenFields }); // ///////////////////////////////////// // afterRead - Collection // ///////////////////////////////////// await collectionConfig.hooks.afterRead.reduce(async (priorHook, hook)=>{ await priorHook; user = await hook({ collection: args.collection?.config, context: req.context, doc: user, req }) || user; }, Promise.resolve()); // ///////////////////////////////////// // afterRead - Collection // ///////////////////////////////////// await collectionConfig.hooks.afterRead.reduce(async (priorHook, hook)=>{ await priorHook; user = await hook({ collection: args.collection?.config, context: req.context, doc: user, req }) || user; }, Promise.resolve()); let result = { exp: _jsonwebtoken.default.decode(token).exp, token, user }; // ///////////////////////////////////// // afterOperation - Collection // ///////////////////////////////////// result = await (0, _utils.buildAfterOperation)({ args, collection: args.collection?.config, operation: 'login', result }); if (collectionConfig.auth.removeTokenFromResponses) { delete result.token; } // ///////////////////////////////////// // Return results // ///////////////////////////////////// if (shouldCommit) await (0, _commitTransaction.commitTransaction)(req); return result; } catch (error) { await (0, _killTransaction.killTransaction)(args.req); throw error; } } const _default = login; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/auth/operations/login.ts"],"sourcesContent":["import type { CookieOptions, Response } from 'express'\n\nimport jwt from 'jsonwebtoken'\n\nimport type { GeneratedTypes } from '../../'\nimport type { Collection } from '../../collections/config/types'\nimport type { PayloadRequest } from '../../express/types'\nimport type { User } from '../types'\n\nimport { buildAfterOperation } from '../../collections/operations/utils'\nimport { AuthenticationError, LockedAuth } from '../../errors'\nimport { afterRead } from '../../fields/hooks/afterRead'\nimport { commitTransaction } from '../../utilities/commitTransaction'\nimport getCookieExpiration from '../../utilities/getCookieExpiration'\nimport { initTransaction } from '../../utilities/initTransaction'\nimport { killTransaction } from '../../utilities/killTransaction'\nimport sanitizeInternalFields from '../../utilities/sanitizeInternalFields'\nimport isLocked from '../isLocked'\nimport { authenticateLocalStrategy } from '../strategies/local/authenticate'\nimport { incrementLoginAttempts } from '../strategies/local/incrementLoginAttempts'\nimport { resetLoginAttempts } from '../strategies/local/resetLoginAttempts'\nimport { getFieldsToSign } from './getFieldsToSign'\n\nexport type Result = {\n  exp?: number\n  token?: string\n  user?: User\n}\n\nexport type Arguments = {\n  collection: Collection\n  data: {\n    email: string\n    password: string\n  }\n  depth?: number\n  overrideAccess?: boolean\n  req: PayloadRequest\n  res?: Response\n  showHiddenFields?: boolean\n}\n\nasync function login<TSlug extends keyof GeneratedTypes['collections']>(\n  incomingArgs: Arguments,\n): Promise<Result & { user: GeneratedTypes['collections'][TSlug] }> {\n  let args = incomingArgs\n\n  try {\n    const shouldCommit = await initTransaction(args.req)\n\n    // /////////////////////////////////////\n    // beforeOperation - Collection\n    // /////////////////////////////////////\n\n    await args.collection.config.hooks.beforeOperation.reduce(async (priorHook, hook) => {\n      await priorHook\n\n      args =\n        (await hook({\n          args,\n          collection: args.collection?.config,\n          context: args.req.context,\n          operation: 'login',\n          req: args.req,\n        })) || args\n    }, Promise.resolve())\n\n    const {\n      collection: { config: collectionConfig },\n      data,\n      depth,\n      overrideAccess,\n      req,\n      req: {\n        fallbackLocale,\n        locale,\n        payload,\n        payload: { config, secret },\n      },\n      showHiddenFields,\n    } = args\n\n    // /////////////////////////////////////\n    // Login\n    // /////////////////////////////////////\n\n    const { email: unsanitizedEmail, password } = data\n\n    const email = unsanitizedEmail ? unsanitizedEmail.toLowerCase().trim() : null\n\n    let user = await payload.db.findOne<any>({\n      collection: collectionConfig.slug,\n      req,\n      where: { email: { equals: email.toLowerCase() } },\n    })\n\n    if (!user || (args.collection.config.auth.verify && user._verified === false)) {\n      throw new AuthenticationError(req.t)\n    }\n\n    if (user && isLocked(user.lockUntil)) {\n      throw new LockedAuth(req.t)\n    }\n\n    const authResult = await authenticateLocalStrategy({ doc: user, password })\n\n    user = sanitizeInternalFields(user)\n\n    const maxLoginAttemptsEnabled = args.collection.config.auth.maxLoginAttempts > 0\n\n    if (!authResult) {\n      if (maxLoginAttemptsEnabled) {\n        await incrementLoginAttempts({\n          collection: collectionConfig,\n          doc: user,\n          payload: req.payload,\n          req,\n        })\n      }\n\n      if (shouldCommit) await commitTransaction(req)\n\n      throw new AuthenticationError(req.t)\n    }\n\n    if (maxLoginAttemptsEnabled) {\n      await resetLoginAttempts({\n        collection: collectionConfig,\n        doc: user,\n        payload: req.payload,\n        req,\n      })\n    }\n\n    const fieldsToSign = getFieldsToSign({\n      collectionConfig,\n      email,\n      user,\n    })\n\n    await collectionConfig.hooks.beforeLogin.reduce(async (priorHook, hook) => {\n      await priorHook\n\n      user =\n        (await hook({\n          collection: args.collection?.config,\n          context: args.req.context,\n          req: args.req,\n          user,\n        })) || user\n    }, Promise.resolve())\n\n    const token = jwt.sign(fieldsToSign, secret, {\n      expiresIn: collectionConfig.auth.tokenExpiration,\n    })\n\n    if (args.res) {\n      const cookieOptions: CookieOptions = {\n        domain: undefined,\n        expires: getCookieExpiration(collectionConfig.auth.tokenExpiration),\n        httpOnly: true,\n        path: '/',\n        sameSite: collectionConfig.auth.cookies.sameSite,\n        secure: collectionConfig.auth.cookies.secure,\n      }\n\n      if (collectionConfig.auth.cookies.domain)\n        cookieOptions.domain = collectionConfig.auth.cookies.domain\n\n      args.res.cookie(`${config.cookiePrefix}-token`, token, cookieOptions)\n    }\n\n    req.user = user\n\n    // /////////////////////////////////////\n    // afterLogin - Collection\n    // /////////////////////////////////////\n\n    await collectionConfig.hooks.afterLogin.reduce(async (priorHook, hook) => {\n      await priorHook\n\n      user =\n        (await hook({\n          collection: args.collection?.config,\n          context: args.req.context,\n          req: args.req,\n          token,\n          user,\n        })) || user\n    }, Promise.resolve())\n\n    // /////////////////////////////////////\n    // afterRead - Fields\n    // /////////////////////////////////////\n\n    user = await afterRead({\n      collection: collectionConfig,\n      context: req.context,\n      depth,\n      doc: user,\n      fallbackLocale,\n      global: null,\n      locale,\n      overrideAccess,\n      req,\n      showHiddenFields,\n    })\n\n    // /////////////////////////////////////\n    // afterRead - Collection\n    // /////////////////////////////////////\n\n    await collectionConfig.hooks.afterRead.reduce(async (priorHook, hook) => {\n      await priorHook\n\n      user =\n        (await hook({\n          collection: args.collection?.config,\n          context: req.context,\n          doc: user,\n          req,\n        })) || user\n    }, Promise.resolve())\n\n    // /////////////////////////////////////\n    // afterRead - Collection\n    // /////////////////////////////////////\n\n    await collectionConfig.hooks.afterRead.reduce(async (priorHook, hook) => {\n      await priorHook\n\n      user =\n        (await hook({\n          collection: args.collection?.config,\n          context: req.context,\n          doc: user,\n          req,\n        })) || user\n    }, Promise.resolve())\n\n    let result: Result & { user: GeneratedTypes['collections'][TSlug] } = {\n      exp: (jwt.decode(token) as jwt.JwtPayload).exp,\n      token,\n      user,\n    }\n\n    // /////////////////////////////////////\n    // afterOperation - Collection\n    // /////////////////////////////////////\n\n    result = await buildAfterOperation<GeneratedTypes['collections'][TSlug]>({\n      args,\n      collection: args.collection?.config,\n      operation: 'login',\n      result,\n    })\n\n    if (collectionConfig.auth.removeTokenFromResponses) {\n      delete result.token\n    }\n\n    // /////////////////////////////////////\n    // Return results\n    // /////////////////////////////////////\n\n    if (shouldCommit) await commitTransaction(req)\n\n    return result\n  } catch (error: unknown) {\n    await killTransaction(args.req)\n    throw error\n  }\n}\n\nexport default login\n"],"names":["login","incomingArgs","args","shouldCommit","initTransaction","req","collection","config","hooks","beforeOperation","reduce","priorHook","hook","context","operation","Promise","resolve","collectionConfig","data","depth","overrideAccess","fallbackLocale","locale","payload","secret","showHiddenFields","email","unsanitizedEmail","password","toLowerCase","trim","user","db","findOne","slug","where","equals","auth","verify","_verified","AuthenticationError","t","isLocked","lockUntil","LockedAuth","authResult","authenticateLocalStrategy","doc","sanitizeInternalFields","maxLoginAttemptsEnabled","maxLoginAttempts","incrementLoginAttempts","commitTransaction","resetLoginAttempts","fieldsToSign","getFieldsToSign","beforeLogin","token","jwt","sign","expiresIn","tokenExpiration","res","cookieOptions","domain","undefined","expires","getCookieExpiration","httpOnly","path","sameSite","cookies","secure","cookie","cookiePrefix","afterLogin","afterRead","global","result","exp","decode","buildAfterOperation","removeTokenFromResponses","error","killTransaction"],"mappings":"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"}