Spaces:
Sleeping
Sleeping
Upload 18 files
Browse files- config/auth.js +15 -5
- config/db.js +15 -10
- controllers/authController.js +19 -3
- middleware/auth.js +7 -3
- models/User.js +14 -4
- server.js +14 -0
config/auth.js
CHANGED
@@ -1,12 +1,22 @@
|
|
1 |
const jwt = require('jsonwebtoken');
|
|
|
2 |
|
3 |
// 生成 JWT Token
|
4 |
const generateToken = (userId) => {
|
5 |
-
|
6 |
-
|
7 |
-
|
8 |
-
|
9 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
10 |
};
|
11 |
|
12 |
module.exports = { generateToken };
|
|
|
1 |
const jwt = require('jsonwebtoken');
|
2 |
+
const logger = require('../utils/logger');
|
3 |
|
4 |
// 生成 JWT Token
|
5 |
const generateToken = (userId) => {
|
6 |
+
try {
|
7 |
+
// 确保 userId 是字符串类型
|
8 |
+
const userIdStr = String(userId);
|
9 |
+
const token = jwt.sign(
|
10 |
+
{ id: userIdStr },
|
11 |
+
process.env.JWT_SECRET,
|
12 |
+
{ expiresIn: process.env.JWT_EXPIRE || '24h' }
|
13 |
+
);
|
14 |
+
logger.info(`为用户 ${userIdStr} 生成令牌成功`);
|
15 |
+
return token;
|
16 |
+
} catch (error) {
|
17 |
+
logger.error(`生成令牌失败: ${error.message}`);
|
18 |
+
throw error;
|
19 |
+
}
|
20 |
};
|
21 |
|
22 |
module.exports = { generateToken };
|
config/db.js
CHANGED
@@ -22,25 +22,30 @@ const connectDB = async () => {
|
|
22 |
const createDefaultAdmin = async () => {
|
23 |
try {
|
24 |
const User = require('../models/User');
|
25 |
-
|
26 |
-
|
|
|
|
|
|
|
|
|
|
|
27 |
// 检查是否已存在管理员用户
|
28 |
const adminExists = await User.findOne({ username: process.env.ADMIN_USERNAME });
|
29 |
|
30 |
-
if (!adminExists
|
31 |
-
//
|
32 |
-
const
|
33 |
-
|
34 |
-
await User.create({
|
35 |
username: process.env.ADMIN_USERNAME,
|
36 |
-
password:
|
37 |
isAdmin: true
|
38 |
});
|
39 |
|
40 |
-
logger.info(
|
|
|
|
|
41 |
}
|
42 |
} catch (error) {
|
43 |
-
logger.error(
|
44 |
}
|
45 |
};
|
46 |
|
|
|
22 |
const createDefaultAdmin = async () => {
|
23 |
try {
|
24 |
const User = require('../models/User');
|
25 |
+
|
26 |
+
// 先检查环境变量
|
27 |
+
if (!process.env.ADMIN_USERNAME || !process.env.ADMIN_PASSWORD) {
|
28 |
+
logger.warn('管理员账户环境变量未设置,跳过默认管理员创建');
|
29 |
+
return;
|
30 |
+
}
|
31 |
+
|
32 |
// 检查是否已存在管理员用户
|
33 |
const adminExists = await User.findOne({ username: process.env.ADMIN_USERNAME });
|
34 |
|
35 |
+
if (!adminExists) {
|
36 |
+
// 不手动哈希密码,让模型的中间件处理加密
|
37 |
+
const admin = await User.create({
|
|
|
|
|
38 |
username: process.env.ADMIN_USERNAME,
|
39 |
+
password: process.env.ADMIN_PASSWORD, // 直接使用原始密码
|
40 |
isAdmin: true
|
41 |
});
|
42 |
|
43 |
+
logger.info(`默认管理员创建成功: ${admin._id}`);
|
44 |
+
} else {
|
45 |
+
logger.info('管理员账户已存在,无需创建');
|
46 |
}
|
47 |
} catch (error) {
|
48 |
+
logger.error(`创建默认管理员出错: ${error.message}`);
|
49 |
}
|
50 |
};
|
51 |
|
controllers/authController.js
CHANGED
@@ -1,6 +1,7 @@
|
|
1 |
const asyncHandler = require('express-async-handler');
|
2 |
const User = require('../models/User');
|
3 |
const { generateToken } = require('../config/auth');
|
|
|
4 |
|
5 |
// @desc 用户注册
|
6 |
// @route POST /api/auth/register
|
@@ -41,18 +42,33 @@ const registerUser = asyncHandler(async (req, res) => {
|
|
41 |
const loginUser = asyncHandler(async (req, res) => {
|
42 |
const { username, password } = req.body;
|
43 |
|
|
|
|
|
44 |
// 查找用户
|
45 |
const user = await User.findOne({ username });
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
46 |
|
47 |
-
|
48 |
-
|
|
|
|
|
49 |
res.json({
|
50 |
_id: user._id,
|
51 |
username: user.username,
|
52 |
isAdmin: user.isAdmin,
|
53 |
-
token:
|
54 |
});
|
55 |
} else {
|
|
|
56 |
res.status(401);
|
57 |
throw new Error('用户名或密码错误');
|
58 |
}
|
|
|
1 |
const asyncHandler = require('express-async-handler');
|
2 |
const User = require('../models/User');
|
3 |
const { generateToken } = require('../config/auth');
|
4 |
+
const logger = require('../utils/logger');
|
5 |
|
6 |
// @desc 用户注册
|
7 |
// @route POST /api/auth/register
|
|
|
42 |
const loginUser = asyncHandler(async (req, res) => {
|
43 |
const { username, password } = req.body;
|
44 |
|
45 |
+
logger.info(`登录尝试: ${username}`);
|
46 |
+
|
47 |
// 查找用户
|
48 |
const user = await User.findOne({ username });
|
49 |
+
|
50 |
+
if (!user) {
|
51 |
+
logger.warn(`用户不存在: ${username}`);
|
52 |
+
res.status(401);
|
53 |
+
throw new Error('用户名或密码错误');
|
54 |
+
}
|
55 |
+
|
56 |
+
// 检查密码
|
57 |
+
const isMatch = await user.matchPassword(password);
|
58 |
+
logger.info(`密码匹配结果: ${isMatch}`);
|
59 |
|
60 |
+
if (isMatch) {
|
61 |
+
const token = generateToken(user._id);
|
62 |
+
logger.info(`登录成功: ${username}`);
|
63 |
+
|
64 |
res.json({
|
65 |
_id: user._id,
|
66 |
username: user.username,
|
67 |
isAdmin: user.isAdmin,
|
68 |
+
token: token,
|
69 |
});
|
70 |
} else {
|
71 |
+
logger.warn(`密码不匹配: ${username}`);
|
72 |
res.status(401);
|
73 |
throw new Error('用户名或密码错误');
|
74 |
}
|
middleware/auth.js
CHANGED
@@ -1,6 +1,7 @@
|
|
1 |
const jwt = require('jsonwebtoken');
|
2 |
const asyncHandler = require('express-async-handler');
|
3 |
const User = require('../models/User');
|
|
|
4 |
|
5 |
// 保护路由 - 验证 JWT Token
|
6 |
const protect = asyncHandler(async (req, res, next) => {
|
@@ -11,26 +12,29 @@ const protect = asyncHandler(async (req, res, next) => {
|
|
11 |
try {
|
12 |
// 获取 token
|
13 |
token = req.headers.authorization.split(' ')[1];
|
|
|
14 |
|
15 |
// 验证 token
|
16 |
const decoded = jwt.verify(token, process.env.JWT_SECRET);
|
|
|
17 |
|
18 |
// 获取用户并添加到请求对象中,不包含密码
|
19 |
req.user = await User.findById(decoded.id).select('-password');
|
20 |
|
21 |
if (!req.user) {
|
|
|
22 |
res.status(401);
|
23 |
throw new Error('未授权,用户不存在');
|
24 |
}
|
25 |
|
26 |
next();
|
27 |
} catch (error) {
|
|
|
28 |
res.status(401);
|
29 |
throw new Error('未授权,token 无效');
|
30 |
}
|
31 |
-
}
|
32 |
-
|
33 |
-
if (!token) {
|
34 |
res.status(401);
|
35 |
throw new Error('未授权,未提供 token');
|
36 |
}
|
|
|
1 |
const jwt = require('jsonwebtoken');
|
2 |
const asyncHandler = require('express-async-handler');
|
3 |
const User = require('../models/User');
|
4 |
+
const logger = require('../utils/logger');
|
5 |
|
6 |
// 保护路由 - 验证 JWT Token
|
7 |
const protect = asyncHandler(async (req, res, next) => {
|
|
|
12 |
try {
|
13 |
// 获取 token
|
14 |
token = req.headers.authorization.split(' ')[1];
|
15 |
+
logger.info(`验证令牌: ${token.substring(0, 15)}...`);
|
16 |
|
17 |
// 验证 token
|
18 |
const decoded = jwt.verify(token, process.env.JWT_SECRET);
|
19 |
+
logger.info(`令牌有效,用户ID: ${decoded.id}`);
|
20 |
|
21 |
// 获取用户并添加到请求对象中,不包含密码
|
22 |
req.user = await User.findById(decoded.id).select('-password');
|
23 |
|
24 |
if (!req.user) {
|
25 |
+
logger.warn(`令牌有效但用户不存在: ${decoded.id}`);
|
26 |
res.status(401);
|
27 |
throw new Error('未授权,用户不存在');
|
28 |
}
|
29 |
|
30 |
next();
|
31 |
} catch (error) {
|
32 |
+
logger.error(`令牌验证失败: ${error.message}`);
|
33 |
res.status(401);
|
34 |
throw new Error('未授权,token 无效');
|
35 |
}
|
36 |
+
} else {
|
37 |
+
logger.warn(`未提供认证令牌: ${req.originalUrl}`);
|
|
|
38 |
res.status(401);
|
39 |
throw new Error('未授权,未提供 token');
|
40 |
}
|
models/User.js
CHANGED
@@ -26,17 +26,27 @@ const userSchema = new mongoose.Schema(
|
|
26 |
|
27 |
// 密码匹配方法
|
28 |
userSchema.methods.matchPassword = async function (enteredPassword) {
|
29 |
-
|
|
|
|
|
|
|
|
|
|
|
30 |
};
|
31 |
|
32 |
// 保存前加密密码
|
33 |
userSchema.pre('save', async function (next) {
|
34 |
if (!this.isModified('password')) {
|
35 |
-
next();
|
36 |
}
|
37 |
|
38 |
-
|
39 |
-
|
|
|
|
|
|
|
|
|
|
|
40 |
});
|
41 |
|
42 |
const User = mongoose.model('User', userSchema);
|
|
|
26 |
|
27 |
// 密码匹配方法
|
28 |
userSchema.methods.matchPassword = async function (enteredPassword) {
|
29 |
+
try {
|
30 |
+
return await bcrypt.compare(enteredPassword, this.password);
|
31 |
+
} catch (error) {
|
32 |
+
console.error('密码比较出错:', error);
|
33 |
+
return false;
|
34 |
+
}
|
35 |
};
|
36 |
|
37 |
// 保存前加密密码
|
38 |
userSchema.pre('save', async function (next) {
|
39 |
if (!this.isModified('password')) {
|
40 |
+
return next(); // 修复:确保不重复加密
|
41 |
}
|
42 |
|
43 |
+
try {
|
44 |
+
const salt = await bcrypt.genSalt(10);
|
45 |
+
this.password = await bcrypt.hash(this.password, salt);
|
46 |
+
next();
|
47 |
+
} catch (error) {
|
48 |
+
next(error);
|
49 |
+
}
|
50 |
});
|
51 |
|
52 |
const User = mongoose.model('User', userSchema);
|
server.js
CHANGED
@@ -10,6 +10,20 @@ const logger = require('./utils/logger');
|
|
10 |
// 加载环境变量
|
11 |
dotenv.config();
|
12 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
13 |
// 初始化 Express 应用
|
14 |
const app = express();
|
15 |
|
|
|
10 |
// 加载环境变量
|
11 |
dotenv.config();
|
12 |
|
13 |
+
// 验证关键环境变量
|
14 |
+
const requiredEnvVars = ['MONGODB_URI', 'JWT_SECRET', 'ADMIN_USERNAME', 'ADMIN_PASSWORD'];
|
15 |
+
const missingEnvVars = requiredEnvVars.filter(envVar => !process.env[envVar]);
|
16 |
+
|
17 |
+
if (missingEnvVars.length > 0) {
|
18 |
+
logger.error(`缺少必要的环境变量: ${missingEnvVars.join(', ')}`);
|
19 |
+
process.exit(1);
|
20 |
+
}
|
21 |
+
|
22 |
+
logger.info('环境变量加载成功');
|
23 |
+
logger.info(`NODE_ENV: ${process.env.NODE_ENV}`);
|
24 |
+
logger.info(`ADMIN_USERNAME: ${process.env.ADMIN_USERNAME}`);
|
25 |
+
logger.info(`JWT_SECRET: ${process.env.JWT_SECRET ? '已设置' : '未设置'}`);
|
26 |
+
|
27 |
// 初始化 Express 应用
|
28 |
const app = express();
|
29 |
|