Source: app/api/auth/[...nextauth]/route.js

  1. import clientPromise from "@/lib/mongodb";
  2. import { MongoDBAdapter } from "@auth/mongodb-adapter";
  3. import NextAuth from "next-auth";
  4. import EmailProvider from "next-auth/providers/email";
  5. import { createTransport } from "nodemailer";
  6. async function sendVerificationRequest(params) {
  7. const { identifier, url, provider, theme } = params;
  8. const allowedEmails = process.env.ALLOWED_EMAILS?.split(",") ?? [];
  9. if (!allowedEmails.includes(identifier)) {
  10. throw new Error("Email could not be sent");
  11. }
  12. const { host } = new URL(url);
  13. // NOTE: You are not required to use `nodemailer`, use whatever you want.
  14. const transport = createTransport(provider.server);
  15. const result = await transport.sendMail({
  16. to: identifier,
  17. from: provider.from,
  18. subject: `Sign in to ${host}`,
  19. text: text({ url, host }),
  20. html: html({ url, host, theme }),
  21. });
  22. const failed = result.rejected.concat(result.pending).filter(Boolean);
  23. if (failed.length) {
  24. throw new Error(`Email(s) (${failed.join(", ")}) could not be sent`);
  25. }
  26. }
  27. /**
  28. * Email HTML body
  29. * Insert invisible space into domains from being turned into a hyperlink by email
  30. * clients like Outlook and Apple mail, as this is confusing because it seems
  31. * like they are supposed to click on it to sign in.
  32. *
  33. * @note We don't add the email address to avoid needing to escape it, if you do, remember to sanitize it!
  34. */
  35. function html(params) {
  36. const { url, host, theme } = params;
  37. const escapedHost = host.replace(/\./g, "​.");
  38. const brandColor = theme.brandColor || "#346df1";
  39. const color = {
  40. background: "#f9f9f9",
  41. text: "#444",
  42. mainBackground: "#fff",
  43. buttonBackground: brandColor,
  44. buttonBorder: brandColor,
  45. buttonText: theme.buttonText || "#fff",
  46. };
  47. return `
  48. <body style="background: ${color.background};">
  49. <table width="100%" border="0" cellspacing="20" cellpadding="0"
  50. style="background: ${color.mainBackground}; max-width: 600px; margin: auto; border-radius: 10px;">
  51. <tr>
  52. <td align="center"
  53. style="padding: 10px 0px; font-size: 22px; font-family: Helvetica, Arial, sans-serif; color: ${color.text};">
  54. Sign in to <strong>${escapedHost}</strong>
  55. </td>
  56. </tr>
  57. <tr>
  58. <td align="center" style="padding: 20px 0;">
  59. <table border="0" cellspacing="0" cellpadding="0">
  60. <tr>
  61. <td align="center" style="border-radius: 5px;" bgcolor="${color.buttonBackground}"><a href="${url}"
  62. target="_blank"
  63. style="font-size: 18px; font-family: Helvetica, Arial, sans-serif; color: ${color.buttonText}; text-decoration: none; border-radius: 5px; padding: 10px 20px; border: 1px solid ${color.buttonBorder}; display: inline-block; font-weight: bold;">Sign
  64. in</a></td>
  65. </tr>
  66. </table>
  67. </td>
  68. </tr>
  69. <tr>
  70. <td align="center"
  71. style="padding: 0px 0px 10px 0px; font-size: 16px; line-height: 22px; font-family: Helvetica, Arial, sans-serif; color: ${color.text};">
  72. If you did not request this email you can safely ignore it.
  73. </td>
  74. </tr>
  75. </table>
  76. </body>
  77. `;
  78. }
  79. /** Email Text body (fallback for email clients that don't render HTML, e.g. feature phones) */
  80. function text({ url, host }) {
  81. return `Sign in to ${host}\n${url}\n\n`;
  82. }
  83. export const authOptions = {
  84. secret: process.env.AUTH_SECRET,
  85. adapter: MongoDBAdapter(clientPromise),
  86. providers: [
  87. EmailProvider({
  88. server: {
  89. host: process.env.EMAIL_SERVER_HOST,
  90. port: process.env.EMAIL_SERVER_PORT,
  91. auth: {
  92. user: process.env.EMAIL_SERVER_USER,
  93. pass: process.env.EMAIL_SERVER_PASSWORD,
  94. },
  95. },
  96. from: process.env.EMAIL_FROM,
  97. sendVerificationRequest,
  98. }),
  99. ],
  100. };
  101. const handler = NextAuth(authOptions);
  102. export { handler as GET, handler as POST };