init
This commit is contained in:
97
api/server.js
Normal file
97
api/server.js
Normal file
@@ -0,0 +1,97 @@
|
||||
const express = require('express');
|
||||
const bodyParser = require('body-parser');
|
||||
const cors = require('cors');
|
||||
const nodemailer = require('nodemailer');
|
||||
const dotenv = require('dotenv');
|
||||
|
||||
// 加载环境变量
|
||||
dotenv.config();
|
||||
|
||||
const app = express();
|
||||
const port = process.env.PORT || 3001;
|
||||
|
||||
// 中间件
|
||||
app.use(cors());
|
||||
app.use(bodyParser.json());
|
||||
|
||||
// 创建Nodemailer传输器
|
||||
const transporter = nodemailer.createTransport({
|
||||
service: process.env.EMAIL_SERVICE || 'smtp', // 邮件服务商,例如:'gmail', 'qq', '163'等
|
||||
host: process.env.EMAIL_HOST || 'smtp.163.com', // SMTP服务器
|
||||
port: parseInt(process.env.EMAIL_PORT || '587'),
|
||||
secure: process.env.EMAIL_SECURE === 'true', // true for 465, false for other ports
|
||||
auth: {
|
||||
user: process.env.EMAIL_USER || '', // 发送方邮箱
|
||||
pass: process.env.EMAIL_PASS || '', // 授权码或密码
|
||||
},
|
||||
});
|
||||
|
||||
// 存储IP地址最后发送邮件的时间
|
||||
const lastEmailSentTime = new Map();
|
||||
|
||||
// 速率限制中间件 - 60秒内只能发送一封邮件
|
||||
const rateLimitMiddleware = (req, res, next) => {
|
||||
const ip = req.ip || req.connection.remoteAddress;
|
||||
const currentTime = Date.now();
|
||||
const lastTime = lastEmailSentTime.get(ip) || 0;
|
||||
const timeDiff = currentTime - lastTime;
|
||||
|
||||
// 如果距离上次发送不足60秒,则拒绝请求
|
||||
if (lastTime && timeDiff < 60000) {
|
||||
const remainingTime = Math.ceil((60000 - timeDiff) / 1000);
|
||||
return res.status(429).json({
|
||||
error: `请求过于频繁,请在${remainingTime}秒后再试`,
|
||||
remainingTime
|
||||
});
|
||||
}
|
||||
|
||||
// 更新最后发送时间并继续处理请求
|
||||
lastEmailSentTime.set(ip, currentTime);
|
||||
next();
|
||||
};
|
||||
|
||||
// 邮件发送接口
|
||||
app.post('/api/send-email', rateLimitMiddleware, async (req, res) => {
|
||||
const { name, email, subject, message } = req.body;
|
||||
|
||||
if (!name || !email || !message) {
|
||||
return res.status(400).json({ error: '请填写所有必填字段' });
|
||||
}
|
||||
|
||||
try {
|
||||
// 邮件选项
|
||||
const mailOptions = {
|
||||
from: `"个人网站联系表单" <${process.env.EMAIL_USER}>`,
|
||||
to: process.env.EMAIL_RECIPIENT || process.env.EMAIL_USER, // 接收方邮箱
|
||||
subject: subject ? `新的联系表单消息: ${subject}` : '新的联系表单消息',
|
||||
html: `
|
||||
<h3>您有一条新的联系表单消息</h3>
|
||||
<p><strong>姓名:</strong> ${name}</p>
|
||||
<p><strong>邮箱:</strong> ${email}</p>
|
||||
<p><strong>主题:</strong> ${subject || '无'}</p>
|
||||
<p><strong>消息:</strong></p>
|
||||
<p>${message.replace(/\n/g, '<br>')}</p>
|
||||
`,
|
||||
// 回复选项,回复时会发送到表单提交者的邮箱
|
||||
replyTo: email
|
||||
};
|
||||
|
||||
// 发送邮件
|
||||
await transporter.sendMail(mailOptions);
|
||||
|
||||
return res.status(200).json({ success: true, message: '邮件已成功发送' });
|
||||
} catch (error) {
|
||||
console.error('发送邮件时出错:', error);
|
||||
return res.status(500).json({ error: '发送邮件失败,请稍后再试' });
|
||||
}
|
||||
});
|
||||
|
||||
// 测试路由
|
||||
app.get('/api/test', (req, res) => {
|
||||
res.json({ message: 'API服务器运行正常' });
|
||||
});
|
||||
|
||||
// 启动服务器
|
||||
app.listen(port, () => {
|
||||
console.log(`邮件服务API正在端口 ${port} 上运行`);
|
||||
});
|
||||
Reference in New Issue
Block a user