2026-02-11 18:11:33 +08:00
|
|
|
import { logger } from "@certd/basic";
|
2025-07-14 01:36:40 +08:00
|
|
|
import { CommonException, Constants, CrudController, PermissionException, SysSettingsService } from "@certd/lib-server";
|
2026-02-11 18:11:33 +08:00
|
|
|
import { ALL, Body, Controller, Get, Inject, Post, Provide, Query } from "@midwayjs/core";
|
|
|
|
|
import * as fs from "fs";
|
|
|
|
|
import { In } from "typeorm";
|
|
|
|
|
import { UserGrantSetting } from "../../../modules/mine/service/models.js";
|
|
|
|
|
import { UserSettingsService } from "../../../modules/mine/service/user-settings-service.js";
|
|
|
|
|
import { HistoryLogEntity } from "../../../modules/pipeline/entity/history-log.js";
|
|
|
|
|
import { HistoryEntity } from "../../../modules/pipeline/entity/history.js";
|
2025-07-14 01:36:40 +08:00
|
|
|
import { PipelineEntity } from "../../../modules/pipeline/entity/pipeline.js";
|
|
|
|
|
import { HistoryLogService } from "../../../modules/pipeline/service/history-log-service.js";
|
2026-02-11 18:11:33 +08:00
|
|
|
import { HistoryService } from "../../../modules/pipeline/service/history-service.js";
|
2025-07-14 01:36:40 +08:00
|
|
|
import { PipelineService } from "../../../modules/pipeline/service/pipeline-service.js";
|
|
|
|
|
import { AuthService } from "../../../modules/sys/authority/service/auth-service.js";
|
2023-01-29 13:44:19 +08:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 证书
|
|
|
|
|
*/
|
|
|
|
|
@Provide()
|
|
|
|
|
@Controller('/api/pi/history')
|
2023-05-23 18:01:20 +08:00
|
|
|
export class HistoryController extends CrudController<HistoryService> {
|
2023-01-29 13:44:19 +08:00
|
|
|
@Inject()
|
|
|
|
|
service: HistoryService;
|
|
|
|
|
@Inject()
|
2023-06-27 22:45:27 +08:00
|
|
|
pipelineService: PipelineService;
|
|
|
|
|
@Inject()
|
2023-01-29 13:44:19 +08:00
|
|
|
logService: HistoryLogService;
|
|
|
|
|
|
2024-08-05 12:49:44 +08:00
|
|
|
@Inject()
|
|
|
|
|
authService: AuthService;
|
|
|
|
|
|
|
|
|
|
@Inject()
|
|
|
|
|
sysSettingsService: SysSettingsService;
|
|
|
|
|
|
2025-12-28 23:36:53 +08:00
|
|
|
@Inject()
|
|
|
|
|
userSettingsService: UserSettingsService;
|
|
|
|
|
|
2024-10-03 22:03:49 +08:00
|
|
|
getService(): HistoryService {
|
2023-01-29 13:44:19 +08:00
|
|
|
return this.service;
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/page', { summary: Constants.per.authOnly })
|
2024-10-04 00:52:19 +08:00
|
|
|
async page(@Body(ALL) body: any) {
|
2026-02-11 18:11:33 +08:00
|
|
|
const { projectId, userId } = await this.getProjectUserIdRead()
|
|
|
|
|
body.query.projectId = projectId
|
|
|
|
|
|
2024-08-05 12:49:44 +08:00
|
|
|
const isAdmin = await this.authService.isAdmin(this.ctx);
|
|
|
|
|
const publicSettings = await this.sysSettingsService.getPublicSettings();
|
2024-10-04 00:52:19 +08:00
|
|
|
const pipelineQuery: any = {};
|
2024-08-05 12:49:44 +08:00
|
|
|
if (!(publicSettings.managerOtherUserPipeline && isAdmin)) {
|
2026-02-11 18:11:33 +08:00
|
|
|
body.query.userId = userId;
|
|
|
|
|
pipelineQuery.userId = userId;
|
|
|
|
|
}
|
|
|
|
|
if (projectId) {
|
|
|
|
|
pipelineQuery.projectId = projectId;
|
2024-08-05 12:49:44 +08:00
|
|
|
}
|
|
|
|
|
|
2024-10-04 00:52:19 +08:00
|
|
|
let pipelineIds: any = null;
|
|
|
|
|
const pipelineTitle = body.query?.pipelineTitle;
|
2024-10-30 15:19:35 +08:00
|
|
|
delete body.query.pipelineTitle;
|
2024-10-04 00:52:19 +08:00
|
|
|
if (pipelineTitle) {
|
2024-10-14 00:19:55 +08:00
|
|
|
const pipelines = await this.pipelineService.list({
|
|
|
|
|
query: pipelineQuery,
|
|
|
|
|
buildQuery: qb => {
|
2024-10-14 03:17:10 +08:00
|
|
|
qb.andWhere('title like :title', { title: `%${pipelineTitle}%` });
|
2024-10-14 00:19:55 +08:00
|
|
|
},
|
2024-10-04 00:52:19 +08:00
|
|
|
});
|
|
|
|
|
pipelineIds = pipelines.map(p => p.id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const buildQuery = qb => {
|
|
|
|
|
if (pipelineIds) {
|
2024-10-14 03:17:10 +08:00
|
|
|
qb.andWhere({
|
2024-10-04 00:52:19 +08:00
|
|
|
pipelineId: In(pipelineIds),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
};
|
2024-08-05 12:49:44 +08:00
|
|
|
|
2024-10-14 00:19:55 +08:00
|
|
|
const res = await this.service.page({
|
|
|
|
|
query: body.query,
|
|
|
|
|
page: body.page,
|
2024-10-14 14:00:24 +08:00
|
|
|
sort: body.sort,
|
2024-10-14 00:19:55 +08:00
|
|
|
buildQuery,
|
|
|
|
|
});
|
2024-12-03 00:43:43 +08:00
|
|
|
|
2024-10-04 00:52:19 +08:00
|
|
|
return this.ok(res);
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/list', { summary: Constants.per.authOnly })
|
2023-01-29 13:44:19 +08:00
|
|
|
async list(@Body(ALL) body) {
|
2026-02-11 18:11:33 +08:00
|
|
|
const { projectId, userId } = await this.getProjectUserIdRead()
|
2026-02-11 18:17:46 +08:00
|
|
|
if (!body){
|
|
|
|
|
body = {}
|
|
|
|
|
}
|
|
|
|
|
if (projectId){
|
|
|
|
|
body.projectId = projectId
|
|
|
|
|
}
|
2026-02-11 18:11:33 +08:00
|
|
|
|
2025-12-28 23:36:53 +08:00
|
|
|
const isAdmin = this.authService.isAdmin(this.ctx);
|
2024-08-05 12:49:44 +08:00
|
|
|
if (!isAdmin) {
|
2026-02-11 18:11:33 +08:00
|
|
|
body.userId = userId;
|
2024-08-05 12:49:44 +08:00
|
|
|
}
|
2026-02-11 18:11:33 +08:00
|
|
|
|
2023-01-29 13:44:19 +08:00
|
|
|
if (body.pipelineId == null) {
|
|
|
|
|
return this.ok([]);
|
|
|
|
|
}
|
|
|
|
|
const buildQuery = qb => {
|
2025-03-11 00:46:51 +08:00
|
|
|
qb.limit(20);
|
2023-01-29 13:44:19 +08:00
|
|
|
};
|
2025-07-14 01:36:40 +08:00
|
|
|
const withDetail = body.withDetail;
|
|
|
|
|
delete body.withDetail;
|
2025-12-28 23:36:53 +08:00
|
|
|
let select: any = null
|
2025-07-14 01:36:40 +08:00
|
|
|
if (!withDetail) {
|
|
|
|
|
select = {
|
|
|
|
|
pipeline: true, // 后面这里改成false
|
|
|
|
|
id: true,
|
|
|
|
|
userId: true,
|
|
|
|
|
pipelineId: true,
|
|
|
|
|
status: true,
|
|
|
|
|
// startTime: true,
|
|
|
|
|
triggerType: true,
|
|
|
|
|
endTime: true,
|
|
|
|
|
createTime: true,
|
2026-02-11 18:11:33 +08:00
|
|
|
updateTime: true,
|
|
|
|
|
projectId: true,
|
2025-07-14 01:36:40 +08:00
|
|
|
};
|
|
|
|
|
}
|
2024-10-14 00:19:55 +08:00
|
|
|
const listRet = await this.getService().list({
|
|
|
|
|
query: body,
|
2024-10-14 14:00:24 +08:00
|
|
|
sort: { prop: 'id', asc: false },
|
2024-10-14 00:19:55 +08:00
|
|
|
buildQuery,
|
2025-07-14 01:36:40 +08:00
|
|
|
select
|
2024-10-14 00:19:55 +08:00
|
|
|
});
|
2025-07-14 01:36:40 +08:00
|
|
|
|
2024-12-03 10:32:47 +08:00
|
|
|
for (const item of listRet) {
|
|
|
|
|
if (!item.pipeline) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
const json = JSON.parse(item.pipeline);
|
|
|
|
|
delete json.stages;
|
|
|
|
|
item.pipeline = json;
|
2025-07-14 01:36:40 +08:00
|
|
|
|
|
|
|
|
//@ts-ignore
|
|
|
|
|
item.version = json.version;
|
|
|
|
|
item.status = json.status.result
|
|
|
|
|
delete item.pipeline;
|
2024-12-03 10:32:47 +08:00
|
|
|
}
|
2025-07-14 01:36:40 +08:00
|
|
|
|
2023-01-29 13:44:19 +08:00
|
|
|
return this.ok(listRet);
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/add', { summary: Constants.per.authOnly })
|
2023-01-29 13:44:19 +08:00
|
|
|
async add(@Body(ALL) bean: PipelineEntity) {
|
2026-02-11 18:11:33 +08:00
|
|
|
const { projectId, userId } = await this.getProjectUserIdRead()
|
|
|
|
|
bean.projectId = projectId
|
|
|
|
|
bean.userId = userId;
|
2023-01-29 13:44:19 +08:00
|
|
|
return super.add(bean);
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/update', { summary: Constants.per.authOnly })
|
2023-01-29 13:44:19 +08:00
|
|
|
async update(@Body(ALL) bean) {
|
2026-02-13 00:41:40 +08:00
|
|
|
await this.checkOwner(this.getService(), bean.id,"write",true);
|
2024-12-01 02:10:40 +08:00
|
|
|
delete bean.userId;
|
2023-01-29 13:44:19 +08:00
|
|
|
return super.update(bean);
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/save', { summary: Constants.per.authOnly })
|
2023-01-29 13:44:19 +08:00
|
|
|
async save(@Body(ALL) bean: HistoryEntity) {
|
2026-02-11 18:11:33 +08:00
|
|
|
const { projectId,userId } = await this.getProjectUserIdWrite()
|
|
|
|
|
bean.userId = userId;
|
|
|
|
|
bean.projectId = projectId;
|
2023-01-29 13:44:19 +08:00
|
|
|
if (bean.id > 0) {
|
2026-02-11 18:11:33 +08:00
|
|
|
//修改
|
|
|
|
|
delete bean.projectId;
|
|
|
|
|
delete bean.userId;
|
2026-02-13 00:41:40 +08:00
|
|
|
await this.checkOwner(this.getService(), bean.id,"write",true);
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|
2026-02-11 18:11:33 +08:00
|
|
|
|
2023-01-29 13:44:19 +08:00
|
|
|
await this.service.save(bean);
|
|
|
|
|
return this.ok(bean.id);
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/saveLog', { summary: Constants.per.authOnly })
|
2023-01-29 13:44:19 +08:00
|
|
|
async saveLog(@Body(ALL) bean: HistoryLogEntity) {
|
2026-02-11 18:11:33 +08:00
|
|
|
const { projectId,userId } = await this.getProjectUserIdWrite()
|
|
|
|
|
bean.projectId = projectId;
|
|
|
|
|
bean.userId = userId;
|
2023-01-29 13:44:19 +08:00
|
|
|
if (bean.id > 0) {
|
2026-02-11 18:11:33 +08:00
|
|
|
//修改
|
|
|
|
|
delete bean.projectId;
|
|
|
|
|
delete bean.userId;
|
2026-02-13 00:41:40 +08:00
|
|
|
await this.checkOwner(this.logService, bean.id,"write",true);
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|
|
|
|
|
await this.logService.save(bean);
|
|
|
|
|
return this.ok(bean.id);
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/delete', { summary: Constants.per.authOnly })
|
2024-08-30 18:50:53 +08:00
|
|
|
async delete(@Query('id') id: number) {
|
2026-02-13 00:41:40 +08:00
|
|
|
await this.checkOwner(this.getService(), id,"write",true);
|
2024-08-05 12:49:44 +08:00
|
|
|
await super.delete(id);
|
|
|
|
|
return this.ok();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Post('/deleteByIds', { summary: Constants.per.authOnly })
|
2024-08-30 18:50:53 +08:00
|
|
|
async deleteByIds(@Body(ALL) body: any) {
|
2026-02-13 00:41:40 +08:00
|
|
|
let {userId} = await this.checkOwner(this.getService(), body.ids,"write",true);
|
2024-08-05 12:49:44 +08:00
|
|
|
const isAdmin = await this.authService.isAdmin(this.ctx);
|
2026-02-11 18:11:33 +08:00
|
|
|
userId = isAdmin ? null : userId;
|
2024-08-05 12:49:44 +08:00
|
|
|
await this.getService().deleteByIds(body.ids, userId);
|
|
|
|
|
return this.ok();
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/detail', { summary: Constants.per.authOnly })
|
2024-08-30 18:50:53 +08:00
|
|
|
async detail(@Query('id') id: number) {
|
2026-02-13 00:41:40 +08:00
|
|
|
await this.checkOwner(this.getService(), id,"read",true);
|
2023-01-29 13:44:19 +08:00
|
|
|
const detail = await this.service.detail(id);
|
|
|
|
|
return this.ok(detail);
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-27 09:29:43 +08:00
|
|
|
@Post('/logs', { summary: Constants.per.authOnly })
|
2024-08-30 18:50:53 +08:00
|
|
|
async logs(@Query('id') id: number) {
|
2026-02-13 00:41:40 +08:00
|
|
|
await this.checkOwner(this.logService, id,"read",true);
|
2023-01-29 13:44:19 +08:00
|
|
|
const logInfo = await this.logService.info(id);
|
|
|
|
|
return this.ok(logInfo);
|
|
|
|
|
}
|
2023-06-27 22:45:27 +08:00
|
|
|
|
|
|
|
|
@Post('/files', { summary: Constants.per.authOnly })
|
2024-08-30 18:50:53 +08:00
|
|
|
async files(@Query('pipelineId') pipelineId: number, @Query('historyId') historyId: number) {
|
2023-06-27 22:45:27 +08:00
|
|
|
const files = await this.getFiles(historyId, pipelineId);
|
|
|
|
|
return this.ok(files);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async getFiles(historyId, pipelineId) {
|
|
|
|
|
let history = null;
|
|
|
|
|
if (historyId != null) {
|
|
|
|
|
// nothing
|
|
|
|
|
history = await this.service.info(historyId);
|
|
|
|
|
} else if (pipelineId != null) {
|
|
|
|
|
history = await this.service.getLastHistory(pipelineId);
|
|
|
|
|
}
|
|
|
|
|
if (history == null) {
|
|
|
|
|
throw new CommonException('historyId is null');
|
|
|
|
|
}
|
2026-02-11 18:11:33 +08:00
|
|
|
const {projectId} = await this.getProjectUserIdRead()
|
|
|
|
|
if (projectId) {
|
|
|
|
|
//enterprise模式
|
|
|
|
|
if(history.projectId !== projectId){
|
|
|
|
|
throw new PermissionException("您没有权限下载该流水线证书,请先加入该项目:"+history.projectId);
|
|
|
|
|
}
|
|
|
|
|
//有权限下载
|
|
|
|
|
}else if (history.userId !== this.getUserId()) {
|
2025-12-28 23:36:53 +08:00
|
|
|
// 如果是管理员,检查用户是否有授权管理员查看
|
|
|
|
|
const isAdmin = await this.isAdmin()
|
|
|
|
|
if (!isAdmin) {
|
|
|
|
|
throw new PermissionException();
|
|
|
|
|
}
|
|
|
|
|
// 是否允许管理员查看
|
|
|
|
|
const setting = await this.userSettingsService.getSetting<UserGrantSetting>(history.userId, UserGrantSetting, false);
|
|
|
|
|
if (setting?.allowAdminViewCerts!==true) {
|
|
|
|
|
//不允许管理员查看
|
|
|
|
|
throw new PermissionException("该流水线的用户还未授权管理员下载证书,请先让用户在”设置->授权委托“中打开开关");
|
|
|
|
|
}
|
|
|
|
|
//允许管理员查看
|
2023-06-27 22:45:27 +08:00
|
|
|
}
|
|
|
|
|
return await this.service.getFiles(history);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Get('/download', { summary: Constants.per.authOnly })
|
2024-08-30 18:50:53 +08:00
|
|
|
async download(@Query('pipelineId') pipelineId: number, @Query('historyId') historyId: number, @Query('fileId') fileId: string) {
|
2023-06-27 22:45:27 +08:00
|
|
|
const files = await this.getFiles(historyId, pipelineId);
|
|
|
|
|
const file = files.find(f => f.id === fileId);
|
|
|
|
|
if (file == null) {
|
|
|
|
|
throw new CommonException('file not found');
|
|
|
|
|
}
|
|
|
|
|
// koa send file
|
|
|
|
|
// 下载文件的名称
|
|
|
|
|
// const filename = file.filename;
|
|
|
|
|
// 要下载的文件的完整路径
|
|
|
|
|
const path = file.path;
|
2024-05-30 10:12:48 +08:00
|
|
|
logger.info(`download:${path}`);
|
2023-06-27 22:45:27 +08:00
|
|
|
// 以流的形式下载文件
|
|
|
|
|
this.ctx.attachment(path);
|
|
|
|
|
this.ctx.set('Content-Type', 'application/octet-stream');
|
|
|
|
|
|
|
|
|
|
return fs.createReadStream(path);
|
|
|
|
|
}
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|