fix(ui报告): 修改ui报告的样式
--bug=1024353 --user=宋天阳 【测试跟踪】测试计划执行报告,UI报告展示没有对齐 https://www.tapd.cn/55049933/s/1349610
This commit is contained in:
parent
094ca0b791
commit
96d88ba2de
|
@ -3,85 +3,147 @@
|
||||||
<el-row>
|
<el-row>
|
||||||
<el-col>
|
<el-col>
|
||||||
<span v-if="!debug">
|
<span v-if="!debug">
|
||||||
<el-input v-if="nameIsEdit" size="mini" @blur="handleSave(report.name)" @keyup.enter.native="handleSaveKeyUp"
|
<el-input
|
||||||
style="width: 200px" v-model="report.name" maxlength="60" show-word-limit/>
|
v-if="nameIsEdit"
|
||||||
|
size="mini"
|
||||||
|
@blur="handleSave(report.name)"
|
||||||
|
@keyup.enter.native="handleSaveKeyUp"
|
||||||
|
style="width: 200px"
|
||||||
|
v-model="report.name"
|
||||||
|
maxlength="60"
|
||||||
|
show-word-limit
|
||||||
|
/>
|
||||||
<span v-else>
|
<span v-else>
|
||||||
<router-link v-if="isSingleScenario"
|
<router-link
|
||||||
:to="{name: isUi ? 'uiAutomation' : 'ApiAutomation', params: { dataSelectRange: 'edit:' + scenarioId }}">
|
v-if="isSingleScenario"
|
||||||
|
:to="{
|
||||||
|
name: isUi ? 'uiAutomation' : 'ApiAutomation',
|
||||||
|
params: { dataSelectRange: 'edit:' + scenarioId },
|
||||||
|
}"
|
||||||
|
>
|
||||||
{{ report.name }}
|
{{ report.name }}
|
||||||
</router-link>
|
</router-link>
|
||||||
<span v-else>
|
<span v-else>
|
||||||
{{ report.name }}
|
{{ report.name }}
|
||||||
</span>
|
</span>
|
||||||
<i v-if="showCancelButton" class="el-icon-edit" style="cursor:pointer" @click="nameIsEdit = true"
|
<i
|
||||||
@click.stop/>
|
v-if="showCancelButton"
|
||||||
|
class="el-icon-edit"
|
||||||
|
style="cursor: pointer"
|
||||||
|
@click="nameIsEdit = true"
|
||||||
|
@click.stop
|
||||||
|
/>
|
||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
<span v-if="report.endTime || report.createTime">
|
<span v-if="report.endTime || report.createTime">
|
||||||
<span style="margin-left: 10px">{{ $t('report.test_start_time') }}:</span>
|
<span style="margin-left: 10px"
|
||||||
|
>{{ $t("report.test_start_time") }}:</span
|
||||||
|
>
|
||||||
<span class="time"> {{ report.createTime | datetimeFormat }}</span>
|
<span class="time"> {{ report.createTime | datetimeFormat }}</span>
|
||||||
<span style="margin-left: 10px">{{ $t('report.test_end_time') }}:</span>
|
<span style="margin-left: 10px"
|
||||||
|
>{{ $t("report.test_end_time") }}:</span
|
||||||
|
>
|
||||||
<span class="time"> {{ report.endTime | datetimeFormat }}</span>
|
<span class="time"> {{ report.endTime | datetimeFormat }}</span>
|
||||||
</span>
|
</span>
|
||||||
<div style="float: right">
|
<div style="float: right">
|
||||||
<el-button v-if="!isPlan && (!debug || exportFlag) && !isTemplate && !isUi"
|
<el-button
|
||||||
v-permission="['PROJECT_API_REPORT:READ+EXPORT']" :disabled="isReadOnly" class="export-button"
|
v-if="!isPlan && (!debug || exportFlag) && !isTemplate && !isUi"
|
||||||
plain type="primary" size="mini" @click="handleExport(report.name)" style="margin-right: 10px">
|
v-permission="['PROJECT_API_REPORT:READ+EXPORT']"
|
||||||
{{ $t('test_track.plan_view.export_report') }}
|
:disabled="isReadOnly"
|
||||||
|
class="export-button"
|
||||||
|
plain
|
||||||
|
type="primary"
|
||||||
|
size="mini"
|
||||||
|
@click="handleExport(report.name)"
|
||||||
|
style="margin-right: 10px"
|
||||||
|
>
|
||||||
|
{{ $t("test_track.plan_view.export_report") }}
|
||||||
</el-button>
|
</el-button>
|
||||||
|
|
||||||
<el-popover
|
<el-popover
|
||||||
v-if="!isPlan && (!debug || exportFlag) && !isTemplate"
|
v-if="!isPlan && (!debug || exportFlag) && !isTemplate"
|
||||||
v-permission="['PROJECT_PERFORMANCE_REPORT:READ+EXPORT']"
|
v-permission="['PROJECT_PERFORMANCE_REPORT:READ+EXPORT']"
|
||||||
style="margin-right: 10px;float: right;"
|
style="margin-right: 10px; float: right"
|
||||||
placement="bottom"
|
placement="bottom"
|
||||||
width="300">
|
width="300"
|
||||||
|
>
|
||||||
<p>{{ shareUrl }}</p>
|
<p>{{ shareUrl }}</p>
|
||||||
<span style="color: red;float: left;margin-left: 10px;" v-if="application.typeValue">{{
|
<span
|
||||||
$t('commons.validity_period') + application.typeValue
|
style="color: red; float: left; margin-left: 10px"
|
||||||
}}</span>
|
v-if="application.typeValue"
|
||||||
|
>{{ $t("commons.validity_period") + application.typeValue }}</span
|
||||||
|
>
|
||||||
<div style="text-align: right; margin: 0">
|
<div style="text-align: right; margin: 0">
|
||||||
<el-button type="primary" size="mini" :disabled="!shareUrl"
|
<el-button
|
||||||
v-clipboard:copy="shareUrl">{{ $t("commons.copy") }}
|
type="primary"
|
||||||
|
size="mini"
|
||||||
|
:disabled="!shareUrl"
|
||||||
|
v-clipboard:copy="shareUrl"
|
||||||
|
>{{ $t("commons.copy") }}
|
||||||
</el-button>
|
</el-button>
|
||||||
</div>
|
</div>
|
||||||
<el-button slot="reference" :disabled="isReadOnly" type="danger" plain size="mini"
|
<el-button
|
||||||
@click="handleShare(report)">
|
slot="reference"
|
||||||
{{ $t('test_track.plan_view.share_report') }}
|
:disabled="isReadOnly"
|
||||||
|
type="danger"
|
||||||
|
plain
|
||||||
|
size="mini"
|
||||||
|
@click="handleShare(report)"
|
||||||
|
>
|
||||||
|
{{ $t("test_track.plan_view.share_report") }}
|
||||||
</el-button>
|
</el-button>
|
||||||
</el-popover>
|
</el-popover>
|
||||||
|
|
||||||
<el-button v-if="showRerunButton" class="rerun-button" plain size="mini" @click="rerun">
|
<el-button
|
||||||
{{ $t('api_test.automation.rerun') }}
|
v-if="showRerunButton"
|
||||||
|
class="rerun-button"
|
||||||
|
plain
|
||||||
|
size="mini"
|
||||||
|
@click="rerun"
|
||||||
|
>
|
||||||
|
{{ $t("api_test.automation.rerun") }}
|
||||||
</el-button>
|
</el-button>
|
||||||
|
|
||||||
<el-button v-if="showCancelButton" class="export-button" plain size="mini" @click="returnView">
|
<el-button
|
||||||
{{ $t('commons.cancel') }}
|
v-if="showCancelButton"
|
||||||
|
class="export-button"
|
||||||
|
plain
|
||||||
|
size="mini"
|
||||||
|
@click="returnView"
|
||||||
|
>
|
||||||
|
{{ $t("commons.cancel") }}
|
||||||
</el-button>
|
</el-button>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</el-col>
|
</el-col>
|
||||||
</el-row>
|
</el-row>
|
||||||
<el-row v-if="showProjectEnv" type="flex">
|
<el-row v-if="showProjectEnv" type="flex">
|
||||||
<span> {{ $t('commons.environment') + ':' }} </span>
|
<span> {{ $t("commons.environment") + ":" }} </span>
|
||||||
<div v-for="(values,key) in projectEnvMap" :key="key" style="margin-right: 10px">
|
<div
|
||||||
|
v-for="(values, key) in projectEnvMap"
|
||||||
|
:key="key"
|
||||||
|
style="margin-right: 10px"
|
||||||
|
>
|
||||||
{{ key + ":" }}
|
{{ key + ":" }}
|
||||||
<ms-tag v-for="(item,index) in values" :key="index" type="success" :content="item"
|
<ms-tag
|
||||||
style="margin-left: 2px"/>
|
v-for="(item, index) in values"
|
||||||
|
:key="index"
|
||||||
|
type="success"
|
||||||
|
:content="item"
|
||||||
|
style="margin-left: 2px"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</el-row>
|
</el-row>
|
||||||
</header>
|
</header>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import { generateShareInfoWithExpired, getShareRedirectUrl } from "@/api/share";
|
||||||
import {generateShareInfoWithExpired, getShareRedirectUrl} from "@/api/share";
|
|
||||||
import MsTag from "metersphere-frontend/src/components/MsTag";
|
import MsTag from "metersphere-frontend/src/components/MsTag";
|
||||||
import {getCurrentProjectID} from "@/business/utils/sdk-utils";
|
import { getCurrentProjectID } from "@/business/utils/sdk-utils";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "MsApiReportViewHeader",
|
name: "MsApiReportViewHeader",
|
||||||
components: {MsTag},
|
components: { MsTag },
|
||||||
props: {
|
props: {
|
||||||
report: {},
|
report: {},
|
||||||
projectEnvMap: {},
|
projectEnvMap: {},
|
||||||
|
@ -99,17 +161,17 @@ export default {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
isPlan: Boolean
|
isPlan: Boolean,
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
showProjectEnv() {
|
showProjectEnv() {
|
||||||
return this.projectEnvMap && JSON.stringify(this.projectEnvMap) !== '{}';
|
return this.projectEnvMap && JSON.stringify(this.projectEnvMap) !== "{}";
|
||||||
},
|
},
|
||||||
path() {
|
path() {
|
||||||
return "/api/test/edit?id=" + this.report.testId;
|
return "/api/test/edit?id=" + this.report.testId;
|
||||||
},
|
},
|
||||||
scenarioId() {
|
scenarioId() {
|
||||||
if (typeof this.report.scenarioId === 'string') {
|
if (typeof this.report.scenarioId === "string") {
|
||||||
return this.report.scenarioId;
|
return this.report.scenarioId;
|
||||||
} else {
|
} else {
|
||||||
return "";
|
return "";
|
||||||
|
@ -132,79 +194,87 @@ export default {
|
||||||
isReadOnly: false,
|
isReadOnly: false,
|
||||||
nameIsEdit: false,
|
nameIsEdit: false,
|
||||||
shareUrl: "",
|
shareUrl: "",
|
||||||
application: {}
|
application: {},
|
||||||
}
|
};
|
||||||
},
|
|
||||||
created() {
|
|
||||||
},
|
},
|
||||||
|
created() {},
|
||||||
methods: {
|
methods: {
|
||||||
handleExport(name) {
|
handleExport(name) {
|
||||||
this.$emit('reportExport', name);
|
this.$emit("reportExport", name);
|
||||||
},
|
},
|
||||||
handleSave(name) {
|
handleSave(name) {
|
||||||
this.nameIsEdit = false;
|
this.nameIsEdit = false;
|
||||||
this.$emit('reportSave', name);
|
this.$emit("reportSave", name);
|
||||||
},
|
},
|
||||||
handleSaveKeyUp($event) {
|
handleSaveKeyUp($event) {
|
||||||
$event.target.blur();
|
$event.target.blur();
|
||||||
},
|
},
|
||||||
rerun() {
|
rerun() {
|
||||||
let type = this.report.reportType;
|
let type = this.report.reportType;
|
||||||
let rerunObj = {type: type, reportId: this.report.id}
|
let rerunObj = { type: type, reportId: this.report.id };
|
||||||
this.$post('/api/test/exec/rerun', rerunObj, res => {
|
this.$post("/api/test/exec/rerun", rerunObj, (res) => {
|
||||||
if (res.data !== 'SUCCESS') {
|
if (res.data !== "SUCCESS") {
|
||||||
this.$error(res.data);
|
this.$error(res.data);
|
||||||
} else {
|
} else {
|
||||||
this.$success(this.$t('api_test.automation.rerun_success'));
|
this.$success(this.$t("api_test.automation.rerun_success"));
|
||||||
this.returnView();
|
this.returnView();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
returnView() {
|
returnView() {
|
||||||
if (this.isUi) {
|
if (this.isUi) {
|
||||||
this.$router.push('/ui/report');
|
this.$router.push("/ui/report");
|
||||||
} else {
|
} else {
|
||||||
this.$router.push('/api/automation/report');
|
this.$router.push("/api/automation/report");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
handleShare(report) {
|
handleShare(report) {
|
||||||
this.getProjectApplication();
|
this.getProjectApplication();
|
||||||
let pram = {};
|
let pram = {};
|
||||||
pram.customData = report.id;
|
pram.customData = report.id;
|
||||||
pram.shareType = 'UI_REPORT';
|
pram.shareType = "UI_REPORT";
|
||||||
let thisHost = window.location.host;
|
let thisHost = window.location.host;
|
||||||
generateShareInfoWithExpired(pram).then((res) => {
|
generateShareInfoWithExpired(pram).then((res) => {
|
||||||
this.shareUrl = getShareRedirectUrl(res.data);
|
this.shareUrl = getShareRedirectUrl(res.data);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
getProjectApplication() {
|
getProjectApplication() {
|
||||||
let path = "/API_SHARE_REPORT_TIME";
|
let path = "/API_SHARE_REPORT_TIME";
|
||||||
if(this.isUi){
|
if (this.isUi) {
|
||||||
path = "/UI_SHARE_REPORT_TIME";
|
path = "/UI_SHARE_REPORT_TIME";
|
||||||
}
|
}
|
||||||
this.$get('/project_application/get/' + getCurrentProjectID() + path).then(res => {
|
this.$get(
|
||||||
|
"/project_application/get/" + getCurrentProjectID() + path
|
||||||
|
).then((res) => {
|
||||||
if (res.data && res.data.typeValue) {
|
if (res.data && res.data.typeValue) {
|
||||||
let quantity = res.data.typeValue.substring(0, res.data.typeValue.length - 1);
|
let quantity = res.data.typeValue.substring(
|
||||||
let unit = res.data.typeValue.substring(res.data.typeValue.length - 1);
|
0,
|
||||||
if (unit === 'H') {
|
res.data.typeValue.length - 1
|
||||||
res.data.typeValue = quantity + this.$t('commons.date_unit.hour');
|
);
|
||||||
} else if (unit === 'D') {
|
let unit = res.data.typeValue.substring(
|
||||||
res.data.typeValue = quantity + this.$t('commons.date_unit.day');
|
res.data.typeValue.length - 1
|
||||||
} else if (unit === 'M') {
|
);
|
||||||
res.data.typeValue = quantity + this.$t('commons.workspace_unit') + this.$t('commons.date_unit.month');
|
if (unit === "H") {
|
||||||
} else if (unit === 'Y') {
|
res.data.typeValue = quantity + this.$t("commons.date_unit.hour");
|
||||||
res.data.typeValue = quantity + this.$t('commons.date_unit.year');
|
} else if (unit === "D") {
|
||||||
|
res.data.typeValue = quantity + this.$t("commons.date_unit.day");
|
||||||
|
} else if (unit === "M") {
|
||||||
|
res.data.typeValue =
|
||||||
|
quantity +
|
||||||
|
this.$t("commons.workspace_unit") +
|
||||||
|
this.$t("commons.date_unit.month");
|
||||||
|
} else if (unit === "Y") {
|
||||||
|
res.data.typeValue = quantity + this.$t("commons.date_unit.year");
|
||||||
}
|
}
|
||||||
this.application = res.data;
|
this.application = res.data;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|
||||||
.export-button {
|
.export-button {
|
||||||
float: right;
|
float: right;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
|
@ -213,7 +283,10 @@ export default {
|
||||||
.rerun-button {
|
.rerun-button {
|
||||||
float: right;
|
float: right;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
background-color: #F2F9EF;
|
background-color: #f2f9ef;
|
||||||
color: #87C45D;
|
color: #87c45d;
|
||||||
|
}
|
||||||
|
.report-header {
|
||||||
|
min-width: 1200px;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -10,129 +10,251 @@
|
||||||
</el-row>
|
</el-row>
|
||||||
|
|
||||||
<div v-if="isExport">
|
<div v-if="isExport">
|
||||||
<span class="ms-req ms-req-error"
|
<span
|
||||||
v-if="(content.error && content.error>0 )|| (content.errorCode && content.errorCode>0)|| (content.unExecute && content.unExecute>0)">
|
class="ms-req ms-req-error"
|
||||||
<span class="ms-req-span"> {{ content.success + content.error + content.errorCode + content.unExecute }} {{
|
v-if="
|
||||||
isUi ? '指令' : $t('api_report.request')
|
(content.error && content.error > 0) ||
|
||||||
}}</span>
|
(content.errorCode && content.errorCode > 0) ||
|
||||||
</span>
|
(content.unExecute && content.unExecute > 0)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<span class="ms-req-span">
|
||||||
|
{{
|
||||||
|
content.success +
|
||||||
|
content.error +
|
||||||
|
content.errorCode +
|
||||||
|
content.unExecute
|
||||||
|
}}
|
||||||
|
{{ isUi ? "指令" : $t("api_report.request") }}</span
|
||||||
|
>
|
||||||
|
</span>
|
||||||
<span class="ms-req ms-req-success" v-else>
|
<span class="ms-req ms-req-success" v-else>
|
||||||
<span class="ms-req-span"> {{
|
<span class="ms-req-span">
|
||||||
content.success ? content.success + content.error : 0
|
{{ content.success ? content.success + content.error : 0 }}
|
||||||
}} {{ isUi ? '指令' : $t('api_report.request') }}</span>
|
{{ isUi ? "指令" : $t("api_report.request") }}</span
|
||||||
|
>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<ms-chart id="chart" ref="chart" :options="options" :height="220" style="margin-right: 10px"
|
<ms-chart
|
||||||
:autoresize="true" v-else/>
|
id="chart"
|
||||||
<el-row type="flex" justify="center" align="middle" style="width: 150px">
|
ref="chart"
|
||||||
|
:options="options"
|
||||||
|
:height="220"
|
||||||
|
style="margin-right: 10px"
|
||||||
|
:autoresize="true"
|
||||||
|
v-else
|
||||||
|
/>
|
||||||
|
<el-row
|
||||||
|
type="flex"
|
||||||
|
justify="center"
|
||||||
|
align="middle"
|
||||||
|
style="width: 150px"
|
||||||
|
>
|
||||||
<div>
|
<div>
|
||||||
<div class="metric-icon-box" style="height: 26px">
|
<div class="metric-icon-box" style="height: 26px">
|
||||||
<span class="ms-point-success" style="margin: 7px;float: left;"/>
|
<span
|
||||||
|
class="ms-point-success"
|
||||||
|
style="margin: 7px; float: left"
|
||||||
|
/>
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value" style="font-size: 12px">{{ content.success }} {{ $t('api_report.success') }}</div>
|
<div class="value" style="font-size: 12px">
|
||||||
|
{{ content.success }} {{ $t("api_report.success") }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<el-divider></el-divider>
|
<el-divider></el-divider>
|
||||||
<div class="metric-icon-box" style="height: 26px">
|
<div class="metric-icon-box" style="height: 26px">
|
||||||
<span class="ms-point-error" style="margin: 7px;float: left;"/>
|
<span class="ms-point-error" style="margin: 7px; float: left" />
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value" style="font-size: 12px">{{ content.error }} {{ $t('api_report.fail') }}</div>
|
<div class="value" style="font-size: 12px">
|
||||||
|
{{ content.error }} {{ $t("api_report.fail") }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<el-divider v-if="content.errorCode > 0"></el-divider>
|
<el-divider v-if="content.errorCode > 0"></el-divider>
|
||||||
<div class="metric-icon-box" v-if="content.errorCode > 0" style="height: 26px">
|
<div
|
||||||
<span class="ms-point-error-code" style="margin: 7px;float: left;"/>
|
class="metric-icon-box"
|
||||||
|
v-if="content.errorCode > 0"
|
||||||
|
style="height: 26px"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
class="ms-point-error-code"
|
||||||
|
style="margin: 7px; float: left"
|
||||||
|
/>
|
||||||
<div class="metric-box" v-if="content.errorCode > 0">
|
<div class="metric-box" v-if="content.errorCode > 0">
|
||||||
<div class="value" style="font-size: 12px">{{ content.errorCode }}
|
<div class="value" style="font-size: 12px">
|
||||||
{{ $t('error_report_library.option.name') }}
|
{{ content.errorCode }}
|
||||||
|
{{ $t("error_report_library.option.name") }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<el-divider v-if="content.unExecute > 0"></el-divider>
|
<el-divider v-if="content.unExecute > 0"></el-divider>
|
||||||
<div class="metric-icon-box" style="height: 26px" v-if="content.unExecute > 0">
|
<div
|
||||||
<span class="ms-point-unexecute" style="margin: 7px;float: left;"/>
|
class="metric-icon-box"
|
||||||
|
style="height: 26px"
|
||||||
|
v-if="content.unExecute > 0"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
class="ms-point-unexecute"
|
||||||
|
style="margin: 7px; float: left"
|
||||||
|
/>
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value" style="font-size: 12px">{{ content.unExecute }}
|
<div class="value" style="font-size: 12px">
|
||||||
{{ $t('api_test.home_page.detail_card.unexecute') }}
|
{{ content.unExecute }}
|
||||||
|
{{ $t("api_test.home_page.detail_card.unexecute") }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</el-row>
|
</el-row>
|
||||||
</el-row>
|
</el-row>
|
||||||
</div>
|
</div>
|
||||||
<div class="split"></div>
|
<div class="split"></div>
|
||||||
<!-- 场景统计 -->
|
<!-- 场景统计 -->
|
||||||
<div style="width: 50%">
|
<div style="width: 50%">
|
||||||
<el-row type="flex" justify="center" align="middle" v-if="report.reportType !== 'API_INTEGRATED'">
|
<el-row
|
||||||
|
type="flex"
|
||||||
|
justify="center"
|
||||||
|
align="middle"
|
||||||
|
v-if="report.reportType !== 'API_INTEGRATED'"
|
||||||
|
>
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value">{{ content.scenarioTotal ? content.scenarioTotal : 0 }}</div>
|
<div class="value">
|
||||||
<div class="name">{{ $t('api_test.scenario.scenario') }}</div>
|
{{ content.scenarioTotal ? content.scenarioTotal : 0 }}
|
||||||
|
</div>
|
||||||
|
<div class="name">{{ $t("api_test.scenario.scenario") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<span class="ms-point-success"/>
|
<span class="ms-point-success" />
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value">{{ content.scenarioSuccess ? content.scenarioSuccess : 0 }}</div>
|
<div class="value">
|
||||||
<div class="name">{{ $t('api_report.success') }}</div>
|
{{ content.scenarioSuccess ? content.scenarioSuccess : 0 }}
|
||||||
|
</div>
|
||||||
|
<div class="name">{{ $t("api_report.success") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<span class="ms-point-error"/>
|
<span class="ms-point-error" />
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value">{{ content.scenarioError ? content.scenarioError : 0 }}</div>
|
<div class="value">
|
||||||
<div class="name">{{ $t('api_report.fail') }}</div>
|
{{ content.scenarioError ? content.scenarioError : 0 }}
|
||||||
|
</div>
|
||||||
|
<div class="name">{{ $t("api_report.fail") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<span class="ms-point-error-code"
|
<span
|
||||||
v-if="content.scenarioErrorReport > 0 || content.scenarioStepErrorReport > 0 "/>
|
class="ms-point-error-code"
|
||||||
<div class="metric-box" v-if="content.scenarioErrorReport > 0 || content.scenarioStepErrorReport > 0 ">
|
v-if="
|
||||||
<div class="value">{{ content.scenarioErrorReport ? content.scenarioErrorReport : 0 }}</div>
|
content.scenarioErrorReport > 0 ||
|
||||||
<div class="name">{{ $t('error_report_library.option.name') }}</div>
|
content.scenarioStepErrorReport > 0
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
class="metric-box"
|
||||||
|
v-if="
|
||||||
|
content.scenarioErrorReport > 0 ||
|
||||||
|
content.scenarioStepErrorReport > 0
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<div class="value">
|
||||||
|
{{
|
||||||
|
content.scenarioErrorReport ? content.scenarioErrorReport : 0
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
<div class="name">{{ $t("error_report_library.option.name") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<span v-show="showUnExecuteReport" class="ms-point-unexecute"/>
|
<span v-show="showUnExecuteReport" class="ms-point-unexecute" />
|
||||||
<div v-show="showUnExecuteReport" class="metric-box">
|
<div v-show="showUnExecuteReport" class="metric-box">
|
||||||
<div class="value">{{ content.scenarioUnExecute ? content.scenarioUnExecute : 0 }}</div>
|
<div class="value">
|
||||||
<div class="name">{{ $t('api_test.home_page.detail_card.unexecute') }}</div>
|
{{ content.scenarioUnExecute ? content.scenarioUnExecute : 0 }}
|
||||||
|
</div>
|
||||||
|
<div class="name">
|
||||||
|
{{ $t("api_test.home_page.detail_card.unexecute") }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</el-row>
|
</el-row>
|
||||||
<el-divider v-if="report.reportType !== 'API_INTEGRATED'"/>
|
<el-divider v-if="report.reportType !== 'API_INTEGRATED'" />
|
||||||
<el-row type="flex" justify="center" align="middle">
|
<el-row type="flex" justify="center" align="middle">
|
||||||
<el-row type="flex" justify="center" align="middle">
|
<el-row type="flex" justify="center" align="middle">
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value">{{ content.scenarioStepTotal ? content.scenarioStepTotal : 0 }}</div>
|
<div class="value">
|
||||||
|
{{ content.scenarioStepTotal ? content.scenarioStepTotal : 0 }}
|
||||||
|
</div>
|
||||||
<div class="name" v-if="report.reportType === 'API_INTEGRATED'">
|
<div class="name" v-if="report.reportType === 'API_INTEGRATED'">
|
||||||
{{ $t('api_test.definition.request.case') }}
|
{{ $t("api_test.definition.request.case") }}
|
||||||
|
</div>
|
||||||
|
<div class="name" v-else>
|
||||||
|
{{ $t("test_track.plan_view.step") }}
|
||||||
</div>
|
</div>
|
||||||
<div class="name" v-else>{{ $t('test_track.plan_view.step') }}</div>
|
|
||||||
</div>
|
</div>
|
||||||
<span class="ms-point-success"/>
|
<span class="ms-point-success" />
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value">{{ content.scenarioStepSuccess ? content.scenarioStepSuccess : 0 }}</div>
|
<div class="value">
|
||||||
<div class="name">{{ $t('api_report.success') }}</div>
|
{{
|
||||||
|
content.scenarioStepSuccess ? content.scenarioStepSuccess : 0
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
<div class="name">{{ $t("api_report.success") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<span class="ms-point-error"/>
|
<span class="ms-point-error" />
|
||||||
<div class="metric-box">
|
<div class="metric-box">
|
||||||
<div class="value">{{ content.scenarioStepError ? content.scenarioStepError : 0 }}</div>
|
<div class="value">
|
||||||
<div class="name">{{ $t('api_report.fail') }}</div>
|
{{ content.scenarioStepError ? content.scenarioStepError : 0 }}
|
||||||
|
</div>
|
||||||
|
<div class="name">{{ $t("api_report.fail") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<span class="ms-point-error-code"
|
<span
|
||||||
v-if="content.scenarioErrorReport > 0 || content.scenarioStepErrorReport > 0 "/>
|
class="ms-point-error-code"
|
||||||
<div class="metric-box" v-if="content.scenarioErrorReport > 0 || content.scenarioStepErrorReport > 0 ">
|
v-if="
|
||||||
<div class="value">{{ content.scenarioStepErrorReport ? content.scenarioStepErrorReport : 0 }}</div>
|
content.scenarioErrorReport > 0 ||
|
||||||
<div class="name">{{ $t('error_report_library.option.name') }}</div>
|
content.scenarioStepErrorReport > 0
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
class="metric-box"
|
||||||
|
v-if="
|
||||||
|
content.scenarioErrorReport > 0 ||
|
||||||
|
content.scenarioStepErrorReport > 0
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<div class="value">
|
||||||
|
{{
|
||||||
|
content.scenarioStepErrorReport
|
||||||
|
? content.scenarioStepErrorReport
|
||||||
|
: 0
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
<div class="name">
|
||||||
|
{{ $t("error_report_library.option.name") }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<span v-show="showUnExecuteReport && !isUi" class="ms-point-unexecute"/>
|
<span
|
||||||
|
v-show="showUnExecuteReport && !isUi"
|
||||||
|
class="ms-point-unexecute"
|
||||||
|
/>
|
||||||
<div v-show="showUnExecuteReport && !isUi" class="metric-box">
|
<div v-show="showUnExecuteReport && !isUi" class="metric-box">
|
||||||
<div class="value">{{
|
<div class="value">
|
||||||
content.scenarioStepUnExecuteReport ? content.scenarioStepUnExecuteReport : 0
|
{{
|
||||||
|
content.scenarioStepUnExecuteReport
|
||||||
|
? content.scenarioStepUnExecuteReport
|
||||||
|
: 0
|
||||||
}}
|
}}
|
||||||
</div>
|
</div>
|
||||||
<div class="name">{{ $t('api_test.home_page.detail_card.unexecute') }}</div>
|
<div class="name">
|
||||||
|
{{ $t("api_test.home_page.detail_card.unexecute") }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<span v-show="showUnExecuteReport && isUi" class="ms-point-unexecute"/>
|
<span
|
||||||
|
v-show="showUnExecuteReport && isUi"
|
||||||
|
class="ms-point-unexecute"
|
||||||
|
/>
|
||||||
<div v-show="showUnExecuteReport && isUi" class="metric-box">
|
<div v-show="showUnExecuteReport && isUi" class="metric-box">
|
||||||
<div class="value">{{
|
<div class="value">
|
||||||
content.scenarioStepUnExecuteReport ? content.scenarioStepUnExecuteReport : 0
|
{{
|
||||||
|
content.scenarioStepUnExecuteReport
|
||||||
|
? content.scenarioStepUnExecuteReport
|
||||||
|
: 0
|
||||||
}}
|
}}
|
||||||
</div>
|
</div>
|
||||||
<div class="name">{{ $t('api_test.home_page.detail_card.unexecute') }}</div>
|
<div class="name">
|
||||||
|
{{ $t("api_test.home_page.detail_card.unexecute") }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</el-row>
|
</el-row>
|
||||||
</el-row>
|
</el-row>
|
||||||
|
@ -144,22 +266,24 @@
|
||||||
<div class="metric-icon-box">
|
<div class="metric-icon-box">
|
||||||
<i class="el-icon-warning-outline fail"></i>
|
<i class="el-icon-warning-outline fail"></i>
|
||||||
<div class="value">{{ fail }}</div>
|
<div class="value">{{ fail }}</div>
|
||||||
<div class="name">{{ $t('api_report.fail') }}</div>
|
<div class="name">{{ $t("api_report.fail") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="metric-icon-box">
|
<div class="metric-icon-box">
|
||||||
<i class="el-icon-document-checked assertions"></i>
|
<i class="el-icon-document-checked assertions"></i>
|
||||||
<div class="value">{{ assertions }}</div>
|
<div class="value">{{ assertions }}</div>
|
||||||
<div class="name">{{ $t('api_report.assertions_pass') }}</div>
|
<div class="name">{{ $t("api_report.assertions_pass") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="metric-icon-box" v-if="content.errorCode > 0">
|
<div class="metric-icon-box" v-if="content.errorCode > 0">
|
||||||
<i class="el-icon-document-checked assertions"></i>
|
<i class="el-icon-document-checked assertions"></i>
|
||||||
<div class="value">{{ errorCodeAssertions }}</div>
|
<div class="value">{{ errorCodeAssertions }}</div>
|
||||||
<div class="name">{{ $t('error_report_library.assertion') }}</div>
|
<div class="name">{{ $t("error_report_library.assertion") }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="metric-icon-box" v-if="!isUi">
|
<div class="metric-icon-box" v-if="!isUi">
|
||||||
<i class="el-icon-document-copy total"></i>
|
<i class="el-icon-document-copy total"></i>
|
||||||
<div class="value">{{ this.content.total }}</div>
|
<div class="value">{{ this.content.total }}</div>
|
||||||
<div class="name">{{ isUi ? '指令' : $t('api_report.request') }}</div>
|
<div class="name">
|
||||||
|
{{ isUi ? "指令" : $t("api_report.request") }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</el-row>
|
</el-row>
|
||||||
</div>
|
</div>
|
||||||
|
@ -172,7 +296,7 @@ import MsChart from "metersphere-frontend/src/components/chart/MsChart";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "MsMetricChart",
|
name: "MsMetricChart",
|
||||||
components: {MsChart},
|
components: { MsChart },
|
||||||
props: {
|
props: {
|
||||||
report: Object,
|
report: Object,
|
||||||
content: Object,
|
content: Object,
|
||||||
|
@ -180,7 +304,7 @@ export default {
|
||||||
isExport: {
|
isExport: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false,
|
default: false,
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
@ -192,7 +316,7 @@ export default {
|
||||||
scenarioSuccess: 0,
|
scenarioSuccess: 0,
|
||||||
scenarioError: 0,
|
scenarioError: 0,
|
||||||
reqTotal: 0,
|
reqTotal: 0,
|
||||||
}
|
};
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
this.initTime();
|
this.initTime();
|
||||||
|
@ -200,25 +324,25 @@ export default {
|
||||||
methods: {
|
methods: {
|
||||||
initTime() {
|
initTime() {
|
||||||
this.time = this.totalTime;
|
this.time = this.totalTime;
|
||||||
this.seconds = (this.time) / 1000;
|
this.seconds = this.time / 1000;
|
||||||
if (this.seconds >= 1) {
|
if (this.seconds >= 1) {
|
||||||
if (this.seconds < 60) {
|
if (this.seconds < 60) {
|
||||||
this.seconds = Math.round(this.seconds * 100 / 1) / 100;
|
this.seconds = Math.round((this.seconds * 100) / 1) / 100;
|
||||||
this.time = this.seconds + "s"
|
this.time = this.seconds + "s";
|
||||||
}
|
}
|
||||||
if (this.seconds > 60) {
|
if (this.seconds > 60) {
|
||||||
this.minutes = Math.round(this.seconds / 60)
|
this.minutes = Math.round(this.seconds / 60);
|
||||||
this.seconds = Math.round(this.seconds * 100 % 60) / 100;
|
this.seconds = Math.round((this.seconds * 100) % 60) / 100;
|
||||||
this.time = this.minutes + "min" + this.seconds + "s"
|
this.time = this.minutes + "min" + this.seconds + "s";
|
||||||
}
|
}
|
||||||
if (this.minutes > 60) {
|
if (this.minutes > 60) {
|
||||||
this.hour = Math.round(this.minutes / 60)
|
this.hour = Math.round(this.minutes / 60);
|
||||||
this.minutes = Math.round(this.minutes % 60)
|
this.minutes = Math.round(this.minutes % 60);
|
||||||
this.time = this.hour + "hour" + this.minutes + "min" + this.seconds + "s"
|
this.time =
|
||||||
|
this.hour + "hour" + this.minutes + "min" + this.seconds + "s";
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
this.time = this.totalTime + "ms"
|
this.time = this.totalTime + "ms";
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -241,36 +365,38 @@ export default {
|
||||||
},
|
},
|
||||||
options() {
|
options() {
|
||||||
return {
|
return {
|
||||||
color: ['#67C23A', '#F56C6C', '#F6972A', '#9C9B9A'],
|
color: ["#67C23A", "#F56C6C", "#F6972A", "#9C9B9A"],
|
||||||
tooltip: {
|
tooltip: {
|
||||||
trigger: 'item',
|
trigger: "item",
|
||||||
formatter: '{b}: {c} ({d}%)'
|
formatter: "{b}: {c} ({d}%)",
|
||||||
},
|
},
|
||||||
title: [{
|
title: [
|
||||||
text: this.totalCount,
|
{
|
||||||
subtext: this.isUi ? '指令' : this.$t('api_report.request'),
|
text: this.totalCount,
|
||||||
top: 'center',
|
subtext: this.isUi ? "指令" : this.$t("api_report.request"),
|
||||||
left: 'center',
|
top: "center",
|
||||||
textStyle: {
|
left: "center",
|
||||||
rich: {
|
textStyle: {
|
||||||
align: 'center',
|
rich: {
|
||||||
value: {
|
align: "center",
|
||||||
fontSize: 32,
|
value: {
|
||||||
fontWeight: 'bold',
|
fontSize: 32,
|
||||||
padding: [10, 0]
|
fontWeight: "bold",
|
||||||
|
padding: [10, 0],
|
||||||
|
},
|
||||||
|
name: {
|
||||||
|
fontSize: 14,
|
||||||
|
fontWeight: "normal",
|
||||||
|
color: "#7F7F7F",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
name: {
|
},
|
||||||
fontSize: 14,
|
},
|
||||||
fontWeight: 'normal',
|
],
|
||||||
color: '#7F7F7F',
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}],
|
|
||||||
series: [
|
series: [
|
||||||
{
|
{
|
||||||
type: 'pie',
|
type: "pie",
|
||||||
radius: ['80%', '90%'],
|
radius: ["80%", "90%"],
|
||||||
avoidLabelOverlap: false,
|
avoidLabelOverlap: false,
|
||||||
hoverAnimation: false,
|
hoverAnimation: false,
|
||||||
label: {
|
label: {
|
||||||
|
@ -278,24 +404,33 @@ export default {
|
||||||
},
|
},
|
||||||
itemStyle: {
|
itemStyle: {
|
||||||
borderColor: "#FFF",
|
borderColor: "#FFF",
|
||||||
shadowColor: '#E1E1E1',
|
shadowColor: "#E1E1E1",
|
||||||
shadowBlur: 10
|
shadowBlur: 10,
|
||||||
},
|
},
|
||||||
labelLine: {
|
labelLine: {
|
||||||
show: false
|
show: false,
|
||||||
},
|
},
|
||||||
data: [
|
data: [
|
||||||
{value: this.content.success, name: this.$t('api_report.success')},
|
{
|
||||||
{value: this.content.error, name: this.$t('api_report.fail')},
|
value: this.content.success,
|
||||||
{value: this.content.errorCode, name: this.$t('error_report_library.option.name')},
|
name: this.$t("api_report.success"),
|
||||||
{value: this.content.unExecute, name: this.$t('api_test.home_page.detail_card.unexecute')},
|
},
|
||||||
]
|
{ value: this.content.error, name: this.$t("api_report.fail") },
|
||||||
}
|
{
|
||||||
]
|
value: this.content.errorCode,
|
||||||
|
name: this.$t("error_report_library.option.name"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: this.content.unExecute,
|
||||||
|
name: this.$t("api_test.home_page.detail_card.unexecute"),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
fail() {
|
fail() {
|
||||||
return (this.content.error / this.content.total * 100).toFixed(0) + "%";
|
return ((this.content.error / this.content.total) * 100).toFixed(0) + "%";
|
||||||
},
|
},
|
||||||
assertions() {
|
assertions() {
|
||||||
return this.content.passAssertions + " / " + this.content.totalAssertions;
|
return this.content.passAssertions + " / " + this.content.totalAssertions;
|
||||||
|
@ -307,16 +442,22 @@ export default {
|
||||||
return this.report.reportType && this.report.reportType.startsWith("UI");
|
return this.report.reportType && this.report.reportType.startsWith("UI");
|
||||||
},
|
},
|
||||||
showUnExecuteReport() {
|
showUnExecuteReport() {
|
||||||
return (this.content.scenarioStepUnExecuteReport && this.content.scenarioStepUnExecuteReport > 0)
|
return (
|
||||||
|| (this.content.scenarioUnExecute && this.content.scenarioUnExecute > 0) || (this.content.unExecute && this.content.unExecute > 0);
|
(this.content.scenarioStepUnExecuteReport &&
|
||||||
|
this.content.scenarioStepUnExecuteReport > 0) ||
|
||||||
|
(this.content.scenarioUnExecute &&
|
||||||
|
this.content.scenarioUnExecute > 0) ||
|
||||||
|
(this.content.unExecute && this.content.unExecute > 0)
|
||||||
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.metric-container {
|
.metric-container {
|
||||||
padding: 5px 10px;
|
padding: 5px 10px;
|
||||||
|
min-width: 1200px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-container #chart {
|
.metric-container #chart {
|
||||||
|
@ -328,25 +469,25 @@ export default {
|
||||||
.metric-container .split {
|
.metric-container .split {
|
||||||
margin: 20px;
|
margin: 20px;
|
||||||
height: 100px;
|
height: 100px;
|
||||||
border-left: 1px solid #D8DBE1;
|
border-left: 1px solid #d8dbe1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-container .circle {
|
.metric-container .circle {
|
||||||
width: 12px;
|
width: 12px;
|
||||||
height: 12px;
|
height: 12px;
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
box-shadow: 0 0 20px 1px rgba(200, 216, 226, .42);
|
box-shadow: 0 0 20px 1px rgba(200, 216, 226, 0.42);
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-container .circle.success {
|
.metric-container .circle.success {
|
||||||
background-color: #67C23A;
|
background-color: #67c23a;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-container .circle.fail {
|
.metric-container .circle.fail {
|
||||||
background-color: #F56C6C;
|
background-color: #f56c6c;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-box {
|
.metric-box {
|
||||||
|
@ -358,18 +499,18 @@ export default {
|
||||||
.metric-box .value {
|
.metric-box .value {
|
||||||
font-size: 32px;
|
font-size: 32px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
letter-spacing: -.5px;
|
letter-spacing: -0.5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-time .value {
|
.metric-time .value {
|
||||||
font-size: 25px;
|
font-size: 25px;
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
letter-spacing: -.5px;
|
letter-spacing: -0.5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-box .name {
|
.metric-box .name {
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
letter-spacing: -.2px;
|
letter-spacing: -0.2px;
|
||||||
color: #404040;
|
color: #404040;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -381,7 +522,7 @@ export default {
|
||||||
.metric-icon-box .value {
|
.metric-icon-box .value {
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
letter-spacing: -.4px;
|
letter-spacing: -0.4px;
|
||||||
line-height: 28px;
|
line-height: 28px;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
@ -389,12 +530,12 @@ export default {
|
||||||
.metric-icon-box .name {
|
.metric-icon-box .name {
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
letter-spacing: 1px;
|
letter-spacing: 1px;
|
||||||
color: #BFBFBF;
|
color: #bfbfbf;
|
||||||
line-height: 18px;
|
line-height: 18px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.metric-icon-box .fail {
|
.metric-icon-box .fail {
|
||||||
color: #F56C6C;
|
color: #f56c6c;
|
||||||
font-size: 40px;
|
font-size: 40px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -416,11 +557,11 @@ export default {
|
||||||
}
|
}
|
||||||
|
|
||||||
.ms-req-error {
|
.ms-req-error {
|
||||||
border: 5px #F56C6C solid;
|
border: 5px #f56c6c solid;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ms-req-success {
|
.ms-req-success {
|
||||||
border: 5px #67C23A solid;
|
border: 5px #67c23a solid;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ms-req-span {
|
.ms-req-span {
|
||||||
|
@ -440,7 +581,7 @@ export default {
|
||||||
vertical-align: top;
|
vertical-align: top;
|
||||||
margin-left: 20px;
|
margin-left: 20px;
|
||||||
margin-right: 20px;
|
margin-right: 20px;
|
||||||
background-color: #67C23A;
|
background-color: #67c23a;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ms-point-error {
|
.ms-point-error {
|
||||||
|
@ -452,7 +593,7 @@ export default {
|
||||||
vertical-align: top;
|
vertical-align: top;
|
||||||
margin-left: 20px;
|
margin-left: 20px;
|
||||||
margin-right: 20px;
|
margin-right: 20px;
|
||||||
background-color: #F56C6C;
|
background-color: #f56c6c;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ms-point-error-code {
|
.ms-point-error-code {
|
||||||
|
@ -464,7 +605,7 @@ export default {
|
||||||
vertical-align: top;
|
vertical-align: top;
|
||||||
margin-left: 20px;
|
margin-left: 20px;
|
||||||
margin-right: 20px;
|
margin-right: 20px;
|
||||||
background-color: #F6972A;
|
background-color: #f6972a;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ms-point-unexecute {
|
.ms-point-unexecute {
|
||||||
|
@ -476,6 +617,6 @@ export default {
|
||||||
vertical-align: top;
|
vertical-align: top;
|
||||||
margin-left: 20px;
|
margin-left: 20px;
|
||||||
margin-right: 20px;
|
margin-right: 20px;
|
||||||
background-color: #9C9B9A;
|
background-color: #9c9b9a;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -13,14 +13,22 @@
|
||||||
:report="report"
|
:report="report"
|
||||||
:project-env-map="projectEnvMap"
|
:project-env-map="projectEnvMap"
|
||||||
@reportExport="handleExport"
|
@reportExport="handleExport"
|
||||||
@reportSave="handleSave"/>
|
@reportSave="handleSave"
|
||||||
|
/>
|
||||||
|
|
||||||
<!-- content -->
|
<!-- content -->
|
||||||
<main v-if="isNotRunning">
|
<main v-if="isNotRunning">
|
||||||
<!-- content header chart -->
|
<!-- content header chart -->
|
||||||
<ms-metric-chart :content="content" :totalTime="totalTime" :report="report"/>
|
<ms-metric-chart
|
||||||
|
:content="content"
|
||||||
<el-tabs v-model="activeName" @tab-click="handleClick">
|
:totalTime="totalTime"
|
||||||
|
:report="report"
|
||||||
|
/>
|
||||||
|
<el-tabs
|
||||||
|
v-model="activeName"
|
||||||
|
@tab-click="handleClick"
|
||||||
|
style="min-width: 1200px"
|
||||||
|
>
|
||||||
<!-- all step-->
|
<!-- all step-->
|
||||||
<el-tab-pane :label="$t('api_report.total')" name="total">
|
<el-tab-pane :label="$t('api_report.total')" name="total">
|
||||||
<ms-scenario-results
|
<ms-scenario-results
|
||||||
|
@ -31,12 +39,13 @@
|
||||||
:is-share="isShare"
|
:is-share="isShare"
|
||||||
:share-id="shareId"
|
:share-id="shareId"
|
||||||
v-on:requestResult="requestResult"
|
v-on:requestResult="requestResult"
|
||||||
ref="resultsTree"/>
|
ref="resultsTree"
|
||||||
|
/>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
<!-- fail step -->
|
<!-- fail step -->
|
||||||
<el-tab-pane name="fail">
|
<el-tab-pane name="fail">
|
||||||
<template slot="label">
|
<template slot="label">
|
||||||
<span class="fail">{{ $t('api_report.fail') }}</span>
|
<span class="fail">{{ $t("api_report.fail") }}</span>
|
||||||
</template>
|
</template>
|
||||||
<ms-scenario-results
|
<ms-scenario-results
|
||||||
v-on:requestResult="requestResult"
|
v-on:requestResult="requestResult"
|
||||||
|
@ -45,14 +54,16 @@
|
||||||
:is-share="isShare"
|
:is-share="isShare"
|
||||||
:is-template="isTemplate"
|
:is-template="isTemplate"
|
||||||
:share-id="shareId"
|
:share-id="shareId"
|
||||||
:treeData="fullTreeNodes" ref="failsTree"
|
:treeData="fullTreeNodes"
|
||||||
:errorReport="content.error"/>
|
ref="failsTree"
|
||||||
|
:errorReport="content.error"
|
||||||
|
/>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
<!--error step -->
|
<!--error step -->
|
||||||
<el-tab-pane name="errorReport" v-if="content.errorCode > 0">
|
<el-tab-pane name="errorReport" v-if="content.errorCode > 0">
|
||||||
<template slot="label">
|
<template slot="label">
|
||||||
<span class="fail" style="color: #F6972A">
|
<span class="fail" style="color: #f6972a">
|
||||||
{{ $t('error_report_library.option.name') }}
|
{{ $t("error_report_library.option.name") }}
|
||||||
</span>
|
</span>
|
||||||
</template>
|
</template>
|
||||||
<ms-scenario-results
|
<ms-scenario-results
|
||||||
|
@ -62,15 +73,16 @@
|
||||||
:is-template="isTemplate"
|
:is-template="isTemplate"
|
||||||
:share-id="shareId"
|
:share-id="shareId"
|
||||||
:console="content.console"
|
:console="content.console"
|
||||||
:treeData="fullTreeNodes" ref="errorReportTree"/>
|
:treeData="fullTreeNodes"
|
||||||
|
ref="errorReportTree"
|
||||||
|
/>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
<!-- Not performed step -->
|
<!-- Not performed step -->
|
||||||
<el-tab-pane name="unExecute" v-if="content.unExecute > 0">
|
<el-tab-pane name="unExecute" v-if="content.unExecute > 0">
|
||||||
<template slot="label">
|
<template slot="label">
|
||||||
<span class="fail"
|
<span class="fail" style="color: #9c9b9a">
|
||||||
style="color: #9C9B9A">
|
{{ $t("api_test.home_page.detail_card.unexecute") }}
|
||||||
{{ $t('api_test.home_page.detail_card.unexecute') }}
|
</span>
|
||||||
</span>
|
|
||||||
</template>
|
</template>
|
||||||
<ms-scenario-results
|
<ms-scenario-results
|
||||||
v-on:requestResult="requestResult"
|
v-on:requestResult="requestResult"
|
||||||
|
@ -79,18 +91,23 @@
|
||||||
:is-template="isTemplate"
|
:is-template="isTemplate"
|
||||||
:share-id="shareId"
|
:share-id="shareId"
|
||||||
:console="content.console"
|
:console="content.console"
|
||||||
:treeData="fullTreeNodes" ref="unExecuteTree"/>
|
:treeData="fullTreeNodes"
|
||||||
|
ref="unExecuteTree"
|
||||||
|
/>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
<!-- console -->
|
<!-- console -->
|
||||||
<el-tab-pane name="console">
|
<el-tab-pane name="console">
|
||||||
<template slot="label">
|
<template slot="label">
|
||||||
<span class="console">{{ $t('api_test.definition.request.console') }}</span>
|
<span class="console">{{
|
||||||
|
$t("api_test.definition.request.console")
|
||||||
|
}}</span>
|
||||||
</template>
|
</template>
|
||||||
<ms-code-edit
|
<ms-code-edit
|
||||||
:mode="'text'"
|
:mode="'text'"
|
||||||
:read-only="true"
|
:read-only="true"
|
||||||
:data.sync="content.console"
|
:data.sync="content.console"
|
||||||
height="calc(100vh - 500px)"/>
|
height="calc(100vh - 500px)"
|
||||||
|
/>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
</el-tabs>
|
</el-tabs>
|
||||||
</main>
|
</main>
|
||||||
|
@ -101,7 +118,6 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
|
||||||
import MsRequestResult from "./RequestResult";
|
import MsRequestResult from "./RequestResult";
|
||||||
import MsRequestResultTail from "./RequestResultTail";
|
import MsRequestResultTail from "./RequestResultTail";
|
||||||
import MsScenarioResult from "./ScenarioResult";
|
import MsScenarioResult from "./ScenarioResult";
|
||||||
|
@ -110,10 +126,19 @@ import MsScenarioResults from "./ScenarioResults";
|
||||||
import MsContainer from "metersphere-frontend/src/components/MsContainer";
|
import MsContainer from "metersphere-frontend/src/components/MsContainer";
|
||||||
import MsMainContainer from "metersphere-frontend/src/components/MsMainContainer";
|
import MsMainContainer from "metersphere-frontend/src/components/MsMainContainer";
|
||||||
import MsApiReportViewHeader from "./ApiReportViewHeader";
|
import MsApiReportViewHeader from "./ApiReportViewHeader";
|
||||||
import {STEP} from "./Setting";
|
import { STEP } from "./Setting";
|
||||||
import MsCodeEdit from "metersphere-frontend/src/components/MsCodeEdit";
|
import MsCodeEdit from "metersphere-frontend/src/components/MsCodeEdit";
|
||||||
import {getCurrentProjectID, getUUID, hasLicense, windowPrint} from "@/business/utils/sdk-utils";
|
import {
|
||||||
import {getScenarioReport, getScenarioReportAll, getShareScenarioReport} from "@/api/ui-report";
|
getCurrentProjectID,
|
||||||
|
getUUID,
|
||||||
|
hasLicense,
|
||||||
|
windowPrint,
|
||||||
|
} from "@/business/utils/sdk-utils";
|
||||||
|
import {
|
||||||
|
getScenarioReport,
|
||||||
|
getScenarioReportAll,
|
||||||
|
getShareScenarioReport,
|
||||||
|
} from "@/api/ui-report";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "UiShareReportDetail",
|
name: "UiShareReportDetail",
|
||||||
|
@ -121,7 +146,12 @@ export default {
|
||||||
MsApiReportViewHeader,
|
MsApiReportViewHeader,
|
||||||
MsMainContainer,
|
MsMainContainer,
|
||||||
MsCodeEdit,
|
MsCodeEdit,
|
||||||
MsContainer, MsScenarioResults, MsRequestResultTail, MsMetricChart, MsScenarioResult, MsRequestResult
|
MsContainer,
|
||||||
|
MsScenarioResults,
|
||||||
|
MsRequestResultTail,
|
||||||
|
MsMetricChart,
|
||||||
|
MsScenarioResult,
|
||||||
|
MsRequestResult,
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
@ -139,11 +169,11 @@ export default {
|
||||||
reportExportVisible: false,
|
reportExportVisible: false,
|
||||||
fullTreeNodes: [],
|
fullTreeNodes: [],
|
||||||
showRerunButton: false,
|
showRerunButton: false,
|
||||||
stepFilter: new STEP,
|
stepFilter: new STEP(),
|
||||||
exportReportIsOk: false,
|
exportReportIsOk: false,
|
||||||
tempResult: [],
|
tempResult: [],
|
||||||
projectEnvMap: {},
|
projectEnvMap: {},
|
||||||
}
|
};
|
||||||
},
|
},
|
||||||
activated() {
|
activated() {
|
||||||
this.isRequestResult = false;
|
this.isRequestResult = false;
|
||||||
|
@ -160,8 +190,8 @@ export default {
|
||||||
isPlan: Boolean,
|
isPlan: Boolean,
|
||||||
showCancelButton: {
|
showCancelButton: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: true
|
default: true,
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
reportId() {
|
reportId() {
|
||||||
|
@ -173,7 +203,7 @@ export default {
|
||||||
if (this.isTemplate) {
|
if (this.isTemplate) {
|
||||||
this.getReport();
|
this.getReport();
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
filter(index) {
|
filter(index) {
|
||||||
|
@ -198,20 +228,32 @@ export default {
|
||||||
this.showRerunButton = false;
|
this.showRerunButton = false;
|
||||||
},
|
},
|
||||||
rerunVerify() {
|
rerunVerify() {
|
||||||
if (hasLicense() && this.fullTreeNodes && this.fullTreeNodes.length > 0 && !this.isShare) {
|
if (
|
||||||
this.fullTreeNodes.forEach(item => {
|
hasLicense() &&
|
||||||
item.redirect = true;
|
this.fullTreeNodes &&
|
||||||
if (item.totalStatus === 'fail' || item.totalStatus === 'error' || item.unExecuteTotal > 0
|
this.fullTreeNodes.length > 0 &&
|
||||||
|| (item.type === "API" && item.totalStatus === 'unexecute')) {
|
!this.isShare
|
||||||
this.showRerunButton = true;
|
) {
|
||||||
}
|
this.fullTreeNodes.forEach((item) => {
|
||||||
|
item.redirect = true;
|
||||||
|
if (
|
||||||
|
item.totalStatus === "fail" ||
|
||||||
|
item.totalStatus === "error" ||
|
||||||
|
item.unExecuteTotal > 0 ||
|
||||||
|
(item.type === "API" && item.totalStatus === "unexecute")
|
||||||
|
) {
|
||||||
|
this.showRerunButton = true;
|
||||||
}
|
}
|
||||||
)
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
handleClick(tab, event) {
|
handleClick(tab, event) {
|
||||||
this.isRequestResult = false;
|
this.isRequestResult = false;
|
||||||
if (this.report && this.report.reportVersion && this.report.reportVersion > 1) {
|
if (
|
||||||
|
this.report &&
|
||||||
|
this.report.reportVersion &&
|
||||||
|
this.report.reportVersion > 1
|
||||||
|
) {
|
||||||
this.filter(tab.index);
|
this.filter(tab.index);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -219,28 +261,28 @@ export default {
|
||||||
this.isActive = !this.isActive;
|
this.isActive = !this.isActive;
|
||||||
},
|
},
|
||||||
formatResult(res) {
|
formatResult(res) {
|
||||||
let resMap = new Map;
|
let resMap = new Map();
|
||||||
let array = [];
|
let array = [];
|
||||||
if (res && res.scenarios) {
|
if (res && res.scenarios) {
|
||||||
res.scenarios.forEach(item => {
|
res.scenarios.forEach((item) => {
|
||||||
if (item && item.requestResults) {
|
if (item && item.requestResults) {
|
||||||
item.requestResults.forEach(req => {
|
item.requestResults.forEach((req) => {
|
||||||
req.responseResult.console = res.console;
|
req.responseResult.console = res.console;
|
||||||
resMap.set(req.id + req.name, req);
|
resMap.set(req.id + req.name, req);
|
||||||
req.name = item.name + "^@~@^" + req.name + "UUID=" + getUUID();
|
req.name = item.name + "^@~@^" + req.name + "UUID=" + getUUID();
|
||||||
array.push(req);
|
array.push(req);
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
this.formatTree(array, this.fullTreeNodes);
|
this.formatTree(array, this.fullTreeNodes);
|
||||||
this.sort(this.fullTreeNodes);
|
this.sort(this.fullTreeNodes);
|
||||||
this.$emit('refresh', resMap);
|
this.$emit("refresh", resMap);
|
||||||
},
|
},
|
||||||
formatTree(array, tree) {
|
formatTree(array, tree) {
|
||||||
array.map((item) => {
|
array.map((item) => {
|
||||||
let key = item.name;
|
let key = item.name;
|
||||||
let nodeArray = key.split('^@~@^');
|
let nodeArray = key.split("^@~@^");
|
||||||
let children = tree;
|
let children = tree;
|
||||||
//运行场景中如果连续将1个场景引入多次,会出现运行结果合并的情况。
|
//运行场景中如果连续将1个场景引入多次,会出现运行结果合并的情况。
|
||||||
//为了解决这种问题,在转hashTree的时候给场景放了个新ID,前台加载解析的时候也要做处理
|
//为了解决这种问题,在转hashTree的时候给场景放了个新ID,前台加载解析的时候也要做处理
|
||||||
|
@ -263,11 +305,13 @@ export default {
|
||||||
label: nodeArray[i],
|
label: nodeArray[i],
|
||||||
value: item,
|
value: item,
|
||||||
};
|
};
|
||||||
if (i !== (nodeArray.length - 1)) {
|
if (i !== nodeArray.length - 1) {
|
||||||
node.children = [];
|
node.children = [];
|
||||||
} else {
|
} else {
|
||||||
if (item.subRequestResults && item.subRequestResults.length > 0) {
|
if (item.subRequestResults && item.subRequestResults.length > 0) {
|
||||||
let itemChildren = this.deepFormatTreeNode(item.subRequestResults);
|
let itemChildren = this.deepFormatTreeNode(
|
||||||
|
item.subRequestResults
|
||||||
|
);
|
||||||
node.children = itemChildren;
|
node.children = itemChildren;
|
||||||
if (node.label.indexOf("UUID=")) {
|
if (node.label.indexOf("UUID=")) {
|
||||||
node.label = node.label.split("UUID=")[0];
|
node.label = node.label.split("UUID=")[0];
|
||||||
|
@ -281,7 +325,6 @@ export default {
|
||||||
let isExist = false;
|
let isExist = false;
|
||||||
for (let j in children) {
|
for (let j in children) {
|
||||||
if (children[j].label === node.label) {
|
if (children[j].label === node.label) {
|
||||||
|
|
||||||
let idIsPath = true;
|
let idIsPath = true;
|
||||||
//判断ID是否匹配 目前发现问题的只有重复场景,而重复场景是在第二个节点开始合并的。所以这里暂时只判断第二个场景问题。
|
//判断ID是否匹配 目前发现问题的只有重复场景,而重复场景是在第二个节点开始合并的。所以这里暂时只判断第二个场景问题。
|
||||||
//如果出现了其他问题,则需要检查其他问题的数据结构。暂时采用具体问题具体分析的策略
|
//如果出现了其他问题,则需要检查其他问题的数据结构。暂时采用具体问题具体分析的策略
|
||||||
|
@ -312,7 +355,8 @@ export default {
|
||||||
if (i !== nodeArray.length - 1 && !children[j].children) {
|
if (i !== nodeArray.length - 1 && !children[j].children) {
|
||||||
children[j].children = [];
|
children[j].children = [];
|
||||||
}
|
}
|
||||||
children = (i === nodeArray.length - 1 ? children : children[j].children);
|
children =
|
||||||
|
i === nodeArray.length - 1 ? children : children[j].children;
|
||||||
isExist = true;
|
isExist = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -320,21 +364,27 @@ export default {
|
||||||
}
|
}
|
||||||
if (!isExist) {
|
if (!isExist) {
|
||||||
children.push(node);
|
children.push(node);
|
||||||
if (i !== nodeArray.length - 1 && !children[children.length - 1].children) {
|
if (
|
||||||
|
i !== nodeArray.length - 1 &&
|
||||||
|
!children[children.length - 1].children
|
||||||
|
) {
|
||||||
children[children.length - 1].children = [];
|
children[children.length - 1].children = [];
|
||||||
}
|
}
|
||||||
children = (i === nodeArray.length - 1 ? children : children[children.length - 1].children);
|
children =
|
||||||
|
i === nodeArray.length - 1
|
||||||
|
? children
|
||||||
|
: children[children.length - 1].children;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
deepFormatTreeNode(array) {
|
deepFormatTreeNode(array) {
|
||||||
let returnChildren = [];
|
let returnChildren = [];
|
||||||
array.map((item) => {
|
array.map((item) => {
|
||||||
let children = [];
|
let children = [];
|
||||||
let key = item.name.split('^@~@^')[0];
|
let key = item.name.split("^@~@^")[0];
|
||||||
let nodeArray = key.split('<->');
|
let nodeArray = key.split("<->");
|
||||||
//运行场景中如果连续将1个场景引入多次,会出现运行结果合并的情况。
|
//运行场景中如果连续将1个场景引入多次,会出现运行结果合并的情况。
|
||||||
//为了解决这种问题,在转hashTree的时候给场景放了个新ID,前台加载解析的时候也要做处理
|
//为了解决这种问题,在转hashTree的时候给场景放了个新ID,前台加载解析的时候也要做处理
|
||||||
let scenarioId = "";
|
let scenarioId = "";
|
||||||
|
@ -351,17 +401,16 @@ export default {
|
||||||
let node = {
|
let node = {
|
||||||
label: nodeArray[0],
|
label: nodeArray[0],
|
||||||
value: item,
|
value: item,
|
||||||
children: []
|
children: [],
|
||||||
};
|
};
|
||||||
if (item.subRequestResults && item.subRequestResults.length > 0) {
|
if (item.subRequestResults && item.subRequestResults.length > 0) {
|
||||||
let itemChildren = this.deepFormatTreeNode(item.subRequestResults);
|
let itemChildren = this.deepFormatTreeNode(item.subRequestResults);
|
||||||
node.children = itemChildren;
|
node.children = itemChildren;
|
||||||
}
|
}
|
||||||
children.push(node);
|
children.push(node);
|
||||||
children.forEach(itemNode => {
|
children.forEach((itemNode) => {
|
||||||
returnChildren.push(itemNode);
|
returnChildren.push(itemNode);
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
return returnChildren;
|
return returnChildren;
|
||||||
},
|
},
|
||||||
|
@ -380,7 +429,10 @@ export default {
|
||||||
// 排序
|
// 排序
|
||||||
if (scenarioDefinition[i]) {
|
if (scenarioDefinition[i]) {
|
||||||
scenarioDefinition[i].index = Number(i) + 1;
|
scenarioDefinition[i].index = Number(i) + 1;
|
||||||
if (scenarioDefinition[i].children && scenarioDefinition[i].children.length > 0) {
|
if (
|
||||||
|
scenarioDefinition[i].children &&
|
||||||
|
scenarioDefinition[i].children.length > 0
|
||||||
|
) {
|
||||||
this.recursiveSorting(scenarioDefinition[i].children);
|
this.recursiveSorting(scenarioDefinition[i].children);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -390,7 +442,7 @@ export default {
|
||||||
if (this.exportReportIsOk) {
|
if (this.exportReportIsOk) {
|
||||||
this.startExport();
|
this.startExport();
|
||||||
} else {
|
} else {
|
||||||
getScenarioReportAll(this.reportId).then(data => {
|
getScenarioReportAll(this.reportId).then((data) => {
|
||||||
if (data && data.data.content) {
|
if (data && data.data.content) {
|
||||||
let report = JSON.parse(data.data.content);
|
let report = JSON.parse(data.data.content);
|
||||||
if (report.projectEnvMap) {
|
if (report.projectEnvMap) {
|
||||||
|
@ -400,7 +452,8 @@ export default {
|
||||||
this.fullTreeNodes = report.steps;
|
this.fullTreeNodes = report.steps;
|
||||||
this.content.console = report.console;
|
this.content.console = report.console;
|
||||||
this.content.error = report.error;
|
this.content.error = report.error;
|
||||||
let successCount = (report.total - report.error - report.errorCode - report.unExecute);
|
let successCount =
|
||||||
|
report.total - report.error - report.errorCode - report.unExecute;
|
||||||
this.content.success = successCount;
|
this.content.success = successCount;
|
||||||
if (this.report.endTime > 0) {
|
if (this.report.endTime > 0) {
|
||||||
this.totalTime = this.report.endTime - this.report.createTime;
|
this.totalTime = this.report.endTime - this.report.createTime;
|
||||||
|
@ -409,7 +462,7 @@ export default {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.exportReportIsOk = true;
|
this.exportReportIsOk = true;
|
||||||
setTimeout(this.startExport, 500)
|
setTimeout(this.startExport, 500);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -426,13 +479,13 @@ export default {
|
||||||
}
|
}
|
||||||
} else if (this.isShare) {
|
} else if (this.isShare) {
|
||||||
if (this.reportId) {
|
if (this.reportId) {
|
||||||
getShareScenarioReport(this.shareId, this.reportId).then(data => {
|
getShareScenarioReport(this.shareId, this.reportId).then((data) => {
|
||||||
this.checkReport(data.data);
|
this.checkReport(data.data);
|
||||||
this.handleGetScenarioReport(data.data);
|
this.handleGetScenarioReport(data.data);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
getScenarioReport(this.reportId).then(data => {
|
getScenarioReport(this.reportId).then((data) => {
|
||||||
this.checkReport(data.data);
|
this.checkReport(data.data);
|
||||||
this.handleGetScenarioReport(data.data);
|
this.handleGetScenarioReport(data.data);
|
||||||
});
|
});
|
||||||
|
@ -441,7 +494,7 @@ export default {
|
||||||
|
|
||||||
checkReport(data) {
|
checkReport(data) {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
this.$emit('reportNotExist');
|
this.$emit("reportNotExist");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
handleGetScenarioReport(data) {
|
handleGetScenarioReport(data) {
|
||||||
|
@ -450,7 +503,7 @@ export default {
|
||||||
if (this.report.reportVersion && this.report.reportVersion > 1) {
|
if (this.report.reportVersion && this.report.reportVersion > 1) {
|
||||||
this.report.status = data.status;
|
this.report.status = data.status;
|
||||||
if (!this.isNotRunning) {
|
if (!this.isNotRunning) {
|
||||||
setTimeout(this.getReport, 2000)
|
setTimeout(this.getReport, 2000);
|
||||||
} else {
|
} else {
|
||||||
if (data.content) {
|
if (data.content) {
|
||||||
let report = JSON.parse(data.content);
|
let report = JSON.parse(data.content);
|
||||||
|
@ -472,7 +525,11 @@ export default {
|
||||||
}
|
}
|
||||||
this.content.console = report.console;
|
this.content.console = report.console;
|
||||||
this.content.error = report.error;
|
this.content.error = report.error;
|
||||||
let successCount = (report.total - report.error - report.errorCode - report.unExecute);
|
let successCount =
|
||||||
|
report.total -
|
||||||
|
report.error -
|
||||||
|
report.errorCode -
|
||||||
|
report.unExecute;
|
||||||
this.content.success = successCount;
|
this.content.success = successCount;
|
||||||
if (this.report.endTime > 0) {
|
if (this.report.endTime > 0) {
|
||||||
this.totalTime = this.report.endTime - this.report.createTime;
|
this.totalTime = this.report.endTime - this.report.createTime;
|
||||||
|
@ -481,7 +538,11 @@ export default {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 增加失败重跑校验
|
// 增加失败重跑校验
|
||||||
if (this.report && this.report.reportType === 'SCENARIO_INTEGRATED' || this.report.reportType === 'API_INTEGRATED') {
|
if (
|
||||||
|
(this.report &&
|
||||||
|
this.report.reportType === "SCENARIO_INTEGRATED") ||
|
||||||
|
this.report.reportType === "API_INTEGRATED"
|
||||||
|
) {
|
||||||
this.rerunVerify();
|
this.rerunVerify();
|
||||||
}
|
}
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
|
@ -490,8 +551,8 @@ export default {
|
||||||
this.buildReport();
|
this.buildReport();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.$emit('invisible');
|
this.$emit("invisible");
|
||||||
this.$warning(this.$t('commons.report_delete'));
|
this.$warning(this.$t("commons.report_delete"));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
checkOrder(origin) {
|
checkOrder(origin) {
|
||||||
|
@ -500,39 +561,45 @@ export default {
|
||||||
}
|
}
|
||||||
if (Array.isArray(origin)) {
|
if (Array.isArray(origin)) {
|
||||||
this.sortChildren(origin);
|
this.sortChildren(origin);
|
||||||
origin.forEach(v => {
|
origin.forEach((v) => {
|
||||||
if (v.children) {
|
if (v.children) {
|
||||||
this.checkOrder(v.children)
|
this.checkOrder(v.children);
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
sortChildren(source) {
|
sortChildren(source) {
|
||||||
if (!source) {
|
if (!source) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
source.forEach(item => {
|
source.forEach((item) => {
|
||||||
let children = item.children;
|
let children = item.children;
|
||||||
if (children && children.length > 0) {
|
if (children && children.length > 0) {
|
||||||
let tempArr = new Array(children.length);
|
let tempArr = new Array(children.length);
|
||||||
let tempMap = new Map();
|
let tempMap = new Map();
|
||||||
|
|
||||||
for (let i = 0; i < children.length; i++) {
|
for (let i = 0; i < children.length; i++) {
|
||||||
if (!children[i].value || !children[i].value.startTime || children[i].value.startTime === 0) {
|
if (
|
||||||
|
!children[i].value ||
|
||||||
|
!children[i].value.startTime ||
|
||||||
|
children[i].value.startTime === 0
|
||||||
|
) {
|
||||||
//若没有value或未执行的,则step留在当前位置
|
//若没有value或未执行的,则step留在当前位置
|
||||||
tempArr[i] = children[i];
|
tempArr[i] = children[i];
|
||||||
//进行标识
|
//进行标识
|
||||||
tempMap.set(children[i].stepId, children[i])
|
tempMap.set(children[i].stepId, children[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//过滤出还没有指定好位置的step
|
//过滤出还没有指定好位置的step
|
||||||
let arr = children.filter(m => {
|
let arr = children
|
||||||
return !tempMap.get(m.stepId);
|
.filter((m) => {
|
||||||
}).sort((m, n) => {
|
return !tempMap.get(m.stepId);
|
||||||
//按时间排序
|
})
|
||||||
return m.value.startTime - n.value.startTime;
|
.sort((m, n) => {
|
||||||
});
|
//按时间排序
|
||||||
|
return m.value.startTime - n.value.startTime;
|
||||||
|
});
|
||||||
|
|
||||||
//找出arr(已经有序,从头取即可)中时间最小的插入 tempArr 可用位置
|
//找出arr(已经有序,从头取即可)中时间最小的插入 tempArr 可用位置
|
||||||
for (let j = 0, i = 0; j < tempArr.length; j++) {
|
for (let j = 0, i = 0; j < tempArr.length; j++) {
|
||||||
|
@ -548,36 +615,35 @@ export default {
|
||||||
//赋值
|
//赋值
|
||||||
item.children = tempArr;
|
item.children = tempArr;
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
},
|
},
|
||||||
buildReport() {
|
buildReport() {
|
||||||
if (this.report) {
|
if (this.report) {
|
||||||
if (this.isNotRunning) {
|
if (this.isNotRunning) {
|
||||||
|
|
||||||
this.content = JSON.parse(this.report.content);
|
this.content = JSON.parse(this.report.content);
|
||||||
if (!this.content) {
|
if (!this.content) {
|
||||||
this.content = {scenarios: []};
|
this.content = { scenarios: [] };
|
||||||
}
|
}
|
||||||
this.formatResult(this.content);
|
this.formatResult(this.content);
|
||||||
this.getFails();
|
this.getFails();
|
||||||
this.computeTotalTime();
|
this.computeTotalTime();
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
} else {
|
} else {
|
||||||
setTimeout(this.getReport, 2000)
|
setTimeout(this.getReport, 2000);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
this.$error(this.$t('api_report.not_exist'));
|
this.$error(this.$t("api_report.not_exist"));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
getFails() {
|
getFails() {
|
||||||
if (this.isNotRunning) {
|
if (this.isNotRunning) {
|
||||||
this.fails = [];
|
this.fails = [];
|
||||||
let array = [];
|
let array = [];
|
||||||
this.totalTime = 0
|
this.totalTime = 0;
|
||||||
if (this.content.scenarios) {
|
if (this.content.scenarios) {
|
||||||
this.content.scenarios.forEach((scenario) => {
|
this.content.scenarios.forEach((scenario) => {
|
||||||
this.totalTime = this.totalTime + Number(scenario.responseTime)
|
this.totalTime = this.totalTime + Number(scenario.responseTime);
|
||||||
let failScenario = Object.assign({}, scenario);
|
let failScenario = Object.assign({}, scenario);
|
||||||
if (scenario.error > 0) {
|
if (scenario.error > 0) {
|
||||||
this.fails.push(failScenario);
|
this.fails.push(failScenario);
|
||||||
|
@ -588,9 +654,9 @@ export default {
|
||||||
failScenario.requestResults.push(failRequest);
|
failScenario.requestResults.push(failRequest);
|
||||||
array.push(request);
|
array.push(request);
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
this.formatTree(array, this.failsTreeNodes);
|
this.formatTree(array, this.failsTreeNodes);
|
||||||
this.sort(this.failsTreeNodes);
|
this.sort(this.failsTreeNodes);
|
||||||
|
@ -611,14 +677,14 @@ export default {
|
||||||
}
|
}
|
||||||
let resTime;
|
let resTime;
|
||||||
if (startTime === 0 || endTime === 0) {
|
if (startTime === 0 || endTime === 0) {
|
||||||
resTime = 0
|
resTime = 0;
|
||||||
} else {
|
} else {
|
||||||
resTime = endTime - startTime
|
resTime = endTime - startTime;
|
||||||
}
|
}
|
||||||
requestTime = requestTime + resTime;
|
requestTime = requestTime + resTime;
|
||||||
})
|
});
|
||||||
})
|
});
|
||||||
this.totalTime = requestTime
|
this.totalTime = requestTime;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
requestResult(requestResult) {
|
requestResult(requestResult) {
|
||||||
|
@ -631,8 +697,11 @@ export default {
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
formatExportApi(array, scenario) {
|
formatExportApi(array, scenario) {
|
||||||
array.forEach(item => {
|
array.forEach((item) => {
|
||||||
if (this.stepFilter && this.stepFilter.get("AllSamplerProxy").indexOf(item.type) !== -1) {
|
if (
|
||||||
|
this.stepFilter &&
|
||||||
|
this.stepFilter.get("AllSamplerProxy").indexOf(item.type) !== -1
|
||||||
|
) {
|
||||||
if (item.errorCode) {
|
if (item.errorCode) {
|
||||||
item.value.errorCode = item.errorCode;
|
item.value.errorCode = item.errorCode;
|
||||||
}
|
}
|
||||||
|
@ -641,72 +710,82 @@ export default {
|
||||||
if (item.children && item.children.length > 0) {
|
if (item.children && item.children.length > 0) {
|
||||||
this.formatExportApi(item.children, scenario);
|
this.formatExportApi(item.children, scenario);
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
},
|
},
|
||||||
handleExport() {
|
handleExport() {
|
||||||
this.getReportByExport();
|
this.getReportByExport();
|
||||||
},
|
},
|
||||||
startExport() {
|
startExport() {
|
||||||
if (this.report.reportVersion && this.report.reportVersion > 1) {
|
if (this.report.reportVersion && this.report.reportVersion > 1) {
|
||||||
if (this.report.reportType === 'API_INTEGRATED' || this.report.reportType === 'UI_INTEGRATED') {
|
if (
|
||||||
let scenario = {name: "", requestResults: []};
|
this.report.reportType === "API_INTEGRATED" ||
|
||||||
|
this.report.reportType === "UI_INTEGRATED"
|
||||||
|
) {
|
||||||
|
let scenario = { name: "", requestResults: [] };
|
||||||
this.content.scenarios = [scenario];
|
this.content.scenarios = [scenario];
|
||||||
this.formatExportApi(this.fullTreeNodes, scenario);
|
this.formatExportApi(this.fullTreeNodes, scenario);
|
||||||
} else {
|
} else {
|
||||||
if (this.fullTreeNodes) {
|
if (this.fullTreeNodes) {
|
||||||
this.fullTreeNodes.forEach(item => {
|
this.fullTreeNodes.forEach((item) => {
|
||||||
if (item.type === "scenario" || item.type === "UiScenario") {
|
if (item.type === "scenario" || item.type === "UiScenario") {
|
||||||
let scenario = {name: item.label, requestResults: []};
|
let scenario = { name: item.label, requestResults: [] };
|
||||||
if (this.content.scenarios && this.content.scenarios.length > 0) {
|
if (
|
||||||
|
this.content.scenarios &&
|
||||||
|
this.content.scenarios.length > 0
|
||||||
|
) {
|
||||||
this.content.scenarios.push(scenario);
|
this.content.scenarios.push(scenario);
|
||||||
} else {
|
} else {
|
||||||
this.content.scenarios = [scenario];
|
this.content.scenarios = [scenario];
|
||||||
}
|
}
|
||||||
this.formatExportApi(item.children, scenario);
|
this.formatExportApi(item.children, scenario);
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.reportExportVisible = true;
|
this.reportExportVisible = true;
|
||||||
let reset = this.exportReportReset;
|
let reset = this.exportReportReset;
|
||||||
this.$nextTick(() => {
|
this.$nextTick(() => {
|
||||||
windowPrint('apiTestReport', 0.57);
|
windowPrint("apiTestReport", 0.57);
|
||||||
reset();
|
reset();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
handleSave() {
|
handleSave() {
|
||||||
if (!this.report.name) {
|
if (!this.report.name) {
|
||||||
this.$warning(this.$t('api_test.automation.report_name_info'));
|
this.$warning(this.$t("api_test.automation.report_name_info"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.loading = true;
|
this.loading = true;
|
||||||
let url = "/ui/scenario/report/reName";
|
let url = "/ui/scenario/report/reName";
|
||||||
this.result = this.$post(url, {
|
this.result = this.$post(
|
||||||
id: this.report.id,
|
url,
|
||||||
name: this.report.name,
|
{
|
||||||
reportType: this.report.reportType
|
id: this.report.id,
|
||||||
}, response => {
|
name: this.report.name,
|
||||||
this.$success(this.$t('commons.save_success'));
|
reportType: this.report.reportType,
|
||||||
this.loading = false;
|
},
|
||||||
this.$emit('refresh');
|
(response) => {
|
||||||
}, error => {
|
this.$success(this.$t("commons.save_success"));
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
});
|
this.$emit("refresh");
|
||||||
|
},
|
||||||
|
(error) => {
|
||||||
|
this.loading = false;
|
||||||
|
}
|
||||||
|
);
|
||||||
},
|
},
|
||||||
exportReportReset() {
|
exportReportReset() {
|
||||||
this.$router.go(0);
|
this.$router.go(0);
|
||||||
},
|
},
|
||||||
handleProjectChange() {
|
handleProjectChange() {
|
||||||
this.$router.push('/api/automation/report');
|
this.$router.push("/api/automation/report");
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
created() {
|
created() {
|
||||||
this.getReport();
|
this.getReport();
|
||||||
},
|
},
|
||||||
destroyed() {
|
destroyed() {},
|
||||||
},
|
|
||||||
computed: {
|
computed: {
|
||||||
path() {
|
path() {
|
||||||
return "/api/test/edit?id=" + this.report.testId;
|
return "/api/test/edit?id=" + this.report.testId;
|
||||||
|
@ -717,8 +796,8 @@ export default {
|
||||||
projectId() {
|
projectId() {
|
||||||
return getCurrentProjectID();
|
return getCurrentProjectID();
|
||||||
},
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
<style>
|
<style>
|
||||||
.report-container .el-tabs__header {
|
.report-container .el-tabs__header {
|
||||||
|
@ -727,7 +806,6 @@ export default {
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|
||||||
.report-container {
|
.report-container {
|
||||||
height: calc(100vh - 155px);
|
height: calc(100vh - 155px);
|
||||||
min-height: 600px;
|
min-height: 600px;
|
||||||
|
@ -748,7 +826,7 @@ export default {
|
||||||
}
|
}
|
||||||
|
|
||||||
.report-container .fail {
|
.report-container .fail {
|
||||||
color: #F56C6C;
|
color: #f56c6c;
|
||||||
}
|
}
|
||||||
|
|
||||||
.report-container .is-active .fail {
|
.report-container .is-active .fail {
|
||||||
|
@ -768,7 +846,7 @@ export default {
|
||||||
transform: rotate(90deg);
|
transform: rotate(90deg);
|
||||||
}
|
}
|
||||||
|
|
||||||
.report-body{
|
.report-body {
|
||||||
min-width: 750px!important;
|
min-width: 750px !important;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
Loading…
Reference in New Issue