feat(api): 新增客户端添加功能并优化路由加载
- 新增客户端添加 API,实现客户端信息的添加和处理 - 重构路由加载函数,支持递归加载子目录中的路由文件 - 添加日志记录,便于调试和监控 - 优化 HTTP 请求处理,增加错误处理和状态码检查
This commit is contained in:
@@ -1,8 +0,0 @@
|
|||||||
|
|
||||||
|
|
||||||
module.exports = (ctx) => {
|
|
||||||
ctx.body = {
|
|
||||||
message: 'Example POST API',
|
|
||||||
data: ctx.request.body
|
|
||||||
};
|
|
||||||
};
|
|
||||||
36
api/client/add.js
Normal file
36
api/client/add.js
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
const { randomUUID } = require("crypto");
|
||||||
|
const http = require("../../utils/http");
|
||||||
|
const dayjs = require("dayjs");
|
||||||
|
|
||||||
|
module.exports = async (ctx) => {
|
||||||
|
let { id = 4, limitIp, expiryTime, enable = !0, totalGB = 1, subId = "2rv0gb458kbfl532" } = ctx.request.body;
|
||||||
|
const inbound = await http.get(`/api/inbounds/get/${id}`);
|
||||||
|
if (!inbound?.success) return (ctx.body = { code: "1", msg: "获取节点失败" });
|
||||||
|
const uuid = randomUUID(),
|
||||||
|
email = uuid.split("-")[0];
|
||||||
|
expiryTime = dayjs(expiryTime, "YYYY-MM-DD HH:mm:ss").valueOf();
|
||||||
|
const settings = { clients: [{ id: uuid, flow: "", email, limitIp, enable, tgId: "", subId, reset: 0, totalGB, expiryTime }] };
|
||||||
|
http
|
||||||
|
.post("/api/inbounds/addClient", { id, settings: JSON.stringify(settings) })
|
||||||
|
.then((res) => {
|
||||||
|
if (res?.success) {
|
||||||
|
const { remark, port, protocol, streamSettings = "{}" } = inbound.obj || {};
|
||||||
|
const {
|
||||||
|
network = "ws",
|
||||||
|
security = "none",
|
||||||
|
wsSettings: { host, path },
|
||||||
|
} = JSON.parse(streamSettings);
|
||||||
|
return ctx.body = {
|
||||||
|
code: "0",
|
||||||
|
data: `${protocol}://${uuid}@vless.jjcp52.com:${port}?type=${network}&path=${path}&host=${host}&security=${security}#${remark}-${email}`,
|
||||||
|
message: "success",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
ctx.body = {
|
||||||
|
message: "Example POST API",
|
||||||
|
data: ctx.request.body,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
};
|
||||||
47
app.js
47
app.js
@@ -12,22 +12,48 @@ app.use(bodyParser()); // 添加在路由中间件之前
|
|||||||
const router = new Router();
|
const router = new Router();
|
||||||
|
|
||||||
// 自动加载API路由函数
|
// 自动加载API路由函数
|
||||||
const loadAPIRoutes = () => {
|
const loadAPIRoutes = (baseDir = 'api', baseRoute = '/api') => {
|
||||||
const apiDir = path.join(__dirname, "api");
|
const scanDirectory = (dir, routePrefix) => {
|
||||||
const files = fs.readdirSync(apiDir);
|
const entries = fs.readdirSync(dir, { withFileTypes: true });
|
||||||
|
|
||||||
files.forEach((file) => {
|
entries.forEach(entry => {
|
||||||
if (file.endsWith(".js") && file !== "index.js") {
|
const fullPath = path.join(dir, entry.name);
|
||||||
const routePath = `/api/${file.replace(".js", "")}`;
|
const relativePath = path.relative(baseDir, dir);
|
||||||
const handler = require(path.join(apiDir, file));
|
|
||||||
|
|
||||||
router.post(routePath, async (ctx) => {
|
// 构建路由路径:工程目录结构 -> URL路径
|
||||||
await handler(ctx);
|
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) => {
|
router.get("/public", (ctx) => {
|
||||||
ctx.body = "Public content";
|
ctx.body = "Public content";
|
||||||
@@ -50,7 +76,6 @@ app.use(async (ctx, next) => {
|
|||||||
}
|
}
|
||||||
await next();
|
await next();
|
||||||
});
|
});
|
||||||
|
|
||||||
// JWT中间件(保护下方所有路由)
|
// JWT中间件(保护下方所有路由)
|
||||||
app.use(
|
app.use(
|
||||||
koaJwt({
|
koaJwt({
|
||||||
|
|||||||
@@ -11,6 +11,7 @@
|
|||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"axios": "^1.7.9",
|
"axios": "^1.7.9",
|
||||||
|
"dayjs": "^1.11.13",
|
||||||
"dotenv": "^16.4.7",
|
"dotenv": "^16.4.7",
|
||||||
"jsonwebtoken": "^9.0.2",
|
"jsonwebtoken": "^9.0.2",
|
||||||
"koa": "^2.15.4",
|
"koa": "^2.15.4",
|
||||||
|
|||||||
@@ -1,3 +1,15 @@
|
|||||||
require("dotenv").config();
|
require("dotenv").config();
|
||||||
const { default: axios } = require("axios");
|
const { default: axios } = require("axios");
|
||||||
module.exports = (baseURL = process.env.BASE_URL) => axios.create({ baseURL: BASE_URL });
|
const instance = axios.create({ baseURL: process.env.BASE_URL });
|
||||||
|
instance.interceptors.response.use(
|
||||||
|
(response) => {
|
||||||
|
if (response.data.code === 200) {
|
||||||
|
return response.data.data;
|
||||||
|
} else return response;
|
||||||
|
},
|
||||||
|
(error) => {
|
||||||
|
console.log(error);
|
||||||
|
return Promise.reject(error);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
module.exports = instance;
|
||||||
|
|||||||
Reference in New Issue
Block a user