payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
129 lines (128 loc) • 16.5 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "default", {
enumerable: true,
get: function() {
return _default;
}
});
const _crypto = /*#__PURE__*/ _interop_require_default(require("crypto"));
const _utils = require("../../collections/operations/utils");
const _errors = require("../../errors");
const _commitTransaction = require("../../utilities/commitTransaction");
const _initTransaction = require("../../utilities/initTransaction");
const _killTransaction = require("../../utilities/killTransaction");
function _interop_require_default(obj) {
return obj && obj.__esModule ? obj : {
default: obj
};
}
async function forgotPassword(incomingArgs) {
if (!Object.prototype.hasOwnProperty.call(incomingArgs.data, 'email')) {
throw new _errors.APIError('Missing email.', 400);
}
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: 'forgotPassword',
req: args.req
}) || args;
}, Promise.resolve());
const { collection: { config: collectionConfig }, data, disableEmail, expiration, req: { payload: { config, emailOptions, sendEmail: email }, payload, t }, req } = args;
// /////////////////////////////////////
// Forget password
// /////////////////////////////////////
let token = _crypto.default.randomBytes(20);
token = token.toString('hex');
if (!data.email) {
throw new _errors.APIError('Missing email.');
}
let user = await payload.db.findOne({
collection: collectionConfig.slug,
req,
where: {
email: {
equals: data.email.toLowerCase()
}
}
});
if (!user) return null;
user.resetPasswordToken = token;
user.resetPasswordExpiration = new Date(expiration || Date.now() + 3600000).toISOString() // 1 hour
;
user = await payload.update({
id: user.id,
collection: collectionConfig.slug,
data: user,
req
});
if (!disableEmail) {
const serverURL = config.serverURL !== null && config.serverURL !== '' ? config.serverURL : `${req.protocol}://${req.get('host')}`;
let html = `${t('authentication:youAreReceivingResetPassword')}
<a href="${serverURL}${config.routes.admin}/reset/${token}">
${serverURL}${config.routes.admin}/reset/${token}
</a>
${t('authentication:youDidNotRequestPassword')}`;
if (typeof collectionConfig.auth.forgotPassword.generateEmailHTML === 'function') {
html = await collectionConfig.auth.forgotPassword.generateEmailHTML({
req,
token,
user
});
}
let subject = t('authentication:resetYourPassword');
if (typeof collectionConfig.auth.forgotPassword.generateEmailSubject === 'function') {
subject = await collectionConfig.auth.forgotPassword.generateEmailSubject({
req,
token,
user
});
}
// eslint-disable-next-line @typescript-eslint/no-floating-promises
email({
from: `"${emailOptions.fromName}" <${emailOptions.fromAddress}>`,
html,
subject,
to: data.email
});
}
// /////////////////////////////////////
// afterForgotPassword - Collection
// /////////////////////////////////////
await collectionConfig.hooks.afterForgotPassword.reduce(async (priorHook, hook)=>{
await priorHook;
await hook({
args,
collection: args.collection?.config,
context: req.context
});
}, Promise.resolve());
// /////////////////////////////////////
// afterOperation - Collection
// /////////////////////////////////////
token = await (0, _utils.buildAfterOperation)({
args,
collection: args.collection?.config,
operation: 'forgotPassword',
result: token
});
if (shouldCommit) await (0, _commitTransaction.commitTransaction)(req);
return token;
} catch (error) {
await (0, _killTransaction.killTransaction)(args.req);
throw error;
}
}
const _default = forgotPassword;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/auth/operations/forgotPassword.ts"],"sourcesContent":["import crypto from 'crypto'\n\nimport type { Collection } from '../../collections/config/types'\nimport type { PayloadRequest } from '../../express/types'\n\nimport { buildAfterOperation } from '../../collections/operations/utils'\nimport { APIError } from '../../errors'\nimport { commitTransaction } from '../../utilities/commitTransaction'\nimport { initTransaction } from '../../utilities/initTransaction'\nimport { killTransaction } from '../../utilities/killTransaction'\n\nexport type Arguments = {\n  collection: Collection\n  data: {\n    [key: string]: unknown\n    email: string\n  }\n  disableEmail?: boolean\n  expiration?: number\n  req: PayloadRequest\n}\n\nexport type Result = string\n\nasync function forgotPassword(incomingArgs: Arguments): Promise<null | string> {\n  if (!Object.prototype.hasOwnProperty.call(incomingArgs.data, 'email')) {\n    throw new APIError('Missing email.', 400)\n  }\n\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: 'forgotPassword',\n          req: args.req,\n        })) || args\n    }, Promise.resolve())\n\n    const {\n      collection: { config: collectionConfig },\n      data,\n      disableEmail,\n      expiration,\n      req: {\n        payload: { config, emailOptions, sendEmail: email },\n        payload,\n        t,\n      },\n      req,\n    } = args\n\n    // /////////////////////////////////////\n    // Forget password\n    // /////////////////////////////////////\n\n    let token: Buffer | string = crypto.randomBytes(20)\n    token = token.toString('hex')\n\n    type UserDoc = {\n      id: number | string\n      resetPasswordExpiration?: string\n      resetPasswordToken?: string\n    }\n\n    if (!data.email) {\n      throw new APIError('Missing email.')\n    }\n\n    let user = await payload.db.findOne<UserDoc>({\n      collection: collectionConfig.slug,\n      req,\n      where: { email: { equals: data.email.toLowerCase() } },\n    })\n\n    if (!user) return null\n\n    user.resetPasswordToken = token\n    user.resetPasswordExpiration = new Date(expiration || Date.now() + 3600000).toISOString() // 1 hour\n\n    user = await payload.update({\n      id: user.id,\n      collection: collectionConfig.slug,\n      data: user,\n      req,\n    })\n\n    if (!disableEmail) {\n      const serverURL =\n        config.serverURL !== null && config.serverURL !== ''\n          ? config.serverURL\n          : `${req.protocol}://${req.get('host')}`\n\n      let html = `${t('authentication:youAreReceivingResetPassword')}\n    <a href=\"${serverURL}${config.routes.admin}/reset/${token}\">\n     ${serverURL}${config.routes.admin}/reset/${token}\n    </a>\n    ${t('authentication:youDidNotRequestPassword')}`\n\n      if (typeof collectionConfig.auth.forgotPassword.generateEmailHTML === 'function') {\n        html = await collectionConfig.auth.forgotPassword.generateEmailHTML({\n          req,\n          token,\n          user,\n        })\n      }\n\n      let subject = t('authentication:resetYourPassword')\n\n      if (typeof collectionConfig.auth.forgotPassword.generateEmailSubject === 'function') {\n        subject = await collectionConfig.auth.forgotPassword.generateEmailSubject({\n          req,\n          token,\n          user,\n        })\n      }\n\n      // eslint-disable-next-line @typescript-eslint/no-floating-promises\n      email({\n        from: `\"${emailOptions.fromName}\" <${emailOptions.fromAddress}>`,\n        html,\n        subject,\n        to: data.email,\n      })\n    }\n\n    // /////////////////////////////////////\n    // afterForgotPassword - Collection\n    // /////////////////////////////////////\n\n    await collectionConfig.hooks.afterForgotPassword.reduce(async (priorHook, hook) => {\n      await priorHook\n      await hook({ args, collection: args.collection?.config, context: req.context })\n    }, Promise.resolve())\n\n    // /////////////////////////////////////\n    // afterOperation - Collection\n    // /////////////////////////////////////\n\n    token = await buildAfterOperation({\n      args,\n      collection: args.collection?.config,\n      operation: 'forgotPassword',\n      result: token,\n    })\n\n    if (shouldCommit) await commitTransaction(req)\n\n    return token\n  } catch (error: unknown) {\n    await killTransaction(args.req)\n    throw error\n  }\n}\n\nexport default forgotPassword\n"],"names":["forgotPassword","incomingArgs","Object","prototype","hasOwnProperty","call","data","APIError","args","shouldCommit","initTransaction","req","collection","config","hooks","beforeOperation","reduce","priorHook","hook","context","operation","Promise","resolve","collectionConfig","disableEmail","expiration","payload","emailOptions","sendEmail","email","t","token","crypto","randomBytes","toString","user","db","findOne","slug","where","equals","toLowerCase","resetPasswordToken","resetPasswordExpiration","Date","now","toISOString","update","id","serverURL","protocol","get","html","routes","admin","auth","generateEmailHTML","subject","generateEmailSubject","from","fromName","fromAddress","to","afterForgotPassword","buildAfterOperation","result","commitTransaction","error","killTransaction"],"mappings":"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"}