Node.js验证码:从生成到验证的趣味探索之旅
📜 引言:为何需要验证码?
设想你经营着一家热门咖啡店,每日有大量顾客前来消费。突然有机器人不断下单,导致真实顾客难以买到饮品。验证码就如同咖啡店的‘人工甄别系统’,确保每个请求都来自真实的人类顾客。
在Web开发领域,验证码(CAPTCHA)主要发挥以下作用:
- ✅ 抵御机器人暴力破解 (例如登录密码的暴力尝试)
- ✅ 保护敏感接口 (像查询个人资料的接口)
- ✅ 减轻服务器负担 (过滤掉无效的请求)
验证码是现代Web应用中常见的安全机制,能有效防范机器人的恶意攻击。接下来,我们将深入探究Node.js中验证码的生成与验证过程,这就像一场充满趣味的探索之旅!
1. 验证码的基础原理 🧠
验证码(CAPTCHA)的全称是“Completely Automated Public Turing test to tell Computers and Humans Apart”(全自动区分计算机和人类的图灵测试)。其核心思想是:
“创建一种人类易于识别但计算机难以识别的测试”
验证码工作流程示意图
是
否
客户端请求验证码
服务器生成SVG验证码
转换为PNG图片
返回Base64编码图片
用户提交表单+验证码
服务器校验验证码
验证通过?
执行业务逻辑
返回错误提示
2. 技术栈筹备 🛠️
在本次示例中,运用了以下关键技术:
技术 | 作用 | 特点 |
---|---|---|
express |
Web框架 | 轻巧灵活,用于处理HTTP请求 |
svg-captcha |
生成SVG验证码 | 纯JS实现,无需额外编译 |
sharp |
图像处理 | 高性能的图片转换工具 |
express-session |
会话管理 | 用于存储验证码文本 |
3. 验证码生成详解 🎨
让我们剖析代码里验证码生成的部分:
3.1 生成SVG验证码
const svgCaptcha = require('svg-captcha');
const sharp = require('sharp');
const session = require('express-session');
// 配置session中间件
router.use(session({
secret: 'stu_xpx_200701',
resave: false,
saveUninitialized: true,
cookie: { secure: false } // 生产环境建议设置为true(HTTPS)
}));
// 生成验证码路由
router.get('/captcha', async (req, res) => {
// 创建包含4个字符的彩色验证码(排除易混淆字符)
const captcha = svgCaptcha.create({
size: 4, // 4个字符
ignoreChars: '0o1i', // 排除易混淆字符
noise: 2, // 干扰线数量
color: true // 彩色显示
});
// 存储验证码(带1分钟过期时间)
req.session.captcha = {
text: captcha.text.toLowerCase(), // 转为小写存储
expiresAt: Date.now() + 60000 // 当前时间+1分钟
};
💡 关键点:
参数 | 值 | 说明 |
---|---|---|
size | 4 | 验证码包含4个字符 |
ignoreChars | ‘0o1i’ | 排除数字0、字母o、数字1、字母i等易混淆字符 |
noise | 2 | 添加2条干扰线增加识别难度 |
color | true | 使用彩色而非黑白 |
3.2 转换为PNG格式
为何要将SVG转为PNG?
45% 30% 25% 图片格式选择原因 兼容性 安全性 性能
借助sharp
库进行转换:
try {
// 将SVG转为PNG(兼容更多浏览器)
const pngBuffer = await sharp(Buffer.from(captcha.data))
.png()
.toBuffer();
// 返回Base64编码的图片
res.type('png').json({
code: 0,
data: `data:image/png;base64,${pngBuffer.toString('base64')}`,
msg: "验证码图片获取成功"
});
} catch (err) {
console.error('生成PNG失败:', err);
res.status(500).json({ code: -1, msg: "生成验证码失败" });
}
🔧 技术细节:
Buffer.from()
将SVG字符串转为二进制数据sharp()
进行图片格式转换toString('base64')
生成前端可直接显示的DataURL
3.3 存储验证码信息
我们不仅存储验证码文本,还存储过期时间:
req.session.captcha = {
text: captcha.text.toLowerCase(), // 验证码文本
expiresAt: Date.now() + 1 * 60 * 1000 // 1分钟后过期
};
这种设计比单纯存储文本更安全,原因在于:
- 强制验证码有时效性
- 防止重放攻击
- 自动清理过期验证码
4. 验证码验证流程 🔍
当用户提交表单时,需要对验证码进行验证:
4.1 验证步骤分解
用户 服务器 提交表单(姓名,身份证,验证码) 检查必填字段 检查验证码是否存在 检查验证码是否过期 比较验证码文本 返回验证结果 用户
服务器
4.2 关键验证代码
router.post("/query", async (req, res) => {
// 检查必填参数
if (!req.body.studentname || !req.body.id_card || !req.body.code) {
return res.status(400).json({ code: -1, msg: "缺少必要参数" });
}
// 验证码三重校验
if (!req.session.captcha) {
return res.status(400).json({ code: -1, msg: "验证码已过期" });
}
if (req.session.captcha.expiresAt < Date.now()) {
return res.status(400).json({ code: -1, msg: "验证码已过期" });
}
if (req.session.captcha.text !== req.body.code.toLowerCase()) {
return res.status(400).json({ code: -1, msg: "验证码错误" });
}
// 验证通过后立即销毁验证码(一次性使用)
delete req.session.captcha;
// 验证码通过后的代码逻辑...
🛡️ 安全策略:
- 时间过期机制 :1分钟后自动失效
- 大小写无关校验 :统一转为小写比较
- 一次性使用 :验证后立即删除session存储
4.3 验证码的一次性使用
验证通过后立即删除验证码,避免重复使用:
// 验证通过后删除验证码(一次性使用)
delete req.session.captcha;
5. 成果查验
按照上述操作就能生成一个png地址,我们可以对其有效性进行检查,步骤如下:
生成的地址如:
返回地址:data:image/png;base64,${pngBuffer.toString('base64')}
例如:
