## 后端改动 - 添加 Project.envPresets 字段(JSON 格式) - 移除 Deployment.env 字段,统一使用 envVars - 更新部署 DTO,支持 envVars (Record<string, string>) - pipeline-runner 支持解析并注入 envVars 到环境 - 移除稀疏检出模板和相关环境变量 - 优化代码格式(Biome lint & format) ## 前端改动 - 新增 EnvPresetsEditor 组件(支持单选/多选/输入框类型) - 项目创建/编辑界面集成环境预设编辑器 - 部署界面基于预设动态生成环境变量表单 - 移除稀疏检出表单项 - 项目详情页添加环境变量预设配置 tab - 优化部署界面布局(基本参数 & 环境变量分区) ## 文档 - 添加完整文档目录结构(docs/) - 创建设计文档 design-0005(部署流程重构) - 添加 API 文档、架构设计文档等 ## 数据库 - 执行 prisma db push 同步 schema 变更
232 lines
5.6 KiB
TypeScript
232 lines
5.6 KiB
TypeScript
import type { Context } from 'koa';
|
|
import { Controller, Delete, Get, Post, Put } from '../../decorators/route.ts';
|
|
import { GitManager } from '../../libs/git-manager.ts';
|
|
import { log } from '../../libs/logger.ts';
|
|
import { prisma } from '../../libs/prisma.ts';
|
|
import { BusinessError } from '../../middlewares/exception.ts';
|
|
import {
|
|
createProjectSchema,
|
|
listProjectQuerySchema,
|
|
projectIdSchema,
|
|
updateProjectSchema,
|
|
} from './dto.ts';
|
|
|
|
@Controller('/projects')
|
|
export class ProjectController {
|
|
// GET /api/projects - 获取项目列表
|
|
@Get('')
|
|
async list(ctx: Context) {
|
|
const query = listProjectQuerySchema.parse(ctx.query);
|
|
|
|
const whereCondition: any = {
|
|
valid: 1,
|
|
};
|
|
|
|
// 如果提供了名称搜索参数
|
|
if (query?.name) {
|
|
whereCondition.name = {
|
|
contains: query.name,
|
|
};
|
|
}
|
|
|
|
const [total, projects] = await Promise.all([
|
|
prisma.project.count({ where: whereCondition }),
|
|
prisma.project.findMany({
|
|
where: whereCondition,
|
|
skip: query ? (query.page - 1) * query.limit : 0,
|
|
take: query?.limit,
|
|
orderBy: {
|
|
createdAt: 'desc',
|
|
},
|
|
}),
|
|
]);
|
|
|
|
return {
|
|
data: projects,
|
|
pagination: {
|
|
page: query?.page || 1,
|
|
limit: query?.limit || 10,
|
|
total,
|
|
totalPages: Math.ceil(total / (query?.limit || 10)),
|
|
},
|
|
};
|
|
}
|
|
|
|
// GET /api/projects/:id - 获取单个项目
|
|
@Get(':id')
|
|
async show(ctx: Context) {
|
|
const { id } = projectIdSchema.parse(ctx.params);
|
|
|
|
const project = await prisma.project.findFirst({
|
|
where: {
|
|
id,
|
|
valid: 1,
|
|
},
|
|
});
|
|
|
|
if (!project) {
|
|
throw new BusinessError('项目不存在', 1002, 404);
|
|
}
|
|
|
|
// 获取工作目录状态信息
|
|
let workspaceStatus = null;
|
|
if (project.projectDir) {
|
|
try {
|
|
const status = await GitManager.checkWorkspaceStatus(
|
|
project.projectDir,
|
|
);
|
|
let size = 0;
|
|
let gitInfo = null;
|
|
|
|
if (status.exists && !status.isEmpty) {
|
|
size = await GitManager.getDirectorySize(project.projectDir);
|
|
}
|
|
|
|
if (status.hasGit) {
|
|
gitInfo = await GitManager.getGitInfo(project.projectDir);
|
|
}
|
|
|
|
workspaceStatus = {
|
|
...status,
|
|
size,
|
|
gitInfo,
|
|
};
|
|
} catch (error) {
|
|
log.error(
|
|
'project',
|
|
'Failed to get workspace status for project %s: %s',
|
|
project.name,
|
|
(error as Error).message,
|
|
);
|
|
// 即使获取状态失败,也返回项目信息
|
|
workspaceStatus = {
|
|
status: 'error',
|
|
error: (error as Error).message,
|
|
};
|
|
}
|
|
}
|
|
|
|
return {
|
|
...project,
|
|
workspaceStatus,
|
|
};
|
|
}
|
|
|
|
// POST /api/projects - 创建项目
|
|
@Post('')
|
|
async create(ctx: Context) {
|
|
const validatedData = createProjectSchema.parse(ctx.request.body);
|
|
|
|
// 检查工作目录是否已被其他项目使用
|
|
const existingProject = await prisma.project.findFirst({
|
|
where: {
|
|
projectDir: validatedData.projectDir,
|
|
valid: 1,
|
|
},
|
|
});
|
|
|
|
if (existingProject) {
|
|
throw new BusinessError('该工作目录已被其他项目使用', 1003, 400);
|
|
}
|
|
|
|
const project = await prisma.project.create({
|
|
data: {
|
|
name: validatedData.name,
|
|
description: validatedData.description || '',
|
|
repository: validatedData.repository,
|
|
projectDir: validatedData.projectDir,
|
|
envPresets: validatedData.envPresets,
|
|
createdBy: 'system',
|
|
updatedBy: 'system',
|
|
valid: 1,
|
|
},
|
|
});
|
|
|
|
log.info(
|
|
'project',
|
|
'Created new project: %s with projectDir: %s',
|
|
project.name,
|
|
project.projectDir,
|
|
);
|
|
return project;
|
|
}
|
|
|
|
// PUT /api/projects/:id - 更新项目
|
|
@Put(':id')
|
|
async update(ctx: Context) {
|
|
const { id } = projectIdSchema.parse(ctx.params);
|
|
const validatedData = updateProjectSchema.parse(ctx.request.body);
|
|
|
|
// 检查项目是否存在
|
|
const existingProject = await prisma.project.findFirst({
|
|
where: {
|
|
id,
|
|
valid: 1,
|
|
},
|
|
});
|
|
|
|
if (!existingProject) {
|
|
throw new BusinessError('项目不存在', 1002, 404);
|
|
}
|
|
|
|
// 只更新提供的字段
|
|
const updateData: any = {
|
|
updatedBy: 'system',
|
|
};
|
|
|
|
if (validatedData.name !== undefined) {
|
|
updateData.name = validatedData.name;
|
|
}
|
|
if (validatedData.description !== undefined) {
|
|
updateData.description = validatedData.description;
|
|
}
|
|
if (validatedData.repository !== undefined) {
|
|
updateData.repository = validatedData.repository;
|
|
}
|
|
if (validatedData.envPresets !== undefined) {
|
|
updateData.envPresets = validatedData.envPresets;
|
|
}
|
|
|
|
const project = await prisma.project.update({
|
|
where: { id },
|
|
data: updateData,
|
|
});
|
|
|
|
log.info('project', 'Updated project: %s', project.name);
|
|
return project;
|
|
}
|
|
|
|
// DELETE /api/projects/:id - 删除项目(软删除)
|
|
@Delete(':id')
|
|
async destroy(ctx: Context) {
|
|
const { id } = projectIdSchema.parse(ctx.params);
|
|
|
|
// 检查项目是否存在
|
|
const existingProject = await prisma.project.findFirst({
|
|
where: {
|
|
id,
|
|
valid: 1,
|
|
},
|
|
});
|
|
|
|
if (!existingProject) {
|
|
throw new BusinessError('项目不存在', 1002, 404);
|
|
}
|
|
|
|
// 软删除:将 valid 设置为 0
|
|
await prisma.project.update({
|
|
where: { id },
|
|
data: {
|
|
valid: 0,
|
|
updatedBy: 'system',
|
|
},
|
|
});
|
|
|
|
log.info('project', 'Deleted project: %s', existingProject.name);
|
|
|
|
// RESTful 删除成功返回 204 No Content
|
|
ctx.status = 204;
|
|
return null;
|
|
}
|
|
}
|