Files
vless-api/app.js
aixianling b958beaf2f feat(api): 新增客户端添加功能并优化路由加载
- 新增客户端添加 API,实现客户端信息的添加和处理
- 重构路由加载函数,支持递归加载子目录中的路由文件
- 添加日志记录,便于调试和监控
- 优化 HTTP 请求处理,增加错误处理和状态码检查
2025-02-26 14:16:26 +08:00

102 lines
3.1 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

require("dotenv").config();
const Koa = require("koa");
const Router = require("koa-router");
const jwt = require("jsonwebtoken");
const koaJwt = require("koa-jwt");
const fs = require("fs");
const path = require("path");
const bodyParser = require("koa-bodyparser");
const verifyThirdPartyToken = require("./auth/verifyThirdPartyToken");
const app = new Koa();
app.use(bodyParser()); // 添加在路由中间件之前
const router = new Router();
// 自动加载API路由函数
const loadAPIRoutes = (baseDir = 'api', baseRoute = '/api') => {
const scanDirectory = (dir, routePrefix) => {
const entries = fs.readdirSync(dir, { withFileTypes: true });
entries.forEach(entry => {
const fullPath = path.join(dir, entry.name);
const relativePath = path.relative(baseDir, dir);
// 构建路由路径:工程目录结构 -> URL路径
const routePath = path.join(
routePrefix,
relativePath,
entry.name.replace(/\.js$/, '')
).replace(/\\/g, '/'); // Windows路径转URL路径
if (entry.isDirectory()) {
scanDirectory(fullPath, routePrefix); // 递归扫描子目录
} else if (
entry.isFile() &&
path.extname(entry.name) === '.js' &&
entry.name !== 'index.js'
) {
registerRoute(fullPath, routePath);
}
});
};
// 路由注册器支持扩展其他HTTP方法
const registerRoute = (filePath, routePath) => {
try {
const handler = require(filePath);
router.post(routePath, async ctx => await handler(ctx));
console.log(`[Route] POST ${routePath} -> ${filePath}`);
} catch (err) {
console.error(`[Error] Failed to load route ${routePath}:`, err);
}
};
// 初始化扫描
scanDirectory(path.join(__dirname, baseDir), baseRoute);
};
// 公开路由
router.get("/public", (ctx) => {
ctx.body = "Public content";
});
// 自定义中间件解析并验证第三方Token
app.use(async (ctx, next) => {
const authHeader = ctx.headers.authorization;
if (authHeader && authHeader.startsWith('Bearer ')) {
const thirdPartyToken = authHeader.split(' ')[1];
try {
// 这里假设第三方Token可以通过某种方式验证并转换为JWT Token
const decoded = verifyThirdPartyToken(thirdPartyToken); // 假设有一个验证函数
const jwtToken = jwt.sign(decoded, process.env.JWT_SECRET, { expiresIn: "1h" });
ctx.state.user = decoded; // 将用户信息存储在ctx.state中
ctx.headers.authorization = `Bearer ${jwtToken}`; // 替换为JWT Token
} catch (err) {
ctx.throw(401, 'Invalid third-party token');
}
}
await next();
});
// JWT中间件保护下方所有路由
app.use(
koaJwt({
secret: process.env.JWT_SECRET,
}).unless({
path: [/^\/public/, /^\/login/],
})
);
// 加载自动生成的路由
loadAPIRoutes();
// 受保护路由
router.get("/protected", (ctx) => {
ctx.body = `Protected content for ${ctx.state.user.username}`;
});
app.use(router.routes());
app.use(router.allowedMethods());
app.listen(process.env.PORT || 3000, () => {
console.log(`Server running on http://localhost:${process.env.PORT || 3000}`);
});