Files
archived-gitea-ai-assistant/src/controllers/admin.ts
jeffusion 318e6d3688 build: replace tslint with Biome for code quality
- Add @biomejs/biome as dev dependency
- Remove deprecated tslint dependency
- Add biome.json with project-specific rules
- Update lint script to use Biome
- Apply Biome auto-fixes across codebase
2026-03-03 17:03:23 +08:00

124 lines
3.9 KiB
TypeScript

import { Hono } from 'hono';
import { sign } from 'hono/jwt';
import config from '../config';
import { reviewEngine } from '../review/engine';
import { giteaService } from '../services/gitea';
import { logger } from '../utils/logger';
const publicRoutes = new Hono();
const protectedRoutes = new Hono();
// --- Public Routes ---
// 登录接口
publicRoutes.post('/login', async (c) => {
const { password } = await c.req.json();
if (password === config.admin.password) {
const payload = {
sub: 'admin', // Subject
exp: Math.floor(Date.now() / 1000) + 60 * 60 * 24, // Expiration: 24 hours
};
const token = await sign(payload, config.admin.jwtSecret);
return c.json({ token });
}
return c.json({ message: 'Invalid credentials' }, 401);
});
// --- Protected Routes ---
// 获取仓库列表及 Webhook 状态
protectedRoutes.get('/repositories', async (c) => {
try {
const page = Number.parseInt(c.req.query('page') || '1', 10);
const query = c.req.query('q');
const limit = 30; // 每页数量固定,或也可从查询参数获取
const { repos, totalCount } = await giteaService.listAllRepositories(page, limit, query);
const webhookUrl = c.req.url.replace(/\/admin\/api\/repositories.*$/, '/webhook/gitea');
const reposWithStatus = await Promise.all(
repos.map(async (repo) => {
const [owner, repoName] = repo.full_name.split('/');
const hooks = await giteaService.listWebhooks(owner, repoName);
const webhook = hooks.find((h) => h.config.url === webhookUrl);
return {
name: repo.full_name,
webhook_status: webhook ? 'active' : 'inactive',
hook_id: webhook ? webhook.id : null,
};
})
);
return c.json({
data: reposWithStatus,
totalCount,
page,
limit,
});
} catch (error: any) {
logger.error('获取仓库列表失败:', error);
return c.json({ message: 'Failed to fetch repositories', error: error.message }, 500);
}
});
// 创建 Webhook
protectedRoutes.post('/repositories/:owner/:repo/webhook', async (c) => {
const { owner, repo } = c.req.param();
const webhookUrl = `${new URL(c.req.url).origin}/webhook/gitea`;
try {
await giteaService.createWebhook(owner, repo, webhookUrl);
return c.json({ success: true });
} catch (error: any) {
logger.error(`${owner}/${repo} 创建 Webhook 失败:`, error);
return c.json({ message: 'Failed to create webhook', error: error.message }, 500);
}
});
// 删除 Webhook
protectedRoutes.delete('/repositories/:owner/:repo/webhook/:hookId', async (c) => {
const { owner, repo, hookId } = c.req.param();
try {
await giteaService.deleteWebhook(owner, repo, Number.parseInt(hookId, 10));
return c.json({ success: true });
} catch (error: any) {
logger.error(`删除 ${owner}/${repo} 的 Webhook 失败:`, error);
return c.json({ message: 'Failed to delete webhook', error: error.message }, 500);
}
});
// 查询审查任务
protectedRoutes.get('/review/runs', async (c) => {
try {
const limit = Number.parseInt(c.req.query('limit') || '50', 10);
const runs = await reviewEngine.listRuns(limit);
return c.json({ data: runs });
} catch (error: any) {
logger.error('获取审查任务列表失败:', error);
return c.json({ message: 'Failed to fetch review runs', error: error.message }, 500);
}
});
// 查询审查任务详情
protectedRoutes.get('/review/runs/:runId', async (c) => {
try {
const { runId } = c.req.param();
const result = await reviewEngine.getRunDetails(runId);
if (!result) {
return c.json({ message: 'Run not found' }, 404);
}
return c.json(result);
} catch (error: any) {
logger.error('获取审查任务详情失败:', error);
return c.json({ message: 'Failed to fetch review run details', error: error.message }, 500);
}
});
export const adminController = {
publicRoutes,
protectedRoutes,
};