feat(任务中心): 接口用例/场景/测试计划/组执行历史的执行结果

This commit is contained in:
baiqi 2024-11-15 14:54:39 +08:00 committed by Craftsman
parent 2c9cbea98c
commit 400194b212
13 changed files with 199 additions and 136 deletions

View File

@ -82,6 +82,7 @@ import {
SortApiCaseUrl, SortApiCaseUrl,
SortApiScenarioUrl, SortApiScenarioUrl,
SortFeatureCaseUrl, SortFeatureCaseUrl,
TaskResultUrl,
TestPlanAndGroupCopyUrl, TestPlanAndGroupCopyUrl,
TestPlanApiAssociatedPageUrl, TestPlanApiAssociatedPageUrl,
TestPlanAssociateBugUrl, TestPlanAssociateBugUrl,
@ -127,6 +128,7 @@ import type {
PlanDetailExecuteHistoryItem, PlanDetailExecuteHistoryItem,
PlanDetailFeatureCaseItem, PlanDetailFeatureCaseItem,
PlanDetailFeatureCaseListQueryParams, PlanDetailFeatureCaseListQueryParams,
PlanExecuteResult,
PlanMinderEditParams, PlanMinderEditParams,
PlanMinderNode, PlanMinderNode,
RunFeatureCaseParams, RunFeatureCaseParams,
@ -522,3 +524,8 @@ export function batchAssociatedBugToMinderCase(data: TableQueryParams) {
export function batchAddBugToMinderCase(data: { request: BugEditFormObject; fileList: File[] }) { export function batchAddBugToMinderCase(data: { request: BugEditFormObject; fileList: File[] }) {
return MSR.uploadFile({ url: BatchAddBugToMinderCaseUrl }, data, '', true); return MSR.uploadFile({ url: BatchAddBugToMinderCaseUrl }, data, '', true);
} }
// 测试计划/组-执行结果
export function getTaskResult(id: string) {
return MSR.get<PlanExecuteResult>({ url: `${TaskResultUrl}/${id}` });
}

View File

@ -188,3 +188,5 @@ export const BatchAddBugToCaseUrl = '/test-plan/functional/case/batch/add-bug';
export const BatchAssociatedBugToMinderCaseUrl = '/test-plan/functional/case/minder/batch/associate-bug'; export const BatchAssociatedBugToMinderCaseUrl = '/test-plan/functional/case/minder/batch/associate-bug';
// 测试计划-详情-用例列表-脑图批量新建缺陷 // 测试计划-详情-用例列表-脑图批量新建缺陷
export const BatchAddBugToMinderCaseUrl = '/test-plan/functional/case/minder/batch/add-bug'; export const BatchAddBugToMinderCaseUrl = '/test-plan/functional/case/minder/batch/add-bug';
// 测试计划/组-执行结果
export const TaskResultUrl = '/test-plan/report/get-task';

View File

@ -6,6 +6,7 @@ import type { customFieldsItem } from '@/models/caseManagement/featureCase';
import type { TableQueryParams } from '@/models/common'; import type { TableQueryParams } from '@/models/common';
import { BatchApiParams, DragSortParams } from '@/models/common'; import { BatchApiParams, DragSortParams } from '@/models/common';
import { CaseLinkEnum, LastExecuteResults } from '@/enums/caseEnum'; import { CaseLinkEnum, LastExecuteResults } from '@/enums/caseEnum';
import type { ExecuteStatusEnum } from '@/enums/taskCenter';
import { import {
type PlanMinderAssociateType, type PlanMinderAssociateType,
type PlanMinderCollectionType, type PlanMinderCollectionType,
@ -13,6 +14,8 @@ import {
testPlanTypeEnum, testPlanTypeEnum,
} from '@/enums/testPlanEnum'; } from '@/enums/testPlanEnum';
import type { TaskReportDetail } from '../apiTest/report';
export type planStatusType = 'PREPARED' | 'UNDERWAY' | 'COMPLETED' | 'ARCHIVED'; export type planStatusType = 'PREPARED' | 'UNDERWAY' | 'COMPLETED' | 'ARCHIVED';
export interface AssociateFunctionalCaseItem { export interface AssociateFunctionalCaseItem {
@ -383,7 +386,7 @@ export interface PlanDetailExecuteHistoryItem {
id: string; id: string;
num: string; num: string;
triggerMode: string; // 执行方式 triggerMode: string; // 执行方式
execResult: string; // 执行结果 execResult: ExecuteStatusEnum; // 执行结果
operationUser: string; operationUser: string;
startTime: number; startTime: number;
endTime: number; endTime: number;
@ -447,3 +450,17 @@ export interface PlanMinderEditParams {
editList: PlanMinderEditListItem[]; editList: PlanMinderEditListItem[];
deletedIds: string[]; deletedIds: string[];
} }
export interface PlanExecuteResultExecuteCaseCount {
success: number;
error: number;
fakeError: number;
block: number;
pending: number;
}
export interface PlanExecuteResult extends TaskReportDetail {
taskName: string;
reportId: string;
childPlans: { id: string; name: string }[]; // 子计划
createUser: string;
executeCaseCount: PlanExecuteResultExecuteCaseCount;
}

View File

@ -35,9 +35,16 @@
<template #status="{ record }"> <template #status="{ record }">
<ExecutionStatus :status="record.status" :module-type="ReportEnum.API_REPORT" /> <ExecutionStatus :status="record.status" :module-type="ReportEnum.API_REPORT" />
</template> </template>
<template #executeStatus="{ record }">
<ExecStatus :status="record.execStatus" />
</template>
<template #operation="{ record, rowIndex }"> <template #operation="{ record, rowIndex }">
<div v-if="record.historyDeleted"> <div v-if="record.historyDeleted">
<a-tooltip :content="t('common.executionResultCleaned')" position="top"> <a-tooltip
v-if="record.execStatus !== ExecuteStatusEnum.PENDING"
:content="t('common.executionResultCleaned')"
position="top"
>
<MsButton <MsButton
:disabled=" :disabled="
record.historyDeleted || record.historyDeleted ||
@ -51,6 +58,7 @@
</div> </div>
<div v-else> <div v-else>
<MsButton <MsButton
v-if="record.execStatus !== ExecuteStatusEnum.PENDING"
:disabled=" :disabled="
record.historyDeleted || record.historyDeleted ||
!hasAnyPermission(['PROJECT_API_DEFINITION_CASE:READ+EXECUTE', 'PROJECT_API_REPORT:READ']) !hasAnyPermission(['PROJECT_API_DEFINITION_CASE:READ+EXECUTE', 'PROJECT_API_REPORT:READ'])
@ -83,6 +91,7 @@
import useTable from '@/components/pure/ms-table/useTable'; import useTable from '@/components/pure/ms-table/useTable';
import MsTag from '@/components/pure/ms-tag/ms-tag.vue'; import MsTag from '@/components/pure/ms-tag/ms-tag.vue';
import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue'; import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue';
import ExecStatus from '@/views/taskCenter/component/execStatus.vue';
import { getApiCaseExecuteHistory } from '@/api/modules/api-test/management'; import { getApiCaseExecuteHistory } from '@/api/modules/api-test/management';
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
@ -90,9 +99,9 @@
import { hasAnyPermission } from '@/utils/permission'; import { hasAnyPermission } from '@/utils/permission';
import { ExecuteHistoryItem } from '@/models/apiTest/scenario'; import { ExecuteHistoryItem } from '@/models/apiTest/scenario';
// import { ReportExecStatus } from '@/enums/apiEnum';
import { ReportEnum, ReportStatus, TriggerModeLabel } from '@/enums/reportEnum'; import { ReportEnum, ReportStatus, TriggerModeLabel } from '@/enums/reportEnum';
import { FilterSlotNameEnum } from '@/enums/tableFilterEnum'; import { FilterSlotNameEnum } from '@/enums/tableFilterEnum';
import { ExecuteStatusEnum } from '@/enums/taskCenter';
import { triggerModeOptions } from '@/views/api-test/report/utils'; import { triggerModeOptions } from '@/views/api-test/report/utils';
@ -111,15 +120,6 @@
}); });
}); });
// const ExecStatusList = computed(() => {
// return Object.values(ReportExecStatus).map((e) => {
// return {
// value: e,
// key: e,
// };
// });
// });
const showResponse = ref(false); const showResponse = ref(false);
const props = defineProps<{ const props = defineProps<{
@ -152,6 +152,12 @@
}, },
width: 150, width: 150,
}, },
{
title: 'ms.taskCenter.executeStatus',
dataIndex: 'executeStatus',
slotName: 'executeStatus',
width: 150,
},
{ {
title: 'report.result', title: 'report.result',
dataIndex: 'status', dataIndex: 'status',

View File

@ -25,9 +25,16 @@
<template #status="{ record }"> <template #status="{ record }">
<ExecutionStatus :status="record.status" :module-type="ReportEnum.API_SCENARIO_REPORT" /> <ExecutionStatus :status="record.status" :module-type="ReportEnum.API_SCENARIO_REPORT" />
</template> </template>
<template #executeStatus="{ record }">
<ExecStatus :status="record.execStatus" />
</template>
<template #operation="{ record }"> <template #operation="{ record }">
<div v-if="record.historyDeleted"> <div v-if="record.historyDeleted">
<a-tooltip :content="t('common.executionResultCleaned')" position="top"> <a-tooltip
v-if="record.execStatus !== ExecuteStatusEnum.PENDING"
:content="t('common.executionResultCleaned')"
position="top"
>
<MsButton <MsButton
:disabled=" :disabled="
record.historyDeleted || record.historyDeleted ||
@ -41,6 +48,7 @@
</div> </div>
<div v-else> <div v-else>
<MsButton <MsButton
v-if="record.execStatus !== ExecuteStatusEnum.PENDING"
:disabled=" :disabled="
record.historyDeleted || record.historyDeleted ||
!hasAnyPermission(['PROJECT_API_SCENARIO:READ+EXECUTE', 'PROJECT_API_REPORT:READ']) !hasAnyPermission(['PROJECT_API_SCENARIO:READ+EXECUTE', 'PROJECT_API_REPORT:READ'])
@ -74,15 +82,16 @@
import useTable from '@/components/pure/ms-table/useTable'; import useTable from '@/components/pure/ms-table/useTable';
import MsTag from '@/components/pure/ms-tag/ms-tag.vue'; import MsTag from '@/components/pure/ms-tag/ms-tag.vue';
import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue'; import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue';
import ExecStatus from '@/views/taskCenter/component/execStatus.vue';
import { getExecuteHistory } from '@/api/modules/api-test/scenario'; import { getExecuteHistory } from '@/api/modules/api-test/scenario';
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
import { hasAnyPermission } from '@/utils/permission'; import { hasAnyPermission } from '@/utils/permission';
import { ExecuteHistoryItem } from '@/models/apiTest/scenario'; import { ExecuteHistoryItem } from '@/models/apiTest/scenario';
import { ReportExecStatus } from '@/enums/apiEnum';
import { ReportEnum, ReportStatus, TriggerModeLabel } from '@/enums/reportEnum'; import { ReportEnum, ReportStatus, TriggerModeLabel } from '@/enums/reportEnum';
import { FilterSlotNameEnum } from '@/enums/tableFilterEnum'; import { FilterSlotNameEnum } from '@/enums/tableFilterEnum';
import { ExecuteStatusEnum } from '@/enums/taskCenter';
import { triggerModeOptions } from '@/views/api-test/report/utils'; import { triggerModeOptions } from '@/views/api-test/report/utils';
@ -108,15 +117,6 @@
}); });
}); });
const ExecStatusList = computed(() => {
return Object.values(ReportExecStatus).map((e) => {
return {
value: e,
key: e,
};
});
});
const columns: MsTableColumn = [ const columns: MsTableColumn = [
{ {
title: 'apiScenario.executeHistory.num', title: 'apiScenario.executeHistory.num',
@ -135,6 +135,12 @@
}, },
width: 100, width: 100,
}, },
{
title: 'ms.taskCenter.executeStatus',
dataIndex: 'executeStatus',
slotName: 'executeStatus',
width: 150,
},
{ {
title: 'report.result', title: 'report.result',
dataIndex: 'status', dataIndex: 'status',

View File

@ -1527,7 +1527,7 @@
...propsRes.value.filter, ...propsRes.value.filter,
}; };
const { selectedIds, selectAll, excludeIds } = batchParams.value; const { selectedIds, selectAll, excludeIds } = batchParams.value;
await batchEditScenario({ await scenarioBatchEditSchedule({
selectIds: selectedIds || [], selectIds: selectedIds || [],
selectAll: !!selectAll, selectAll: !!selectAll,
excludeIds: excludeIds || [], excludeIds: excludeIds || [],
@ -1538,7 +1538,7 @@
keyword: keyword.value, keyword: keyword.value,
}, },
type: 'Schedule', type: 'Schedule',
scheduleOpen: enable, enable,
}); });
Message.success( Message.success(
enable enable

View File

@ -44,10 +44,10 @@
<execStatus :status="filterContent.value" /> <execStatus :status="filterContent.value" />
</template> </template>
<template #result="{ record }"> <template #result="{ record }">
<executionStatus :status="record.result" /> <executeResultStatus :status="record.result" />
</template> </template>
<template #[FilterSlotNameEnum.GLOBAL_TASK_CENTER_EXEC_RESULT]="{ filterContent }"> <template #[FilterSlotNameEnum.GLOBAL_TASK_CENTER_EXEC_RESULT]="{ filterContent }">
<executionStatus :status="filterContent.value" /> <executeResultStatus :status="filterContent.value" />
</template> </template>
<template #triggerMode="{ record }"> <template #triggerMode="{ record }">
{{ t(executeMethodMap[record.triggerMode]) }} {{ t(executeMethodMap[record.triggerMode]) }}
@ -80,9 +80,15 @@
> >
{{ t('common.stop') }} {{ t('common.stop') }}
</MsButton> </MsButton>
<MsButton v-if="record.status !== ExecuteStatusEnum.PENDING" @click="checkExecuteResult(record)"> <a-tooltip
v-if="record.status !== ExecuteStatusEnum.PENDING"
:content="t('common.executionResultCleaned')"
:disabled="!record.deleted"
>
<MsButton :disabled="record.resultDeleted" @click="checkExecuteResult(record)">
{{ t('ms.taskCenter.executeResult') }} {{ t('ms.taskCenter.executeResult') }}
</MsButton> </MsButton>
</a-tooltip>
</template> </template>
</ms-base-table> </ms-base-table>
<caseExecuteResultDrawer <caseExecuteResultDrawer
@ -116,7 +122,7 @@
import MsTag from '@/components/pure/ms-tag/ms-tag.vue'; import MsTag from '@/components/pure/ms-tag/ms-tag.vue';
import MsCascader from '@/components/business/ms-cascader/index.vue'; import MsCascader from '@/components/business/ms-cascader/index.vue';
import execStatus from './execStatus.vue'; import execStatus from './execStatus.vue';
import executionStatus from './executionStatus.vue'; import executeResultStatus from './executeResultStatus.vue';
import { import {
getOrganizationExecuteTaskDetailList, getOrganizationExecuteTaskDetailList,

View File

@ -36,10 +36,10 @@
<execStatus :status="filterContent.value" /> <execStatus :status="filterContent.value" />
</template> </template>
<template #result="{ record }"> <template #result="{ record }">
<executionStatus :status="record.result" /> <executeResultStatus :status="record.result" />
</template> </template>
<template #[FilterSlotNameEnum.GLOBAL_TASK_CENTER_EXEC_RESULT]="{ filterContent }"> <template #[FilterSlotNameEnum.GLOBAL_TASK_CENTER_EXEC_RESULT]="{ filterContent }">
<executionStatus :status="filterContent.value" /> <executeResultStatus :status="filterContent.value" />
</template> </template>
<template #triggerMode="{ record }"> <template #triggerMode="{ record }">
{{ t(executeMethodMap[record.triggerMode]) }} {{ t(executeMethodMap[record.triggerMode]) }}
@ -127,7 +127,7 @@
import batchTaskReportDrawer from './batchTaskReportDrawer.vue'; import batchTaskReportDrawer from './batchTaskReportDrawer.vue';
import execStatus from './execStatus.vue'; import execStatus from './execStatus.vue';
import executeRatePopper from './executeRatePopper.vue'; import executeRatePopper from './executeRatePopper.vue';
import executionStatus from './executionStatus.vue'; import executeResultStatus from './executeResultStatus.vue';
import CaseReportDrawer from '@/views/api-test/report/component/caseReportDrawer.vue'; import CaseReportDrawer from '@/views/api-test/report/component/caseReportDrawer.vue';
import ReportDetailDrawer from '@/views/api-test/report/component/reportDetailDrawer.vue'; import ReportDetailDrawer from '@/views/api-test/report/component/reportDetailDrawer.vue';

View File

@ -6,7 +6,7 @@
:disabled="record.caseTotal === 0 || record.status === ExecuteStatusEnum.PENDING" :disabled="record.caseTotal === 0 || record.status === ExecuteStatusEnum.PENDING"
@popup-visible-change="($event) => handleExecuteRatePopVisibleChange($event)" @popup-visible-change="($event) => handleExecuteRatePopVisibleChange($event)"
> >
<div>{{ record.executeRate || '0.00' }}%</div> <div :class="props.class">{{ record.executeRate || '0.00' }}%</div>
<template #content> <template #content>
<a-spin :loading="record.loading" class="flex w-[130px] flex-col gap-[8px]"> <a-spin :loading="record.loading" class="flex w-[130px] flex-col gap-[8px]">
<div class="ms-taskCenter-execute-rate-item"> <div class="ms-taskCenter-execute-rate-item">
@ -62,12 +62,15 @@
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
import { TaskCenterStatisticsItem, TaskCenterTaskItem } from '@/models/taskCenter'; import { TaskCenterStatisticsItem, TaskCenterTaskItem } from '@/models/taskCenter';
import { PlanExecuteResultExecuteCaseCount } from '@/models/testPlan/testPlan';
import { ExecuteStatusEnum } from '@/enums/taskCenter'; import { ExecuteStatusEnum } from '@/enums/taskCenter';
import { executeFinishedRateMap } from './config'; import { executeFinishedRateMap } from './config';
const props = defineProps<{ const props = defineProps<{
executeTaskStatisticsRequest: (ids: string[]) => Promise<TaskCenterStatisticsItem[]>; class?: string;
executeCaseCount?: PlanExecuteResultExecuteCaseCount;
executeTaskStatisticsRequest?: (ids: string[]) => Promise<TaskCenterStatisticsItem[]>;
}>(); }>();
const { t } = useI18n(); const { t } = useI18n();
@ -80,7 +83,7 @@
}); });
async function handleExecuteRatePopVisibleChange(_visible: boolean) { async function handleExecuteRatePopVisibleChange(_visible: boolean) {
if (_visible) { if (_visible && props.executeTaskStatisticsRequest) {
try { try {
record.value.loading = true; record.value.loading = true;
const res = await props.executeTaskStatisticsRequest([record.value.id]); const res = await props.executeTaskStatisticsRequest([record.value.id]);
@ -96,6 +99,11 @@
} finally { } finally {
record.value.loading = false; record.value.loading = false;
} }
} else if (props.executeCaseCount) {
record.value.pendingCount = props.executeCaseCount.pending;
record.value.successCount = props.executeCaseCount.success;
record.value.fakeErrorCount = props.executeCaseCount.fakeError;
record.value.errorCount = props.executeCaseCount.error;
} }
} }
</script> </script>

View File

@ -7,6 +7,9 @@
<template #triggerMode="{ record }"> <template #triggerMode="{ record }">
<span>{{ t(TriggerModeLabel[record.triggerMode as keyof typeof TriggerModeLabel]) }}</span> <span>{{ t(TriggerModeLabel[record.triggerMode as keyof typeof TriggerModeLabel]) }}</span>
</template> </template>
<template #executeStatus="{ record }">
<ExecStatus :status="record.execStatus" />
</template>
<template #lastExecResult="{ record }"> <template #lastExecResult="{ record }">
<ExecutionStatus v-if="record.execResult" :status="record.execResult" :module-type="ReportEnum.API_REPORT" /> <ExecutionStatus v-if="record.execResult" :status="record.execResult" :module-type="ReportEnum.API_REPORT" />
</template> </template>
@ -18,7 +21,11 @@
</a-tooltip> </a-tooltip>
</template> </template>
<template #operation="{ record }"> <template #operation="{ record }">
<a-tooltip :content="t('common.executionResultCleaned')" :disabled="!record.deleted"> <a-tooltip
v-if="record.execStatus !== ExecuteStatusEnum.PENDING"
:content="t('common.executionResultCleaned')"
:disabled="!record.deleted"
>
<MsButton <MsButton
:disabled="record.deleted || !hasAnyPermission(['PROJECT_TEST_PLAN_REPORT:READ'])" :disabled="record.deleted || !hasAnyPermission(['PROJECT_TEST_PLAN_REPORT:READ'])"
class="!mr-0" class="!mr-0"
@ -42,6 +49,7 @@
import useTable from '@/components/pure/ms-table/useTable'; import useTable from '@/components/pure/ms-table/useTable';
import executeResultDrawer from '../executeResultDrawer.vue'; import executeResultDrawer from '../executeResultDrawer.vue';
import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue'; import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue';
import ExecStatus from '@/views/taskCenter/component/execStatus.vue';
import { getPlanDetailExecuteHistory } from '@/api/modules/test-plan/testPlan'; import { getPlanDetailExecuteHistory } from '@/api/modules/test-plan/testPlan';
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
@ -50,6 +58,7 @@
import type { PlanDetailExecuteHistoryItem } from '@/models/testPlan/testPlan'; import type { PlanDetailExecuteHistoryItem } from '@/models/testPlan/testPlan';
import { PlanReportStatus, ReportEnum, TriggerModeLabel } from '@/enums/reportEnum'; import { PlanReportStatus, ReportEnum, TriggerModeLabel } from '@/enums/reportEnum';
import { FilterSlotNameEnum } from '@/enums/tableFilterEnum'; import { FilterSlotNameEnum } from '@/enums/tableFilterEnum';
import { ExecuteStatusEnum } from '@/enums/taskCenter';
import { triggerModeOptions } from '@/views/api-test/report/utils'; import { triggerModeOptions } from '@/views/api-test/report/utils';
@ -88,6 +97,12 @@
showTooltip: true, showTooltip: true,
width: 150, width: 150,
}, },
{
title: 'ms.taskCenter.executeStatus',
dataIndex: 'executeStatus',
slotName: 'executeStatus',
width: 150,
},
{ {
title: 'common.executionResult', title: 'common.executionResult',
dataIndex: 'execResult', dataIndex: 'execResult',
@ -118,20 +133,11 @@
width: 100, width: 100,
}, },
]; ];
const { propsRes, propsEvent, loadList, setLoadListParams } = useTable( const { propsRes, propsEvent, loadList, setLoadListParams } = useTable(getPlanDetailExecuteHistory, {
getPlanDetailExecuteHistory,
{
columns, columns,
scroll: { x: '100%' }, scroll: { x: '100%' },
selectable: false, selectable: false,
}, });
(record) => {
return {
...record,
startTime: dayjs(record.startTime).format('YYYY-MM-DD HH:mm:ss'),
};
}
);
function loadExecuteList() { function loadExecuteList() {
setLoadListParams({ setLoadListParams({
@ -149,7 +155,7 @@
} }
function getStartAndEndTime(record: PlanDetailExecuteHistoryItem) { function getStartAndEndTime(record: PlanDetailExecuteHistoryItem) {
return `${dayjs(record.startTime).format('YYYY-MM-DD HH:mm:ss')}${t('common.to')}${ return `${record.startTime ? dayjs(record.startTime).format('YYYY-MM-DD HH:mm:ss') : '-'}${t('common.to')}${
record.endTime ? dayjs(record.endTime).format('YYYY-MM-DD HH:mm:ss') : '-' record.endTime ? dayjs(record.endTime).format('YYYY-MM-DD HH:mm:ss') : '-'
}`; }`;
} }

View File

@ -5,7 +5,7 @@
<a-tag :color="executeResultMap[props.planDetail.execResult]?.color"> <a-tag :color="executeResultMap[props.planDetail.execResult]?.color">
{{ t(executeResultMap[props.planDetail.execResult]?.label || '-') }} {{ t(executeResultMap[props.planDetail.execResult]?.label || '-') }}
</a-tag> </a-tag>
<div class="one-line-text flex-1">{{ detail.name }}</div> <div class="one-line-text flex-1">{{ detail.taskName }}</div>
</div> </div>
<div class="flex justify-end"> <div class="flex justify-end">
<MsButton type="icon" status="secondary" class="!rounded-[var(--border-radius-small)]" @click="init"> <MsButton type="icon" status="secondary" class="!rounded-[var(--border-radius-small)]" @click="init">
@ -15,23 +15,22 @@
</div> </div>
</template> </template>
<a-spin :loading="loading" class="block min-h-[200px]"> <a-spin :loading="loading" class="block min-h-[200px]">
<MsDescription :descriptions="detail.description" :column="3" :line-gap="8" one-line-value> <MsDescription :descriptions="detail.description" :column="2" :line-gap="8" one-line-value>
<template #value="{ item }"> <template #value="{ item }">
<execStatus <execStatus v-if="item.key === 'status'" :status="props.planDetail.execResult" size="small" />
v-if="item.key === 'status'"
:status="props.planDetail.execResult as unknown as ExecuteStatusEnum"
size="small"
/>
<a-select <a-select
v-else-if="item.key === 'testPlan'" v-else-if="item.key === 'testPlan'"
v-model:model-value="activePlan" v-model:model-value="activePlan"
:options="testPlanGroups" :options="testPlanGroups"
size="small"
@change="searchList"
></a-select> ></a-select>
<executeRatePopper <executeRatePopper
v-else-if="item.key === 'rate'" v-else-if="item.key === 'rate'"
v-model:visible="executeRateVisible" v-model:visible="executeRateVisible"
v-model:record="detail" v-model:record="detail"
:execute-task-statistics-request="fakeStatisticsRequest" :execute-case-count="detail.executeCaseCount"
class="inline-block"
/> />
<a-tooltip <a-tooltip
v-else v-else
@ -46,7 +45,13 @@
</template> </template>
</MsDescription> </MsDescription>
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<MsTab v-model:active-key="activeTable" :content-tab-list="contentTabList" :show-badge="false" /> <MsTab
v-model:active-key="activeTable"
:content-tab-list="contentTabList"
:show-badge="false"
class="no-content"
@change="searchList"
/>
<a-input-search <a-input-search
v-model:model-value="keyword" v-model:model-value="keyword"
:placeholder="t('report.detail.caseDetailSearchPlaceholder')" :placeholder="t('report.detail.caseDetailSearchPlaceholder')"
@ -89,6 +94,7 @@
</a-spin> </a-spin>
</MsDrawer> </MsDrawer>
<CaseAndScenarioReportDrawer <CaseAndScenarioReportDrawer
v-if="reportVisible"
v-model:visible="reportVisible" v-model:visible="reportVisible"
:report-id="apiReportId" :report-id="apiReportId"
do-not-show-share do-not-show-share
@ -99,6 +105,7 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { SelectOptionData } from '@arco-design/web-vue';
import dayjs from 'dayjs'; import dayjs from 'dayjs';
import MsButton from '@/components/pure/ms-button/index.vue'; import MsButton from '@/components/pure/ms-button/index.vue';
@ -110,12 +117,10 @@
import type { MsTableColumn } from '@/components/pure/ms-table/type'; import type { MsTableColumn } from '@/components/pure/ms-table/type';
import useTable from '@/components/pure/ms-table/useTable'; import useTable from '@/components/pure/ms-table/useTable';
import caseLevel from '@/components/business/ms-case-associate/caseLevel.vue'; import caseLevel from '@/components/business/ms-case-associate/caseLevel.vue';
import CaseAndScenarioReportDrawer from '@/views/api-test/components/caseAndScenarioReportDrawer.vue';
import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue'; import ExecutionStatus from '@/views/api-test/report/component/reportStatus.vue';
import execStatus from '@/views/taskCenter/component/execStatus.vue'; import execStatus from '@/views/taskCenter/component/execStatus.vue';
import executeRatePopper from '@/views/taskCenter/component/executeRatePopper.vue'; import executeRatePopper from '@/views/taskCenter/component/executeRatePopper.vue';
import { getCaseTaskReport } from '@/api/modules/api-test/report';
import { import {
getApiPage, getApiPage,
getScenarioPage, getScenarioPage,
@ -124,6 +129,7 @@
reportScenarioDetail, reportScenarioDetail,
reportStepDetail, reportStepDetail,
} from '@/api/modules/test-plan/report'; } from '@/api/modules/test-plan/report';
import { getTaskResult } from '@/api/modules/test-plan/testPlan';
import { useI18n } from '@/hooks/useI18n'; import { useI18n } from '@/hooks/useI18n';
import useOpenNewPage from '@/hooks/useOpenNewPage'; import useOpenNewPage from '@/hooks/useOpenNewPage';
@ -132,11 +138,14 @@
import { ReportEnum } from '@/enums/reportEnum'; import { ReportEnum } from '@/enums/reportEnum';
import { ApiTestRouteEnum } from '@/enums/routeEnum'; import { ApiTestRouteEnum } from '@/enums/routeEnum';
import { FilterSlotNameEnum } from '@/enums/tableFilterEnum'; import { FilterSlotNameEnum } from '@/enums/tableFilterEnum';
import { ExecuteStatusEnum } from '@/enums/taskCenter';
import { casePriorityOptions, lastReportStatusListOptions } from '@/views/api-test/components/config'; import { casePriorityOptions, lastReportStatusListOptions } from '@/views/api-test/components/config';
import { executeResultMap, executeStatusMap } from '@/views/taskCenter/component/config'; import { executeResultMap, executeStatusMap } from '@/views/taskCenter/component/config';
const CaseAndScenarioReportDrawer = defineAsyncComponent(
() => import('@/views/api-test/components/caseAndScenarioReportDrawer.vue')
);
const props = defineProps<{ const props = defineProps<{
planDetail: PlanDetailExecuteHistoryItem; planDetail: PlanDetailExecuteHistoryItem;
}>(); }>();
@ -147,72 +156,9 @@
const visible = defineModel<boolean>('visible', { required: true }); const visible = defineModel<boolean>('visible', { required: true });
const loading = ref(false); const loading = ref(false);
const detail = ref<any>({ description: [] }); const detail = ref<any>({ description: [] });
const testPlanGroups = ref<any[]>([]); const testPlanGroups = ref<SelectOptionData[]>([]);
const executeRateVisible = ref(false); const executeRateVisible = ref(false);
const activePlan = ref('');
async function fakeStatisticsRequest() {
return Promise.resolve([]);
}
async function init() {
try {
loading.value = true;
const res = await getCaseTaskReport(props.planDetail.id);
const { apiReportDetailDTOList } = res;
const [caseDetail] = apiReportDetailDTOList;
detail.value = {
name: caseDetail?.requestName,
description: [
{
label: t('ms.taskCenter.executeStatus'),
key: 'status',
value: t(executeStatusMap[res.status].label),
},
{
label: t('ms.taskCenter.taskCreateTime'),
value: res.createTime ? dayjs(res.createTime).format('YYYY-MM-DD HH:mm:ss') : '-',
},
{
label: t('ms.taskCenter.operationUser'),
value: props.planDetail.operationUser,
},
{
label: t('ms.taskCenter.taskStartTime'),
value: res.startTime ? dayjs(res.startTime).format('YYYY-MM-DD HH:mm:ss') : '-',
},
{
label: t('ms.taskCenter.executeFinishedRate'),
key: 'rate',
value: res.result,
},
{
label: t('ms.taskCenter.taskEndTime'),
value: res.endTime ? dayjs(res.endTime).format('YYYY-MM-DD HH:mm:ss') : '-',
},
{
label: t('menu.testPlan'),
key: 'testPlan',
value: '',
},
] as Description[],
};
testPlanGroups.value = [
{
id: '1',
name: 'A',
},
{
id: '2',
name: 'B',
},
];
} catch (error) {
// eslint-disable-next-line no-console
console.log(error);
} finally {
loading.value = false;
}
}
const columns: MsTableColumn = [ const columns: MsTableColumn = [
{ {
@ -263,7 +209,6 @@
}, },
]; ];
const activePlan = ref('1');
const activeTable = ref<'case' | 'scenario'>('case'); const activeTable = ref<'case' | 'scenario'>('case');
const contentTabList = [ const contentTabList = [
{ value: 'case', label: t('report.detail.apiCaseDetails') }, { value: 'case', label: t('report.detail.apiCaseDetails') },
@ -311,25 +256,85 @@
} }
function searchList() { function searchList() {
currentCaseTable.value.setLoadListParams({ keyword: keyword.value }); currentCaseTable.value.setLoadListParams({
keyword: keyword.value,
reportId: detail.value.reportId,
planId: activePlan.value || detail.value.id,
});
currentCaseTable.value.loadList(); currentCaseTable.value.loadList();
} }
// //
function toDetail(record: PlanDetailExecuteHistoryItem) { function toDetail(record: ApiOrScenarioCaseItem) {
if (activeTable.value === 'scenario') { if (activeTable.value === 'scenario') {
openNewPage(ApiTestRouteEnum.API_TEST_SCENARIO, { openNewPage(ApiTestRouteEnum.API_TEST_SCENARIO, {
id: record.id, id: record.id,
pId: record.id, pId: record.projectId,
}); });
} else { } else {
openNewPage(ApiTestRouteEnum.API_TEST_MANAGEMENT, { openNewPage(ApiTestRouteEnum.API_TEST_MANAGEMENT, {
cId: record.id, cId: record.id,
pId: record.id, pId: record.projectId,
}); });
} }
} }
async function init() {
try {
loading.value = true;
const res = await getTaskResult(props.planDetail.id);
detail.value = {
description: [
{
label: t('ms.taskCenter.executeStatus'),
key: 'status',
value: t(executeStatusMap[res.status].label),
},
{
label: t('ms.taskCenter.taskCreateTime'),
value: res.createTime ? dayjs(res.createTime).format('YYYY-MM-DD HH:mm:ss') : '-',
},
{
label: t('ms.taskCenter.operationUser'),
value: res.createUser,
},
{
label: t('ms.taskCenter.taskStartTime'),
value: res.startTime ? dayjs(res.startTime).format('YYYY-MM-DD HH:mm:ss') : '-',
},
{
label: t('ms.taskCenter.executeFinishedRate'),
key: 'rate',
value: res.result,
},
{
label: t('ms.taskCenter.taskEndTime'),
value: res.endTime ? dayjs(res.endTime).format('YYYY-MM-DD HH:mm:ss') : '-',
},
] as Description[],
...res,
};
if (res.childPlans.length) {
detail.value.description.push({
label: t('testPlan.testPlanIndex.testPlan'),
key: 'testPlan',
value: '',
});
testPlanGroups.value = res.childPlans.map((item) => ({
value: item.id,
label: item.name,
}));
activePlan.value = res.childPlans[0]?.id;
}
searchList();
} catch (error) {
// eslint-disable-next-line no-console
console.log(error);
} finally {
loading.value = false;
}
}
watch( watch(
() => visible.value, () => visible.value,
(val) => { (val) => {

View File

@ -94,7 +94,7 @@
:content-tab-list="tabList" :content-tab-list="tabList"
:change-interceptor="changeTabInterceptor" :change-interceptor="changeTabInterceptor"
no-content no-content
class="relative mx-[16px] border-b" class="relative mx-[16px]"
/> />
</MsCard> </MsCard>
<MsCard class="mt-[16px]" simple has-breadcrumb no-content-padding> <MsCard class="mt-[16px]" simple has-breadcrumb no-content-padding>