perf: cancel task

pull/21/merge
xiaojunnuo 2023-07-03 11:45:32 +08:00
parent 3eeb1f77aa
commit bc65c0a786
7 changed files with 64 additions and 5 deletions

View File

@ -52,8 +52,10 @@ export class Executor {
this.lastStatusMap = new RunnableCollection(lastRuntime?.pipeline); this.lastStatusMap = new RunnableCollection(lastRuntime?.pipeline);
} }
cancel() { async cancel() {
this.canceled = true; this.canceled = true;
this.runtime?.cancel(this.pipeline);
await this.onChanged(this.runtime);
} }
async run(runtimeId: any = 0, triggerType: string) { async run(runtimeId: any = 0, triggerType: string) {

View File

@ -92,6 +92,19 @@ export class RunHistory {
this.logError(runnable, e); this.logError(runnable, e);
} }
cancel(runnable: Runnable) {
const now = new Date().getTime();
const status = runnable.status;
_.merge(status, {
status: ResultType.canceled,
endTime: now,
result: ResultType.canceled,
message: "用户取消",
});
this.log(runnable, "任务取消");
}
log(runnable: Runnable, text: string) { log(runnable: Runnable, text: string) {
// @ts-ignore // @ts-ignore
this._loggers[runnable.id].info(`[${runnable.title}]<id:${runnable.id}> [${runnable.runnableType}]`, text); this._loggers[runnable.id].info(`[${runnable.title}]<id:${runnable.id}> [${runnable.runnableType}]`, text);

View File

@ -106,6 +106,7 @@ export enum ResultType {
start = "start", start = "start",
success = "success", success = "success",
error = "error", error = "error",
canceled = "canceled",
skip = "skip", skip = "skip",
none = "none", none = "none",
} }

View File

@ -67,6 +67,14 @@ export function Trigger(id: any) {
}); });
} }
export function Cancel(historyId: any) {
return request({
url: apiPrefix + "/cancel",
method: "post",
params: { historyId }
});
}
export async function GetFiles(pipelineId: number) { export async function GetFiles(pipelineId: number) {
return request({ return request({
url: historyApiPrefix + "/files", url: historyApiPrefix + "/files",

View File

@ -9,6 +9,8 @@
<a-tag v-if="isCurrent" class="pointer" color="green" :closable="true" @close="cancel"></a-tag> <a-tag v-if="isCurrent" class="pointer" color="green" :closable="true" @close="cancel"></a-tag>
<a-tag v-else-if="!editMode" class="pointer" color="blue" @click="view"></a-tag> <a-tag v-else-if="!editMode" class="pointer" color="blue" @click="view"></a-tag>
<a-tag v-if="status.value === 'start'" class="pointer" color="red" @click="cancelTask"></a-tag>
</p> </p>
</a-timeline-item> </a-timeline-item>
</template> </template>
@ -16,6 +18,8 @@
<script lang="ts"> <script lang="ts">
import { defineComponent, ref, provide, Ref, watch, computed } from "vue"; import { defineComponent, ref, provide, Ref, watch, computed } from "vue";
import { statusUtil } from "/@/views/certd/pipeline/pipeline/utils/util.status"; import { statusUtil } from "/@/views/certd/pipeline/pipeline/utils/util.status";
import * as api from "../../api";
import { Modal, notification } from "ant-design-vue";
export default defineComponent({ export default defineComponent({
name: "PiHistoryTimelineItem", name: "PiHistoryTimelineItem",
props: { props: {
@ -38,7 +42,7 @@ export default defineComponent({
} }
}, },
emits: ["view", "cancel"], emits: ["view", "cancel"],
setup(props, ctx) { setup(props: any, ctx: any) {
const status = computed(() => { const status = computed(() => {
return statusUtil.get(props.runnable?.status?.result); return statusUtil.get(props.runnable?.status?.result);
}); });
@ -49,10 +53,25 @@ export default defineComponent({
function cancel() { function cancel() {
ctx.emit("cancel"); ctx.emit("cancel");
} }
function cancelTask() {
Modal.confirm({
title: "确认取消",
content: "确认取消该任务吗?",
okText: "确认",
cancelText: "取消",
onOk: async () => {
await api.Cancel(props.runnable.id);
}
});
notification.success({
message: "任务取消成功"
});
}
return { return {
status, status,
cancel, cancel,
view view,
cancelTask
}; };
} }
}); });

View File

@ -11,6 +11,7 @@ import { CrudController } from '../../../basic/crud-controller';
import { PipelineService } from '../service/pipeline-service'; import { PipelineService } from '../service/pipeline-service';
import { PipelineEntity } from '../entity/pipeline'; import { PipelineEntity } from '../entity/pipeline';
import { Constants } from '../../../basic/constants'; import { Constants } from '../../../basic/constants';
import { HistoryService } from '../service/history-service';
/** /**
* *
@ -20,6 +21,8 @@ import { Constants } from '../../../basic/constants';
export class PipelineController extends CrudController<PipelineService> { export class PipelineController extends CrudController<PipelineService> {
@Inject() @Inject()
service: PipelineService; service: PipelineService;
@Inject()
historyService: HistoryService;
getService() { getService() {
return this.service; return this.service;
@ -77,4 +80,11 @@ export class PipelineController extends CrudController<PipelineService> {
await this.service.trigger(id); await this.service.trigger(id);
return this.ok({}); return this.ok({});
} }
@Post('/cancel', { summary: Constants.per.authOnly })
async cancel(@Query('historyId') historyId) {
await this.historyService.checkUserId(historyId, this.ctx.user.id);
await this.service.cancel(historyId);
return this.ok({});
}
} }

View File

@ -4,7 +4,7 @@ import { In, Repository } from 'typeorm';
import { BaseService } from '../../../basic/base-service'; import { BaseService } from '../../../basic/base-service';
import { PipelineEntity } from '../entity/pipeline'; import { PipelineEntity } from '../entity/pipeline';
import { PipelineDetail } from '../entity/vo/pipeline-detail'; import { PipelineDetail } from '../entity/vo/pipeline-detail';
import { Executor, Pipeline, RunHistory } from '@certd/pipeline'; import { Executor, Pipeline, ResultType, RunHistory } from '@certd/pipeline';
import { AccessService } from './access-service'; import { AccessService } from './access-service';
import { DbStorage } from './db-storage'; import { DbStorage } from './db-storage';
import { StorageService } from './storage-service'; import { StorageService } from './storage-service';
@ -235,7 +235,13 @@ export class PipelineService extends BaseService<PipelineEntity> {
async cancel(historyId: number) { async cancel(historyId: number) {
const executor = runningTasks.get(historyId); const executor = runningTasks.get(historyId);
if (executor) { if (executor) {
executor.cancel(); await executor.cancel();
} else {
const entity = await this.historyService.info(historyId);
const pipeline: Pipeline = JSON.parse(entity.pipeline);
pipeline.status.status = ResultType.canceled;
const runtime = new RunHistory(historyId, null, pipeline);
await this.saveHistory(runtime);
} }
} }