当前位置: 首页 > ds >正文

通过 Node.js 搭配 Nodemailer 实现邮箱验证码发送

目录

一、整体思路

二、环境准备

三、代码详细步骤

1. 引入模块并初始化

2. 配置 Nodemailer 邮件服务

3. 定义发送验证码接口

4. 定义校验验证码接口

5. 启动服务器

四、补充说明

一、整体思路

  1. 接收邮箱地址请求
    用户在前端填写邮箱,发送到后端接口(如 /send-code)。

  2. 后端生成验证码并缓存
    后端生成随机验证码,存入 Redis 或内存(带过期时间)。

  3. 使用 Nodemailer 发邮件
    配置邮箱服务器,用 Nodemailer 发送邮件,将验证码发到用户邮箱。

  4. 用户提交验证码校验
    后端验证用户提交的验证码是否正确并且未过期。

二、环境准备

首先,需要安装必要的 Node.js 包:

npm install express nodemailer redis
  • express:搭建 HTTP 服务器。

  • nodemailer:发送邮件。

  • redis:存验证码,设置过期时间。

三、代码详细步骤

1. 引入模块并初始化

const express = require('express');
const nodemailer = require('nodemailer');
const redis = require('redis');
const { promisify } = require('util');const app = express();
app.use(express.json());// 连接 Redis
const redisClient = redis.createClient();
const setAsync = promisify(redisClient.set).bind(redisClient);
const getAsync = promisify(redisClient.get).bind(redisClient);

2. 配置 Nodemailer 邮件服务

可以选择 SMTP 邮件服务器,比如:

  • Gmail

  • QQ邮箱

  • 阿里云邮件推送

  • SendGrid

示例(以 QQ 邮箱为例):

const transporter = nodemailer.createTransport({host: 'smtp.qq.com',port: 465,secure: true, // true for 465, false for other portsauth: {user: '你的QQ邮箱@qq.com', // 发送方邮箱pass: '授权码'              // 不是密码!是SMTP授权码!}
});

注意:QQ邮箱需要开启 SMTP 服务,并拿到一个授权码,不能用普通密码。

3. 定义发送验证码接口

app.post('/send-code', async (req, res) => {const { email } = req.body;if (!email) {return res.status(400).send({ error: 'Email is required' });}// 生成6位随机验证码const code = Math.floor(100000 + Math.random() * 900000).toString();// 保存到 Redis,设置过期时间5分钟await setAsync(`verify:${email}`, code, 'EX', 300);// 发送邮件const mailOptions = {from: '你的QQ邮箱@qq.com',to: email,subject: '您的验证码',text: `您的验证码是 ${code},有效期5分钟,请勿泄露。`};try {await transporter.sendMail(mailOptions);res.send({ message: '验证码发送成功' });} catch (err) {console.error('发送失败', err);res.status(500).send({ error: '验证码发送失败' });}
});

4. 定义校验验证码接口

app.post('/verify-code', async (req, res) => {const { email, code } = req.body;if (!email || !code) {return res.status(400).send({ error: 'Email and code are required' });}const savedCode = await getAsync(`verify:${email}`);if (!savedCode) {return res.status(400).send({ error: '验证码已过期或不存在' });}if (savedCode === code) {return res.send({ message: '验证成功' });} else {return res.status(400).send({ error: '验证码错误' });}
});

5. 启动服务器

const PORT = process.env.PORT || 3000;
app.listen(PORT, () => {console.log(`服务器运行在 http://localhost:${PORT}`);
});

四、补充说明

  1. 限流控制(防止恶意刷验证码)

    • 比如一个邮箱1分钟内只能发送一次。

    • 可以在 Redis 里加一层锁或者计数器。

  2. 验证码加密

    • 验证码可以使用简单加密(例如哈希存储),提高安全性。

  3. 防止暴力破解

    • 连续错误输验证码次数超过3次,锁定一段时间。

  4. 生产环境发送 HTML 格式邮件

    • 邮件正文可以用 HTML 格式美化,比如加品牌 Logo、验证码提示等。

  5. 异步处理发送邮件

    • 邮件发送可以做成异步,或者通过消息队列(如 RabbitMQ)解耦,提高响应速度。

http://www.xdnf.cn/news/3005.html

相关文章:

  • 五、UI自动化测试05--PyTest框架
  • grafana/loki 设置日志保留时间
  • Cursor —— AI编辑器 使用详解
  • 【动态导通电阻】 GaN PiN二极管电导调制对动态 RON 的影响
  • 从Windows开发迁移到信创开发的指南:国产替代背景下的技术路径与实践
  • Webshell管理工具的流量特征
  • 桂链:使用Fabric的测试网络
  • 043-代码味道-循环依赖
  • LeetCode58_最后一个单词的长度
  • QT控件 参考Qt的PIMPL设计模式实现使用QWidget控件绘制3D饼状图表和3D柱状图表,使用QChartView绘制圆柱体图表
  • CORS跨域学习
  • opencv 模板匹配
  • [USACO08DEC] Hay For Sale S Java
  • React Native 太慢:kotlin-gradle-plugin-2.0.21-gradle76.jar 下载太慢
  • Code Complete代码大全20年纪念版附录书籍等
  • 归并排序排序总结
  • 某高端制造企业知识中枢升级,基于悦数 Graph RAG 打造工业级「故障排查最强大脑」
  • OceanBase数据库-学习笔记5-用户
  • 《系统分析师-第三阶段—总结(七)》
  • C++入门(缺省参数/函数/引用)
  • 组件轮播与样式结构重用实验
  • Linux《进程概念(中)》
  • 在Arduino U8g2库中显示中文的方法
  • 「Mac畅玩AIGC与多模态06」开发篇02 - 开发第一个知识库问答应用
  • 电流探头的创新应用与霍尔效应原理
  • word文档插入公式后行距变大怎么办?
  • 大模型入门
  • 码蹄集——进制输出、求最大公约数、最小公倍数
  • 【时时三省】(C语言基础)循环结构程序设计习题2
  • 如何从大规模点集中筛选出距离不小于指定值的点