refactor(性能测试): 切割csv文件
This commit is contained in:
parent
a752897ab4
commit
ca472e8378
|
@ -12,6 +12,7 @@ public class EngineContext {
|
||||||
private String resourcePoolId;
|
private String resourcePoolId;
|
||||||
private String reportId;
|
private String reportId;
|
||||||
private Integer resourceIndex;
|
private Integer resourceIndex;
|
||||||
|
private double[] ratios;
|
||||||
private Map<String, Object> properties = new HashMap<>();
|
private Map<String, Object> properties = new HashMap<>();
|
||||||
private Map<String, byte[]> testResourceFiles = new HashMap<>();
|
private Map<String, byte[]> testResourceFiles = new HashMap<>();
|
||||||
|
|
||||||
|
@ -92,6 +93,13 @@ public class EngineContext {
|
||||||
this.resourceIndex = resourceIndex;
|
this.resourceIndex = resourceIndex;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public double[] getRatios() {
|
||||||
|
return ratios;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRatios(double[] ratios) {
|
||||||
|
this.ratios = ratios;
|
||||||
|
}
|
||||||
|
|
||||||
public Map<String, byte[]> getTestResourceFiles() {
|
public Map<String, byte[]> getTestResourceFiles() {
|
||||||
return testResourceFiles;
|
return testResourceFiles;
|
||||||
|
|
|
@ -106,6 +106,7 @@ public class EngineFactory {
|
||||||
engineContext.setResourcePoolId(loadTest.getTestResourcePoolId());
|
engineContext.setResourcePoolId(loadTest.getTestResourcePoolId());
|
||||||
engineContext.setReportId(reportId);
|
engineContext.setReportId(reportId);
|
||||||
engineContext.setResourceIndex(resourceIndex);
|
engineContext.setResourceIndex(resourceIndex);
|
||||||
|
engineContext.setRatios(ratios);
|
||||||
|
|
||||||
if (StringUtils.isNotEmpty(loadTest.getLoadConfiguration())) {
|
if (StringUtils.isNotEmpty(loadTest.getLoadConfiguration())) {
|
||||||
final JSONArray jsonArray = JSONObject.parseArray(loadTest.getLoadConfiguration());
|
final JSONArray jsonArray = JSONObject.parseArray(loadTest.getLoadConfiguration());
|
||||||
|
@ -155,6 +156,15 @@ public class EngineFactory {
|
||||||
MSException.throwException("File type unknown");
|
MSException.throwException("File type unknown");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (CollectionUtils.isNotEmpty(resourceFiles)) {
|
||||||
|
Map<String, byte[]> data = new HashMap<>();
|
||||||
|
resourceFiles.forEach(cf -> {
|
||||||
|
FileContent csvContent = fileService.getFileContent(cf.getId());
|
||||||
|
data.put(cf.getName(), csvContent.getFile());
|
||||||
|
});
|
||||||
|
engineContext.setTestResourceFiles(data);
|
||||||
|
}
|
||||||
|
|
||||||
try (ByteArrayInputStream source = new ByteArrayInputStream(jmxBytes)) {
|
try (ByteArrayInputStream source = new ByteArrayInputStream(jmxBytes)) {
|
||||||
String content = engineSourceParser.parse(engineContext, source);
|
String content = engineSourceParser.parse(engineContext, source);
|
||||||
engineContext.setContent(content);
|
engineContext.setContent(content);
|
||||||
|
@ -166,15 +176,6 @@ public class EngineFactory {
|
||||||
MSException.throwException(e);
|
MSException.throwException(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (CollectionUtils.isNotEmpty(resourceFiles)) {
|
|
||||||
Map<String, byte[]> data = new HashMap<>();
|
|
||||||
resourceFiles.forEach(cf -> {
|
|
||||||
FileContent csvContent = fileService.getFileContent(cf.getId());
|
|
||||||
data.put(cf.getName(), csvContent.getFile());
|
|
||||||
});
|
|
||||||
engineContext.setTestResourceFiles(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
return engineContext;
|
return engineContext;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -22,8 +22,10 @@ import javax.xml.transform.TransformerFactory;
|
||||||
import javax.xml.transform.dom.DOMSource;
|
import javax.xml.transform.dom.DOMSource;
|
||||||
import javax.xml.transform.stream.StreamResult;
|
import javax.xml.transform.stream.StreamResult;
|
||||||
import java.io.StringWriter;
|
import java.io.StringWriter;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.StringTokenizer;
|
||||||
|
|
||||||
public class JmeterDocumentParser implements DocumentParser {
|
public class JmeterDocumentParser implements DocumentParser {
|
||||||
private final static String HASH_TREE_ELEMENT = "hashTree";
|
private final static String HASH_TREE_ELEMENT = "hashTree";
|
||||||
|
@ -267,12 +269,76 @@ public class JmeterDocumentParser implements DocumentParser {
|
||||||
if (StringUtils.equals(filenameTag, "filename")) {
|
if (StringUtils.equals(filenameTag, "filename")) {
|
||||||
// 截取文件名
|
// 截取文件名
|
||||||
handleFilename(item);
|
handleFilename(item);
|
||||||
|
// 切割CSV文件
|
||||||
|
splitCsvFile(item);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void splitCsvFile(Node item) {
|
||||||
|
Object csvConfig = context.getProperty("csvConfig");
|
||||||
|
if (csvConfig == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
double[] ratios = context.getRatios();
|
||||||
|
int resourceIndex = context.getResourceIndex();
|
||||||
|
String filename = item.getTextContent();
|
||||||
|
byte[] content = context.getTestResourceFiles().get(filename);
|
||||||
|
StringTokenizer tokenizer = new StringTokenizer(new String(content), "\n");
|
||||||
|
if (!tokenizer.hasMoreTokens()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
StringBuilder csv = new StringBuilder();
|
||||||
|
Object config = ((JSONObject) csvConfig).get(filename);
|
||||||
|
boolean csvSplit = ((JSONObject) (config)).getBooleanValue("csvSplit");
|
||||||
|
if (!csvSplit) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
boolean csvHasHeader = ((JSONObject) (config)).getBooleanValue("csvHasHeader");
|
||||||
|
if (csvHasHeader) {
|
||||||
|
String header = tokenizer.nextToken();
|
||||||
|
csv.append(header).append("\n");
|
||||||
|
}
|
||||||
|
int count = tokenizer.countTokens();
|
||||||
|
|
||||||
|
long current, offset = 0;
|
||||||
|
|
||||||
|
// 计算偏移量
|
||||||
|
for (int k = 0; k < resourceIndex; k++) {
|
||||||
|
offset += Math.round(count * ratios[k]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resourceIndex + 1 == ratios.length) {
|
||||||
|
current = count - offset; // 最后一个点可以分到的数量
|
||||||
|
} else {
|
||||||
|
current = Math.round(count * ratios[resourceIndex]); // 当前节点可以分到的数量
|
||||||
|
}
|
||||||
|
|
||||||
|
long index = 1;
|
||||||
|
while (tokenizer.hasMoreTokens()) {
|
||||||
|
if (current == 0) { // 节点一个都没有分到,把所有的数据都给这个节点(极端情况)
|
||||||
|
String line = tokenizer.nextToken();
|
||||||
|
csv.append(line).append("\n");
|
||||||
|
} else {
|
||||||
|
if (index < offset) {
|
||||||
|
tokenizer.nextToken();
|
||||||
|
index++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (index > current + offset) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
String line = tokenizer.nextToken();
|
||||||
|
csv.append(line).append("\n");
|
||||||
|
}
|
||||||
|
index++;
|
||||||
|
}
|
||||||
|
// 替换文件
|
||||||
|
context.getTestResourceFiles().put(filename, csv.toString().getBytes(StandardCharsets.UTF_8));
|
||||||
|
}
|
||||||
|
|
||||||
private void processResponseAssertion(Element element) {
|
private void processResponseAssertion(Element element) {
|
||||||
NodeList childNodes = element.getChildNodes();
|
NodeList childNodes = element.getChildNodes();
|
||||||
for (int i = 0; i < childNodes.getLength(); i++) {
|
for (int i = 0; i < childNodes.getLength(); i++) {
|
||||||
|
|
|
@ -368,6 +368,7 @@ export default {
|
||||||
fileChange(threadGroups) {
|
fileChange(threadGroups) {
|
||||||
let handler = this.$refs.pressureConfig;
|
let handler = this.$refs.pressureConfig;
|
||||||
|
|
||||||
|
let csvSet = new Set;
|
||||||
threadGroups.forEach(tg => {
|
threadGroups.forEach(tg => {
|
||||||
tg.threadNumber = tg.threadNumber || 10;
|
tg.threadNumber = tg.threadNumber || 10;
|
||||||
tg.duration = tg.duration || 10;
|
tg.duration = tg.duration || 10;
|
||||||
|
@ -377,12 +378,21 @@ export default {
|
||||||
tg.threadType = tg.threadType || 'DURATION';
|
tg.threadType = tg.threadType || 'DURATION';
|
||||||
tg.iterateNum = tg.iterateNum || 1;
|
tg.iterateNum = tg.iterateNum || 1;
|
||||||
tg.iterateRampUp = tg.iterateRampUp || 10;
|
tg.iterateRampUp = tg.iterateRampUp || 10;
|
||||||
|
|
||||||
|
if (tg.csvFiles) {
|
||||||
|
tg.csvFiles.map(item => csvSet.add(item));
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
let csvFiles = [];
|
||||||
|
for (const f of csvSet) {
|
||||||
|
csvFiles.push({name: f, csvSplit: false, csvHasHeader: true});
|
||||||
|
}
|
||||||
|
|
||||||
this.$set(handler, "threadGroups", threadGroups);
|
this.$set(handler, "threadGroups", threadGroups);
|
||||||
|
|
||||||
this.$refs.basicConfig.threadGroups = threadGroups;
|
this.$refs.basicConfig.threadGroups = threadGroups;
|
||||||
this.$refs.pressureConfig.threadGroups = threadGroups;
|
this.$refs.pressureConfig.threadGroups = threadGroups;
|
||||||
|
this.$refs.advancedConfig.csvFiles = csvFiles;
|
||||||
|
|
||||||
handler.calculateTotalChart();
|
handler.calculateTotalChart();
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,5 +1,110 @@
|
||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
|
<!-- 基本配置 -->
|
||||||
|
<el-row>
|
||||||
|
<el-col :span="6">
|
||||||
|
<el-form :inline="true">
|
||||||
|
<el-form-item>
|
||||||
|
<div>{{ $t('load_test.connect_timeout') }}</div>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-input-number
|
||||||
|
:disabled="readOnly" size="mini" v-model="timeout"
|
||||||
|
:min="0"/>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
ms
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="6">
|
||||||
|
<el-form :inline="true">
|
||||||
|
<el-form-item>
|
||||||
|
<div>{{ $t('load_test.response_timeout') }}</div>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-input-number
|
||||||
|
:disabled="readOnly" size="mini" :min="0"
|
||||||
|
v-model="responseTimeout"/>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
ms
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="6">
|
||||||
|
<el-form :inline="true">
|
||||||
|
<el-form-item>
|
||||||
|
<div>
|
||||||
|
{{ $t('load_test.granularity') }}
|
||||||
|
<el-popover
|
||||||
|
placement="left"
|
||||||
|
width="300"
|
||||||
|
trigger="hover">
|
||||||
|
<el-table :data="granularityData">
|
||||||
|
<el-table-column property="start" :label="$t('load_test.duration')">
|
||||||
|
<template v-slot:default="scope">
|
||||||
|
<span>{{ scope.row.start }}S - {{ scope.row.end }}S</span>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column property="granularity" :label="$t('load_test.granularity')"/>
|
||||||
|
</el-table>
|
||||||
|
<i slot="reference" class="el-icon-info pointer"/>
|
||||||
|
</el-popover>
|
||||||
|
</div>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-select v-model="granularity" :placeholder="$t('commons.please_select')" size="mini"
|
||||||
|
clearable>
|
||||||
|
<el-option v-for="op in granularityData" :key="op.granularity" :label="op.granularity"
|
||||||
|
:value="op.granularity"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="6">
|
||||||
|
<el-form :inline="true">
|
||||||
|
<el-form-item>
|
||||||
|
<div>{{ $t('load_test.custom_http_code') }}</div>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-input
|
||||||
|
:disabled="readOnly" size="mini" v-model="statusCodeStr"
|
||||||
|
:placeholder="$t('load_test.separated_by_commas')"
|
||||||
|
@input="checkStatusCode"></el-input>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<!-- csv 配置 -->
|
||||||
|
<el-row>
|
||||||
|
<el-col :span="8">
|
||||||
|
<h3>CSVDataSet</h3>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row>
|
||||||
|
<el-col :span="24">
|
||||||
|
<el-table :data="csvFiles" size="mini" class="tb-edit" align="center" border highlight-current-row>
|
||||||
|
<el-table-column
|
||||||
|
align="center"
|
||||||
|
prop="name"
|
||||||
|
:label="$t('commons.name')">
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column align="center" prop="csvSplit" :label="$t('load_test.csv_split')">
|
||||||
|
<template v-slot:default="{row}">
|
||||||
|
<el-switch :disabled="readOnly" v-model="row.csvSplit"/>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column align="center" prop="csvHasHeader" :label="$t('load_test.csv_has_header')">
|
||||||
|
<template v-slot:default="{row}">
|
||||||
|
<el-switch :disabled="readOnly || !row.csvSplit" v-model="row.csvHasHeader"/>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<!-- 参数列表 -->
|
||||||
<el-row>
|
<el-row>
|
||||||
<el-col :span="8">
|
<el-col :span="8">
|
||||||
<h3>{{ $t('load_test.params') }}</h3>
|
<h3>{{ $t('load_test.params') }}</h3>
|
||||||
|
@ -74,82 +179,6 @@
|
||||||
</el-col>
|
</el-col>
|
||||||
</el-row>
|
</el-row>
|
||||||
|
|
||||||
<el-row>
|
|
||||||
<el-col :span="8">
|
|
||||||
<el-form :inline="true">
|
|
||||||
<el-form-item>
|
|
||||||
<div>{{ $t('load_test.connect_timeout') }}</div>
|
|
||||||
</el-form-item>
|
|
||||||
<el-form-item>
|
|
||||||
<el-input-number :disabled="readOnly" size="mini" v-model="timeout"
|
|
||||||
:min="0"></el-input-number>
|
|
||||||
</el-form-item>
|
|
||||||
<el-form-item>
|
|
||||||
ms
|
|
||||||
</el-form-item>
|
|
||||||
</el-form>
|
|
||||||
</el-col>
|
|
||||||
<el-col :span="8">
|
|
||||||
<el-form :inline="true">
|
|
||||||
<el-form-item>
|
|
||||||
<div>{{ $t('load_test.response_timeout') }}</div>
|
|
||||||
</el-form-item>
|
|
||||||
<el-form-item>
|
|
||||||
<el-input-number :disabled="readOnly" size="mini" :min="0"
|
|
||||||
v-model="responseTimeout"></el-input-number>
|
|
||||||
</el-form-item>
|
|
||||||
<el-form-item>
|
|
||||||
ms
|
|
||||||
</el-form-item>
|
|
||||||
</el-form>
|
|
||||||
</el-col>
|
|
||||||
</el-row>
|
|
||||||
<el-row>
|
|
||||||
<el-col :span="8">
|
|
||||||
<el-form :inline="true">
|
|
||||||
<el-form-item>
|
|
||||||
<div>{{ $t('load_test.custom_http_code') }}</div>
|
|
||||||
</el-form-item>
|
|
||||||
<el-form-item>
|
|
||||||
<el-input :disabled="readOnly" size="mini" v-model="statusCodeStr"
|
|
||||||
:placeholder="$t('load_test.separated_by_commas')"
|
|
||||||
@input="checkStatusCode"></el-input>
|
|
||||||
</el-form-item>
|
|
||||||
</el-form>
|
|
||||||
</el-col>
|
|
||||||
</el-row>
|
|
||||||
<el-row>
|
|
||||||
<el-col :span="8">
|
|
||||||
<el-form :inline="true">
|
|
||||||
<el-form-item>
|
|
||||||
<div>
|
|
||||||
{{ $t('load_test.granularity') }}
|
|
||||||
<el-popover
|
|
||||||
placement="bottom"
|
|
||||||
width="400"
|
|
||||||
trigger="hover">
|
|
||||||
<el-table :data="granularityData">
|
|
||||||
<el-table-column property="start" :label="$t('load_test.duration')">
|
|
||||||
<template v-slot:default="scope">
|
|
||||||
<span>{{ scope.row.start }}S - {{ scope.row.end }}S</span>
|
|
||||||
</template>
|
|
||||||
</el-table-column>
|
|
||||||
<el-table-column property="granularity" :label="$t('load_test.granularity')"/>
|
|
||||||
</el-table>
|
|
||||||
<i slot="reference" class="el-icon-info pointer"/>
|
|
||||||
</el-popover>
|
|
||||||
</div>
|
|
||||||
</el-form-item>
|
|
||||||
<el-form-item>
|
|
||||||
<el-select v-model="granularity" :placeholder="$t('commons.please_select')" size="mini" clearable>
|
|
||||||
<el-option v-for="op in granularityData" :key="op.granularity" :label="op.granularity"
|
|
||||||
:value="op.granularity"></el-option>
|
|
||||||
</el-select>
|
|
||||||
</el-form-item>
|
|
||||||
</el-form>
|
|
||||||
</el-col>
|
|
||||||
</el-row>
|
|
||||||
|
|
||||||
<el-row>
|
<el-row>
|
||||||
<el-col :span="8">
|
<el-col :span="8">
|
||||||
<h3>监控集成</h3>
|
<h3>监控集成</h3>
|
||||||
|
@ -158,6 +187,8 @@
|
||||||
</el-button>
|
</el-button>
|
||||||
</el-col>
|
</el-col>
|
||||||
</el-row>
|
</el-row>
|
||||||
|
<el-row>
|
||||||
|
|
||||||
<el-col :span="24">
|
<el-col :span="24">
|
||||||
<el-table :data="monitorParams" size="mini" class="tb-edit" align="center" border highlight-current-row>
|
<el-table :data="monitorParams" size="mini" class="tb-edit" align="center" border highlight-current-row>
|
||||||
<el-table-column
|
<el-table-column
|
||||||
|
@ -206,6 +237,7 @@
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
</el-table>
|
</el-table>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
<edit-monitor ref="monitorDialog" :testId="testId" :list.sync="monitorParams"/>
|
<edit-monitor ref="monitorDialog" :testId="testId" :list.sync="monitorParams"/>
|
||||||
</div>
|
</div>
|
||||||
|
@ -227,6 +259,8 @@ export default {
|
||||||
domains: [],
|
domains: [],
|
||||||
params: [],
|
params: [],
|
||||||
monitorParams: [],
|
monitorParams: [],
|
||||||
|
csvFiles: [],
|
||||||
|
csvConfig: [],
|
||||||
statusCodeStr: '',
|
statusCodeStr: '',
|
||||||
granularity: undefined,
|
granularity: undefined,
|
||||||
granularityData: [
|
granularityData: [
|
||||||
|
@ -257,6 +291,14 @@ export default {
|
||||||
if (this.testId) {
|
if (this.testId) {
|
||||||
this.getAdvancedConfig();
|
this.getAdvancedConfig();
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
csvFiles() {
|
||||||
|
if (this.csvConfig && this.csvFiles) {
|
||||||
|
this.csvFiles.forEach(f => {
|
||||||
|
f.csvSplit = this.csvConfig[f.name].csvSplit;
|
||||||
|
f.csvHasHeader = this.csvConfig[f.name].csvHasHeader;
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
@ -272,6 +314,7 @@ export default {
|
||||||
this.params = data.params || [];
|
this.params = data.params || [];
|
||||||
this.granularity = data.granularity;
|
this.granularity = data.granularity;
|
||||||
this.monitorParams = data.monitorParams || [];
|
this.monitorParams = data.monitorParams || [];
|
||||||
|
this.csvConfig = data.csvConfig;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
@ -357,6 +400,10 @@ export default {
|
||||||
responseTimeout: this.responseTimeout,
|
responseTimeout: this.responseTimeout,
|
||||||
statusCode: statusCode,
|
statusCode: statusCode,
|
||||||
params: this.params,
|
params: this.params,
|
||||||
|
csvConfig: this.csvFiles.reduce((result, curr) => {
|
||||||
|
result[curr.name] = {csvHasHeader: curr.csvHasHeader, csvSplit: curr.csvSplit};
|
||||||
|
return result;
|
||||||
|
}, {}),
|
||||||
domains: this.domains,
|
domains: this.domains,
|
||||||
granularity: this.granularity,
|
granularity: this.granularity,
|
||||||
monitorParams: this.monitorParams
|
monitorParams: this.monitorParams
|
||||||
|
|
|
@ -648,6 +648,8 @@ export default {
|
||||||
load_api_automation_jmx: 'Import API automation scenario',
|
load_api_automation_jmx: 'Import API automation scenario',
|
||||||
project_file_exist: "The file already exists in the project, please import it directly",
|
project_file_exist: "The file already exists in the project, please import it directly",
|
||||||
project_file_update_type_error: 'Updated file types must be consistent',
|
project_file_update_type_error: 'Updated file types must be consistent',
|
||||||
|
csv_has_header: 'Contains Title',
|
||||||
|
csv_split: 'CSV Split',
|
||||||
report: {
|
report: {
|
||||||
diff: "Compare"
|
diff: "Compare"
|
||||||
},
|
},
|
||||||
|
|
|
@ -616,9 +616,9 @@ export default {
|
||||||
param_is_duplicate: '参数名不能重复',
|
param_is_duplicate: '参数名不能重复',
|
||||||
domain_ip_is_empty: '域名和IP不能为空',
|
domain_ip_is_empty: '域名和IP不能为空',
|
||||||
param_name_value_is_empty: '参数名和参数值不能为空',
|
param_name_value_is_empty: '参数名和参数值不能为空',
|
||||||
connect_timeout: '建立连接超时时间',
|
connect_timeout: '连接超时',
|
||||||
response_timeout: '响应超时时间',
|
response_timeout: '响应超时',
|
||||||
custom_http_code: '自定义 HTTP 响应成功状态码',
|
custom_http_code: '自定义响应码',
|
||||||
separated_by_commas: '按逗号分隔',
|
separated_by_commas: '按逗号分隔',
|
||||||
create: '创建测试',
|
create: '创建测试',
|
||||||
run: '一键运行',
|
run: '一键运行',
|
||||||
|
@ -646,6 +646,8 @@ export default {
|
||||||
threadgroup_at_least_one: '至少启用一个线程组',
|
threadgroup_at_least_one: '至少启用一个线程组',
|
||||||
load_api_automation_jmx: '引用接口自动化场景',
|
load_api_automation_jmx: '引用接口自动化场景',
|
||||||
project_file_exist: "项目中已存在该文件,请直接引用",
|
project_file_exist: "项目中已存在该文件,请直接引用",
|
||||||
|
csv_has_header: '包含表头',
|
||||||
|
csv_split: 'CSV分割',
|
||||||
report: {
|
report: {
|
||||||
diff: "对比"
|
diff: "对比"
|
||||||
},
|
},
|
||||||
|
|
|
@ -616,9 +616,9 @@ export default {
|
||||||
param_is_duplicate: '參數名不能重復',
|
param_is_duplicate: '參數名不能重復',
|
||||||
domain_ip_is_empty: '域名和IP不能為空',
|
domain_ip_is_empty: '域名和IP不能為空',
|
||||||
param_name_value_is_empty: '參數名和參數值不能為空',
|
param_name_value_is_empty: '參數名和參數值不能為空',
|
||||||
connect_timeout: '建立連接超時時間',
|
connect_timeout: '連接超時',
|
||||||
response_timeout: '響應超時時間',
|
response_timeout: '響應超時',
|
||||||
custom_http_code: '自定義 HTTP 響應成功狀態碼',
|
custom_http_code: '自定義響應碼',
|
||||||
separated_by_commas: '按逗號分隔',
|
separated_by_commas: '按逗號分隔',
|
||||||
create: '創建測試',
|
create: '創建測試',
|
||||||
run: '一鍵運行',
|
run: '一鍵運行',
|
||||||
|
@ -646,6 +646,8 @@ export default {
|
||||||
threadgroup_at_least_one: '至少啟用一個線程組',
|
threadgroup_at_least_one: '至少啟用一個線程組',
|
||||||
load_api_automation_jmx: '引用接口自動化場景',
|
load_api_automation_jmx: '引用接口自動化場景',
|
||||||
project_file_exist: "項目中已存在該文件,請直接引用",
|
project_file_exist: "項目中已存在該文件,請直接引用",
|
||||||
|
csv_has_header: '包含表头',
|
||||||
|
csv_split: 'CSV分割',
|
||||||
report: {
|
report: {
|
||||||
diff: "對比"
|
diff: "對比"
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in New Issue