refactor(性能测试): 切割csv文件
This commit is contained in:
parent
a752897ab4
commit
ca472e8378
|
@ -12,6 +12,7 @@ public class EngineContext {
|
|||
private String resourcePoolId;
|
||||
private String reportId;
|
||||
private Integer resourceIndex;
|
||||
private double[] ratios;
|
||||
private Map<String, Object> properties = new HashMap<>();
|
||||
private Map<String, byte[]> testResourceFiles = new HashMap<>();
|
||||
|
||||
|
@ -92,6 +93,13 @@ public class EngineContext {
|
|||
this.resourceIndex = resourceIndex;
|
||||
}
|
||||
|
||||
public double[] getRatios() {
|
||||
return ratios;
|
||||
}
|
||||
|
||||
public void setRatios(double[] ratios) {
|
||||
this.ratios = ratios;
|
||||
}
|
||||
|
||||
public Map<String, byte[]> getTestResourceFiles() {
|
||||
return testResourceFiles;
|
||||
|
|
|
@ -106,6 +106,7 @@ public class EngineFactory {
|
|||
engineContext.setResourcePoolId(loadTest.getTestResourcePoolId());
|
||||
engineContext.setReportId(reportId);
|
||||
engineContext.setResourceIndex(resourceIndex);
|
||||
engineContext.setRatios(ratios);
|
||||
|
||||
if (StringUtils.isNotEmpty(loadTest.getLoadConfiguration())) {
|
||||
final JSONArray jsonArray = JSONObject.parseArray(loadTest.getLoadConfiguration());
|
||||
|
@ -155,6 +156,15 @@ public class EngineFactory {
|
|||
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)) {
|
||||
String content = engineSourceParser.parse(engineContext, source);
|
||||
engineContext.setContent(content);
|
||||
|
@ -166,15 +176,6 @@ public class EngineFactory {
|
|||
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;
|
||||
}
|
||||
|
||||
|
|
|
@ -22,8 +22,10 @@ import javax.xml.transform.TransformerFactory;
|
|||
import javax.xml.transform.dom.DOMSource;
|
||||
import javax.xml.transform.stream.StreamResult;
|
||||
import java.io.StringWriter;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.HashSet;
|
||||
import java.util.List;
|
||||
import java.util.StringTokenizer;
|
||||
|
||||
public class JmeterDocumentParser implements DocumentParser {
|
||||
private final static String HASH_TREE_ELEMENT = "hashTree";
|
||||
|
@ -267,12 +269,76 @@ public class JmeterDocumentParser implements DocumentParser {
|
|||
if (StringUtils.equals(filenameTag, "filename")) {
|
||||
// 截取文件名
|
||||
handleFilename(item);
|
||||
// 切割CSV文件
|
||||
splitCsvFile(item);
|
||||
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) {
|
||||
NodeList childNodes = element.getChildNodes();
|
||||
for (int i = 0; i < childNodes.getLength(); i++) {
|
||||
|
|
|
@ -368,6 +368,7 @@ export default {
|
|||
fileChange(threadGroups) {
|
||||
let handler = this.$refs.pressureConfig;
|
||||
|
||||
let csvSet = new Set;
|
||||
threadGroups.forEach(tg => {
|
||||
tg.threadNumber = tg.threadNumber || 10;
|
||||
tg.duration = tg.duration || 10;
|
||||
|
@ -377,12 +378,21 @@ export default {
|
|||
tg.threadType = tg.threadType || 'DURATION';
|
||||
tg.iterateNum = tg.iterateNum || 1;
|
||||
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.$refs.basicConfig.threadGroups = threadGroups;
|
||||
this.$refs.pressureConfig.threadGroups = threadGroups;
|
||||
this.$refs.advancedConfig.csvFiles = csvFiles;
|
||||
|
||||
handler.calculateTotalChart();
|
||||
},
|
||||
|
|
|
@ -1,5 +1,110 @@
|
|||
<template>
|
||||
<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-col :span="8">
|
||||
<h3>{{ $t('load_test.params') }}</h3>
|
||||
|
@ -74,82 +179,6 @@
|
|||
</el-col>
|
||||
</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-col :span="8">
|
||||
<h3>监控集成</h3>
|
||||
|
@ -158,6 +187,8 @@
|
|||
</el-button>
|
||||
</el-col>
|
||||
</el-row>
|
||||
<el-row>
|
||||
|
||||
<el-col :span="24">
|
||||
<el-table :data="monitorParams" size="mini" class="tb-edit" align="center" border highlight-current-row>
|
||||
<el-table-column
|
||||
|
@ -206,6 +237,7 @@
|
|||
</el-table-column>
|
||||
</el-table>
|
||||
</el-col>
|
||||
</el-row>
|
||||
|
||||
<edit-monitor ref="monitorDialog" :testId="testId" :list.sync="monitorParams"/>
|
||||
</div>
|
||||
|
@ -227,6 +259,8 @@ export default {
|
|||
domains: [],
|
||||
params: [],
|
||||
monitorParams: [],
|
||||
csvFiles: [],
|
||||
csvConfig: [],
|
||||
statusCodeStr: '',
|
||||
granularity: undefined,
|
||||
granularityData: [
|
||||
|
@ -257,6 +291,14 @@ export default {
|
|||
if (this.testId) {
|
||||
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: {
|
||||
|
@ -272,6 +314,7 @@ export default {
|
|||
this.params = data.params || [];
|
||||
this.granularity = data.granularity;
|
||||
this.monitorParams = data.monitorParams || [];
|
||||
this.csvConfig = data.csvConfig;
|
||||
}
|
||||
});
|
||||
},
|
||||
|
@ -357,6 +400,10 @@ export default {
|
|||
responseTimeout: this.responseTimeout,
|
||||
statusCode: statusCode,
|
||||
params: this.params,
|
||||
csvConfig: this.csvFiles.reduce((result, curr) => {
|
||||
result[curr.name] = {csvHasHeader: curr.csvHasHeader, csvSplit: curr.csvSplit};
|
||||
return result;
|
||||
}, {}),
|
||||
domains: this.domains,
|
||||
granularity: this.granularity,
|
||||
monitorParams: this.monitorParams
|
||||
|
|
|
@ -648,6 +648,8 @@ export default {
|
|||
load_api_automation_jmx: 'Import API automation scenario',
|
||||
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',
|
||||
csv_has_header: 'Contains Title',
|
||||
csv_split: 'CSV Split',
|
||||
report: {
|
||||
diff: "Compare"
|
||||
},
|
||||
|
|
|
@ -616,9 +616,9 @@ export default {
|
|||
param_is_duplicate: '参数名不能重复',
|
||||
domain_ip_is_empty: '域名和IP不能为空',
|
||||
param_name_value_is_empty: '参数名和参数值不能为空',
|
||||
connect_timeout: '建立连接超时时间',
|
||||
response_timeout: '响应超时时间',
|
||||
custom_http_code: '自定义 HTTP 响应成功状态码',
|
||||
connect_timeout: '连接超时',
|
||||
response_timeout: '响应超时',
|
||||
custom_http_code: '自定义响应码',
|
||||
separated_by_commas: '按逗号分隔',
|
||||
create: '创建测试',
|
||||
run: '一键运行',
|
||||
|
@ -646,6 +646,8 @@ export default {
|
|||
threadgroup_at_least_one: '至少启用一个线程组',
|
||||
load_api_automation_jmx: '引用接口自动化场景',
|
||||
project_file_exist: "项目中已存在该文件,请直接引用",
|
||||
csv_has_header: '包含表头',
|
||||
csv_split: 'CSV分割',
|
||||
report: {
|
||||
diff: "对比"
|
||||
},
|
||||
|
|
|
@ -616,9 +616,9 @@ export default {
|
|||
param_is_duplicate: '參數名不能重復',
|
||||
domain_ip_is_empty: '域名和IP不能為空',
|
||||
param_name_value_is_empty: '參數名和參數值不能為空',
|
||||
connect_timeout: '建立連接超時時間',
|
||||
response_timeout: '響應超時時間',
|
||||
custom_http_code: '自定義 HTTP 響應成功狀態碼',
|
||||
connect_timeout: '連接超時',
|
||||
response_timeout: '響應超時',
|
||||
custom_http_code: '自定義響應碼',
|
||||
separated_by_commas: '按逗號分隔',
|
||||
create: '創建測試',
|
||||
run: '一鍵運行',
|
||||
|
@ -646,6 +646,8 @@ export default {
|
|||
threadgroup_at_least_one: '至少啟用一個線程組',
|
||||
load_api_automation_jmx: '引用接口自動化場景',
|
||||
project_file_exist: "項目中已存在該文件,請直接引用",
|
||||
csv_has_header: '包含表头',
|
||||
csv_split: 'CSV分割',
|
||||
report: {
|
||||
diff: "對比"
|
||||
},
|
||||
|
|
Loading…
Reference in New Issue