Merge remote-tracking branch 'origin/master'

# Conflicts:
#	backend/src/main/java/io/metersphere/api/service/ApiAutomationService.java
This commit is contained in:
song.tianyang 2021-03-09 19:21:19 +08:00
commit 7e7433f801
31 changed files with 539 additions and 332 deletions

View File

@ -11,6 +11,7 @@ import io.metersphere.api.service.ApiAutomationService;
import io.metersphere.base.domain.ApiScenario; import io.metersphere.base.domain.ApiScenario;
import io.metersphere.base.domain.ApiScenarioWithBLOBs; import io.metersphere.base.domain.ApiScenarioWithBLOBs;
import io.metersphere.base.domain.Schedule; import io.metersphere.base.domain.Schedule;
import io.metersphere.commons.constants.ApiRunMode;
import io.metersphere.commons.constants.RoleConstants; import io.metersphere.commons.constants.RoleConstants;
import io.metersphere.commons.utils.PageUtils; import io.metersphere.commons.utils.PageUtils;
import io.metersphere.commons.utils.Pager; import io.metersphere.commons.utils.Pager;
@ -102,12 +103,16 @@ public class ApiAutomationController {
@PostMapping(value = "/run") @PostMapping(value = "/run")
public String run(@RequestBody RunScenarioRequest request) { public String run(@RequestBody RunScenarioRequest request) {
request.setExecuteType(ExecuteType.Completed.name()); request.setExecuteType(ExecuteType.Completed.name());
request.setTriggerMode(ApiRunMode.SCENARIO.name());
request.setRunMode(ApiRunMode.SCENARIO.name());
return apiAutomationService.run(request); return apiAutomationService.run(request);
} }
@PostMapping(value = "/run/batch") @PostMapping(value = "/run/batch")
public String runBatch(@RequestBody RunScenarioRequest request) { public String runBatch(@RequestBody RunScenarioRequest request) {
request.setExecuteType(ExecuteType.Saved.name()); request.setExecuteType(ExecuteType.Saved.name());
request.setTriggerMode(ApiRunMode.SCENARIO.name());
request.setRunMode(ApiRunMode.SCENARIO.name());
return apiAutomationService.run(request); return apiAutomationService.run(request);
} }

View File

@ -85,6 +85,7 @@ public class MsScenario extends MsTestElement {
JSONObject element = JSON.parseObject(scenario.getScenarioDefinition()); JSONObject element = JSON.parseObject(scenario.getScenarioDefinition());
hashTree = mapper.readValue(element.getString("hashTree"), new TypeReference<LinkedList<MsTestElement>>() { hashTree = mapper.readValue(element.getString("hashTree"), new TypeReference<LinkedList<MsTestElement>>() {
}); });
OldVersionUtil.transferHashTree(hashTree);
// 场景变量 // 场景变量
if (StringUtils.isNotEmpty(element.getString("variables"))) { if (StringUtils.isNotEmpty(element.getString("variables"))) {
LinkedList<ScenarioVariable> variables = mapper.readValue(element.getString("variables"), LinkedList<ScenarioVariable> variables = mapper.readValue(element.getString("variables"),

View File

@ -161,6 +161,7 @@ public abstract class MsTestElement {
element = mapper.readValue(apiDefinition.getRequest(), new TypeReference<MsTestElement>() { element = mapper.readValue(apiDefinition.getRequest(), new TypeReference<MsTestElement>() {
}); });
hashTree.add(element); hashTree.add(element);
OldVersionUtil.transferHashTree(hashTree);
} }
} catch (Exception ex) { } catch (Exception ex) {
ex.printStackTrace(); ex.printStackTrace();

View File

@ -0,0 +1,23 @@
package io.metersphere.api.dto.definition.request;
import io.metersphere.commons.utils.SessionUtils;
import org.apache.commons.lang3.StringUtils;
import java.util.List;
public class OldVersionUtil {
public static void transferHashTree(List<MsTestElement> hashTree) {
for (int i = 0; i < hashTree.size(); i++) {
MsTestElement element = hashTree.get(i);
if (StringUtils.isBlank(element.getProjectId())) {
element.setProjectId(SessionUtils.getCurrentProjectId());
}
if (element.getHashTree() != null && element.getHashTree().size() > 0) {
transferHashTree(element.getHashTree());
}
}
}
}

View File

@ -96,7 +96,7 @@ public class MsAssertions extends MsTestElement {
assertion.setJsonValidationBool(true); assertion.setJsonValidationBool(true);
assertion.setExpectNull(false); assertion.setExpectNull(false);
assertion.setInvert(false); assertion.setInvert(false);
assertion.setIsRegex(false); assertion.setIsRegex(true);
return assertion; return assertion;
} }

View File

@ -157,17 +157,6 @@ public class ApiAutomationService {
return request; return request;
} }
public List<String> selectIdsNotExistsInPlan(String projectId, String planId) {
return extApiScenarioMapper.selectIdsNotExistsInPlan(projectId, planId);
}
public void deleteByIds(List<String> nodeIds) {
ApiScenarioExample example = new ApiScenarioExample();
example.createCriteria().andApiScenarioModuleIdIn(nodeIds);
apiScenarioMapper.deleteByExample(example);
}
public void removeToGcByIds(List<String> nodeIds) { public void removeToGcByIds(List<String> nodeIds) {
ApiScenarioExample example = new ApiScenarioExample(); ApiScenarioExample example = new ApiScenarioExample();
example.createCriteria().andApiScenarioModuleIdIn(nodeIds); example.createCriteria().andApiScenarioModuleIdIn(nodeIds);
@ -347,7 +336,7 @@ public class ApiAutomationService {
} }
public byte[] loadFileAsBytes(FileOperationRequest fileOperationRequest) { public byte[] loadFileAsBytes(FileOperationRequest fileOperationRequest) {
File file = new File("/opt/metersphere/data/body/" + fileOperationRequest.getId() + "_" + fileOperationRequest.getName()); File file = new File(FileUtils.BODY_FILE_DIR + fileOperationRequest.getId() + "_" + fileOperationRequest.getName());
try (FileInputStream fis = new FileInputStream(file); try (FileInputStream fis = new FileInputStream(file);
ByteArrayOutputStream bos = new ByteArrayOutputStream(1000);) { ByteArrayOutputStream bos = new ByteArrayOutputStream(1000);) {
byte[] b = new byte[1000]; byte[] b = new byte[1000];
@ -362,7 +351,7 @@ public class ApiAutomationService {
return null; return null;
} }
public void createScenarioReport(String id, String scenarioId, String scenarioName, String triggerMode, String execType, String projectId, String userID) { public APIScenarioReportResult createScenarioReport(String id, String scenarioId, String scenarioName, String triggerMode, String execType, String projectId, String userID) {
APIScenarioReportResult report = new APIScenarioReportResult(); APIScenarioReportResult report = new APIScenarioReportResult();
report.setId(id); report.setId(id);
report.setTestId(id); report.setTestId(id);
@ -384,87 +373,55 @@ public class ApiAutomationService {
report.setProjectId(projectId); report.setProjectId(projectId);
report.setScenarioName(scenarioName); report.setScenarioName(scenarioName);
report.setScenarioId(scenarioId); report.setScenarioId(scenarioId);
apiScenarioReportMapper.insert(report); return report;
} }
/**
* 生成HashTree private void parse(String scenarioDefinition, MsScenario scenario) {
* ObjectMapper mapper = new ObjectMapper();
* @param apiScenarios 场景 mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
* @param request 请求参数 try {
* @param reportIds 报告ID JSONObject element = JSON.parseObject(scenarioDefinition);
* @return hashTree // 多态JSON普通转换会丢失内容需要通过 ObjectMapper 获取
*/ if (element != null && StringUtils.isNotEmpty(element.getString("hashTree"))) {
private HashTree generateHashTree(List<ApiScenarioWithBLOBs> apiScenarios, RunScenarioRequest request, List<String> reportIds) { LinkedList<MsTestElement> elements = mapper.readValue(element.getString("hashTree"),
HashTree jmeterHashTree = new ListedHashTree(); new TypeReference<LinkedList<MsTestElement>>() {
});
scenario.setHashTree(elements);
}
if (element != null && StringUtils.isNotEmpty(element.getString("variables"))) {
LinkedList<ScenarioVariable> variables = mapper.readValue(element.getString("variables"),
new TypeReference<LinkedList<ScenarioVariable>>() {
});
scenario.setVariables(variables);
}
} catch (Exception e) {
e.printStackTrace();
LogUtil.error(e.getMessage());
}
}
private HashTree generateHashTree(ApiScenarioWithBLOBs item, String reportId, Map<String, String> planEnvMap) {
HashTree jmeterHashTree = new HashTree();
MsTestPlan testPlan = new MsTestPlan(); MsTestPlan testPlan = new MsTestPlan();
testPlan.setHashTree(new LinkedList<>()); testPlan.setHashTree(new LinkedList<>());
try { try {
boolean isFirst = true; MsThreadGroup group = new MsThreadGroup();
for (ApiScenarioWithBLOBs item : apiScenarios) { group.setLabel(item.getName());
if (item.getStepTotal() == null || item.getStepTotal() == 0) { group.setName(reportId);
// 只有一个场景且没有测试步骤则提示
if (apiScenarios.size() == 1) {
MSException.throwException((item.getName() + "" + Translator.get("automation_exec_info")));
}
LogUtil.warn(item.getName() + "" + Translator.get("automation_exec_info"));
continue;
}
MsThreadGroup group = new MsThreadGroup();
group.setLabel(item.getName());
group.setName(UUID.randomUUID().toString());
// 批量执行的结果直接存储为报告
if (isFirst) {
group.setName(request.getId());
isFirst = false;
}
ObjectMapper mapper = new ObjectMapper();
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
JSONObject element = JSON.parseObject(item.getScenarioDefinition());
MsScenario scenario = JSONObject.parseObject(item.getScenarioDefinition(), MsScenario.class);
// 多态JSON普通转换会丢失内容需要通过 ObjectMapper 获取 MsScenario scenario = JSONObject.parseObject(item.getScenarioDefinition(), MsScenario.class);
if (element != null && StringUtils.isNotEmpty(element.getString("hashTree"))) { if (planEnvMap.size() > 0) {
LinkedList<MsTestElement> elements = mapper.readValue(element.getString("hashTree"), scenario.setEnvironmentMap(planEnvMap);
new TypeReference<LinkedList<MsTestElement>>() {
});
scenario.setHashTree(elements);
}
if (StringUtils.isNotEmpty(element.getString("variables"))) {
LinkedList<ScenarioVariable> variables = mapper.readValue(element.getString("variables"),
new TypeReference<LinkedList<ScenarioVariable>>() {
});
scenario.setVariables(variables);
}
group.setEnableCookieShare(scenario.isEnableCookieShare());
LinkedList<MsTestElement> scenarios = new LinkedList<>();
scenarios.add(scenario);
// 创建场景报告
if (reportIds != null) {
//如果是测试计划页面触发的执行方式生成报告时createScenarioReport第二个参数需要特殊处理
if (StringUtils.equals(request.getRunMode(), ApiRunMode.SCENARIO_PLAN.name())) {
String testPlanScenarioId = item.getId();
if (request.getScenarioTestPlanIdMap() != null && request.getScenarioTestPlanIdMap().containsKey(item.getId())) {
testPlanScenarioId = request.getScenarioTestPlanIdMap().get(item.getId());
// 获取场景用例单独的执行环境
TestPlanApiScenario planApiScenario = testPlanApiScenarioMapper.selectByPrimaryKey(testPlanScenarioId);
String environment = planApiScenario.getEnvironment();
if (StringUtils.isNotBlank(environment)) {
scenario.setEnvironmentMap(JSON.parseObject(environment, Map.class));
}
}
createScenarioReport(group.getName(), testPlanScenarioId, item.getName(), request.getTriggerMode() == null ? ReportTriggerMode.MANUAL.name() : request.getTriggerMode(),
request.getExecuteType(), item.getProjectId(), request.getReportUserID());
} else {
createScenarioReport(group.getName(), item.getId(), item.getName(), request.getTriggerMode() == null ? ReportTriggerMode.MANUAL.name() : request.getTriggerMode(),
request.getExecuteType(), item.getProjectId(), request.getReportUserID());
}
reportIds.add(group.getName());
}
group.setHashTree(scenarios);
testPlan.getHashTree().add(group);
} }
parse(item.getScenarioDefinition(), scenario);
group.setEnableCookieShare(scenario.isEnableCookieShare());
LinkedList<MsTestElement> scenarios = new LinkedList<>();
scenarios.add(scenario);
group.setHashTree(scenarios);
testPlan.getHashTree().add(group);
} catch (Exception ex) { } catch (Exception ex) {
MSException.throwException(ex.getMessage()); MSException.throwException(ex.getMessage());
} }
@ -482,27 +439,11 @@ public class ApiAutomationService {
config.setOperating(true); config.setOperating(true);
try { try {
ObjectMapper mapper = new ObjectMapper();
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
JSONObject element = JSON.parseObject(apiScenario.getScenarioDefinition());
MsScenario scenario = JSONObject.parseObject(apiScenario.getScenarioDefinition(), MsScenario.class); MsScenario scenario = JSONObject.parseObject(apiScenario.getScenarioDefinition(), MsScenario.class);
// 多态JSON普通转换会丢失内容需要通过 ObjectMapper 获取
if (element != null && StringUtils.isNotEmpty(element.getString("hashTree"))) {
LinkedList<MsTestElement> elements = mapper.readValue(element.getString("hashTree"),
new TypeReference<LinkedList<MsTestElement>>() {
});
scenario.setHashTree(elements);
}
if (element != null && StringUtils.isNotEmpty(element.getString("variables"))) {
LinkedList<ScenarioVariable> variables = mapper.readValue(element.getString("variables"),
new TypeReference<LinkedList<ScenarioVariable>>() {
});
scenario.setVariables(variables);
}
if (scenario == null) { if (scenario == null) {
return null; return null;
} }
parse(apiScenario.getScenarioDefinition(), scenario);
// 针对导入的jmx 处理 // 针对导入的jmx 处理
if (CollectionUtils.isNotEmpty(scenario.getHashTree()) && (scenario.getHashTree().get(0) instanceof MsJmeterElement)) { if (CollectionUtils.isNotEmpty(scenario.getHashTree()) && (scenario.getHashTree().get(0) instanceof MsJmeterElement)) {
scenario.toHashTree(jmeterHashTree, scenario.getHashTree(), config); scenario.toHashTree(jmeterHashTree, scenario.getHashTree(), config);
@ -533,26 +474,62 @@ public class ApiAutomationService {
* @return * @return
*/ */
public String run(RunScenarioRequest request) { public String run(RunScenarioRequest request) {
ServiceUtils.getSelectAllIds(request, request.getCondition(), ServiceUtils.getSelectAllIds(request, request.getCondition(),
(query) -> extApiScenarioMapper.selectIdsByQuery((ApiScenarioRequest) query)); (query) -> extApiScenarioMapper.selectIdsByQuery((ApiScenarioRequest) query));
List<String> ids = request.getIds(); List<String> ids = request.getIds();
//检查是否有正在执行中的情景 //检查是否有正在执行中的情景
this.checkScenarioIsRunning(ids); this.checkScenarioIsRunning(ids);
List<ApiScenarioWithBLOBs> apiScenarios = extApiScenarioMapper.selectIds(ids);
String runMode = ApiRunMode.SCENARIO.name(); List<ApiScenarioWithBLOBs> apiScenarios = extApiScenarioMapper.selectIds(ids);
if (StringUtils.isNotBlank(request.getRunMode()) && StringUtils.equals(request.getRunMode(), ApiRunMode.SCENARIO_PLAN.name())) { // 只有一个场景且没有测试步骤则提示
runMode = ApiRunMode.SCENARIO_PLAN.name(); if (apiScenarios != null && apiScenarios.size() == 1 && (apiScenarios.get(0).getStepTotal() == null || apiScenarios.get(0).getStepTotal() == 0)) {
MSException.throwException((apiScenarios.get(0).getName() + "" + Translator.get("automation_exec_info")));
} }
if (StringUtils.isNotBlank(request.getRunMode()) && StringUtils.equals(request.getRunMode(), ApiRunMode.DEFINITION.name())) { if (StringUtils.isEmpty(request.getTriggerMode())) {
runMode = ApiRunMode.DEFINITION.name(); request.setTriggerMode(ReportTriggerMode.MANUAL.name());
} }
// 调用执行方法 SqlSession sqlSession = sqlSessionFactory.openSession(ExecutorType.BATCH);
List<String> reportIds = new LinkedList<>(); ApiScenarioReportMapper batchMapper = sqlSession.getMapper(ApiScenarioReportMapper.class);
HashTree hashTree = generateHashTree(apiScenarios, request, reportIds); String reportId = request.getId();
jMeterService.runDefinition(JSON.toJSONString(reportIds), hashTree, request.getReportId(), runMode); // 按照场景执行
for (ApiScenarioWithBLOBs item : apiScenarios) {
if (item.getStepTotal() == null || item.getStepTotal() == 0) {
continue;
}
APIScenarioReportResult report;
Map<String, String> planEnvMap = new HashMap<>();
//如果是测试计划页面触发的执行方式生成报告时createScenarioReport第二个参数需要特殊处理
if (StringUtils.equals(request.getRunMode(), ApiRunMode.SCENARIO_PLAN.name())) {
String testPlanScenarioId = item.getId();
if (request.getScenarioTestPlanIdMap() != null && request.getScenarioTestPlanIdMap().containsKey(item.getId())) {
testPlanScenarioId = request.getScenarioTestPlanIdMap().get(item.getId());
// 获取场景用例单独的执行环境
TestPlanApiScenario planApiScenario = testPlanApiScenarioMapper.selectByPrimaryKey(testPlanScenarioId);
String environment = planApiScenario.getEnvironment();
if (StringUtils.isNotBlank(environment)) {
planEnvMap = JSON.parseObject(environment, Map.class);
}
}
report = createScenarioReport(reportId, testPlanScenarioId, item.getName(), request.getTriggerMode(),
request.getExecuteType(), item.getProjectId(), request.getReportUserID());
} else {
report = createScenarioReport(reportId, item.getId(), item.getName(), request.getTriggerMode(),
request.getExecuteType(), item.getProjectId(), request.getReportUserID());
}
// 生成报告和HashTree
HashTree hashTree = generateHashTree(item, reportId, planEnvMap);
//存储报告
batchMapper.insert(report);
// 调用执行方法
jMeterService.runDefinition(report.getId(), hashTree, request.getReportId(), request.getRunMode());
// 重置报告ID
reportId = UUID.randomUUID().toString();
}
sqlSession.flushStatements();
return request.getId(); return request.getId();
} }
@ -598,7 +575,7 @@ public class ApiAutomationService {
public String debugRun(RunDefinitionRequest request, List<MultipartFile> bodyFiles) { public String debugRun(RunDefinitionRequest request, List<MultipartFile> bodyFiles) {
List<String> bodyUploadIds = new ArrayList<>(request.getBodyUploadIds()); List<String> bodyUploadIds = new ArrayList<>(request.getBodyUploadIds());
FileUtils.createBodyFiles(bodyUploadIds, bodyFiles); FileUtils.createBodyFiles(bodyUploadIds, bodyFiles);
Map<String,EnvironmentConfig> envConfig = new HashMap<>(); Map<String, EnvironmentConfig> envConfig = new HashMap<>();
Map<String, String> map = request.getEnvironmentMap(); Map<String, String> map = request.getEnvironmentMap();
if (map != null) { if (map != null) {
map.keySet().forEach(id -> { map.keySet().forEach(id -> {
@ -611,8 +588,9 @@ public class ApiAutomationService {
config.setConfig(envConfig); config.setConfig(envConfig);
HashTree hashTree = request.getTestElement().generateHashTree(config); HashTree hashTree = request.getTestElement().generateHashTree(config);
// 调用执行方法 // 调用执行方法
createScenarioReport(request.getId(), request.getScenarioId(), request.getScenarioName(), ReportTriggerMode.MANUAL.name(), request.getExecuteType(), request.getProjectId(), APIScenarioReportResult reportResult = createScenarioReport(request.getId(), request.getScenarioId(), request.getScenarioName(), ReportTriggerMode.MANUAL.name(), request.getExecuteType(), request.getProjectId(),
SessionUtils.getUserId()); SessionUtils.getUserId());
apiScenarioReportMapper.insert(reportResult);
// 调用执行方法 // 调用执行方法
jMeterService.runDefinition(request.getId(), hashTree, request.getReportId(), ApiRunMode.SCENARIO.name()); jMeterService.runDefinition(request.getId(), hashTree, request.getReportId(), ApiRunMode.SCENARIO.name());
return request.getId(); return request.getId();
@ -761,13 +739,12 @@ public class ApiAutomationService {
if (!apiScenarios.isEmpty()) { if (!apiScenarios.isEmpty()) {
testName = apiScenarios.get(0).getName(); testName = apiScenarios.get(0).getName();
} }
if (CollectionUtils.isEmpty(apiScenarios)) {
return null;
}
MsTestPlan testPlan = new MsTestPlan(); MsTestPlan testPlan = new MsTestPlan();
testPlan.setHashTree(new LinkedList<>()); testPlan.setHashTree(new LinkedList<>());
JmxInfoDTO dto = apiTestService.updateJmxString(generateJmx(apiScenarios.get(0)), testName, true);
HashTree jmeterHashTree = generateHashTree(apiScenarios, request, null);
String jmx = testPlan.getJmx(jmeterHashTree);
JmxInfoDTO dto = apiTestService.updateJmxString(jmx, testName, true);
String name = request.getName() + ".jmx"; String name = request.getName() + ".jmx";
dto.setName(name); dto.setName(name);
@ -931,7 +908,9 @@ public class ApiAutomationService {
Map<String, String> envMap = request.getEnvMap(); Map<String, String> envMap = request.getEnvMap();
Map<String, List<String>> mapping = request.getMapping(); Map<String, List<String>> mapping = request.getMapping();
Set<String> set = mapping.keySet(); Set<String> set = mapping.keySet();
if (set.isEmpty()) { return; } if (set.isEmpty()) {
return;
}
set.forEach(id -> { set.forEach(id -> {
Map<String, String> newEnvMap = new HashMap<>(16); Map<String, String> newEnvMap = new HashMap<>(16);
if (envMap != null && !envMap.isEmpty()) { if (envMap != null && !envMap.isEmpty()) {

View File

@ -152,7 +152,7 @@ public class ApiScenarioReportService {
apiScenarioReportDetailMapper.insert(detail); apiScenarioReportDetailMapper.insert(detail);
TestPlanApiScenario testPlanApiScenario = testPlanApiScenarioMapper.selectByPrimaryKey(report.getScenarioId()); TestPlanApiScenario testPlanApiScenario = testPlanApiScenarioMapper.selectByPrimaryKey(report.getScenarioId());
if(testPlanApiScenario!=null){ if (testPlanApiScenario != null) {
report.setScenarioId(testPlanApiScenario.getApiScenarioId()); report.setScenarioId(testPlanApiScenario.getApiScenarioId());
apiScenarioReportMapper.updateByPrimaryKeySelective(report); apiScenarioReportMapper.updateByPrimaryKeySelective(report);
if (scenarioResult.getError() > 0) { if (scenarioResult.getError() > 0) {
@ -236,17 +236,12 @@ public class ApiScenarioReportService {
/** /**
* 批量更新状态防止重复刷新报告 * 批量更新状态防止重复刷新报告
* *
* @param reportIds * @param reportId
*/ */
private void updateScenarioStatus(String reportIds) { private void updateScenarioStatus(String reportId) {
if (StringUtils.isNotEmpty(reportId)) {
if (StringUtils.isNotEmpty(reportIds)) {
List<String> list = new ArrayList<>(); List<String> list = new ArrayList<>();
try { list.add(reportId);
list = JSON.parseArray(reportIds, String.class);
} catch (Exception e) {
list.add(reportIds);
}
ApiScenarioReportExample scenarioReportExample = new ApiScenarioReportExample(); ApiScenarioReportExample scenarioReportExample = new ApiScenarioReportExample();
scenarioReportExample.createCriteria().andIdIn(list); scenarioReportExample.createCriteria().andIdIn(list);
List<ApiScenarioReport> reportList = apiScenarioReportMapper.selectByExample(scenarioReportExample); List<ApiScenarioReport> reportList = apiScenarioReportMapper.selectByExample(scenarioReportExample);
@ -350,13 +345,13 @@ public class ApiScenarioReportService {
int handleCount = 7000; int handleCount = 7000;
//每次处理的集合 //每次处理的集合
List<String> handleIdList = new ArrayList<>(handleCount); List<String> handleIdList = new ArrayList<>(handleCount);
while (ids.size() > handleCount){ while (ids.size() > handleCount) {
handleIdList = new ArrayList<>(handleCount); handleIdList = new ArrayList<>(handleCount);
List<String> otherIdList = new ArrayList<>(); List<String> otherIdList = new ArrayList<>();
for (int index = 0;index < ids.size();index++){ for (int index = 0; index < ids.size(); index++) {
if(index<handleCount){ if (index < handleCount) {
handleIdList.add(ids.get(index)); handleIdList.add(ids.get(index));
}else{ } else {
otherIdList.add(ids.get(index)); otherIdList.add(ids.get(index));
} }
} }
@ -372,7 +367,7 @@ public class ApiScenarioReportService {
} }
//处理最后剩余的数据 //处理最后剩余的数据
if(!ids.isEmpty()){ if (!ids.isEmpty()) {
ApiScenarioReportDetailExample detailExample = new ApiScenarioReportDetailExample(); ApiScenarioReportDetailExample detailExample = new ApiScenarioReportDetailExample();
detailExample.createCriteria().andReportIdIn(ids); detailExample.createCriteria().andReportIdIn(ids);
apiScenarioReportDetailMapper.deleteByExample(detailExample); apiScenarioReportDetailMapper.deleteByExample(detailExample);
@ -417,9 +412,9 @@ public class ApiScenarioReportService {
} }
public List<ApiScenarioReport> selectLastReportByIds(List<String> ids) { public List<ApiScenarioReport> selectLastReportByIds(List<String> ids) {
if(!ids.isEmpty()){ if (!ids.isEmpty()) {
return extApiScenarioReportMapper.selectLastReportByIds(ids); return extApiScenarioReportMapper.selectLastReportByIds(ids);
}else { } else {
return new ArrayList<>(0); return new ArrayList<>(0);
} }
} }

View File

@ -100,7 +100,7 @@
select test_case_review_test_case.id as id, test_case.id as caseId, test_case.name, test_case.priority, select test_case_review_test_case.id as id, test_case.id as caseId, test_case.name, test_case.priority,
test_case.type, test_case.node_path, test_case.method, test_case.num, test_case_review_test_case.reviewer, test_case.type, test_case.node_path, test_case.method, test_case.num, test_case_review_test_case.reviewer,
test_case.review_status, test_case_review_test_case.update_time, test_case_node.name as model, test_case.review_status, test_case_review_test_case.update_time, test_case_node.name as model,
project.name as projectName, test_case_review_test_case.review_id as reviewId project.name as projectName, test_case_review_test_case.review_id as reviewId,test_case.test_id as testId
from test_case_review_test_case from test_case_review_test_case
inner join test_case on test_case_review_test_case.case_id = test_case.id inner join test_case on test_case_review_test_case.case_id = test_case.id
left join test_case_node on test_case_node.id=test_case.node_id left join test_case_node on test_case_node.id=test_case.node_id

View File

@ -68,8 +68,8 @@ public class TestCaseReviewController {
@PostMapping("/edit") @PostMapping("/edit")
@RequiresRoles(value = {RoleConstants.TEST_USER, RoleConstants.TEST_MANAGER}, logical = Logical.OR) @RequiresRoles(value = {RoleConstants.TEST_USER, RoleConstants.TEST_MANAGER}, logical = Logical.OR)
public void editCaseReview(@RequestBody SaveTestCaseReviewRequest testCaseReview) { public String editCaseReview(@RequestBody SaveTestCaseReviewRequest testCaseReview) {
testCaseReviewService.editCaseReview(testCaseReview); return testCaseReviewService.editCaseReview(testCaseReview);
} }
@GetMapping("/delete/{reviewId}") @GetMapping("/delete/{reviewId}")

View File

@ -8,6 +8,7 @@ import io.metersphere.commons.exception.MSException;
import io.metersphere.commons.user.SessionUser; import io.metersphere.commons.user.SessionUser;
import io.metersphere.commons.utils.CommonBeanFactory; import io.metersphere.commons.utils.CommonBeanFactory;
import io.metersphere.commons.utils.EncryptUtils; import io.metersphere.commons.utils.EncryptUtils;
import io.metersphere.commons.utils.LogUtil;
import io.metersphere.commons.utils.SessionUtils; import io.metersphere.commons.utils.SessionUtils;
import io.metersphere.controller.request.IntegrationRequest; import io.metersphere.controller.request.IntegrationRequest;
import io.metersphere.service.IntegrationService; import io.metersphere.service.IntegrationService;
@ -15,10 +16,21 @@ import io.metersphere.service.ProjectService;
import io.metersphere.track.request.testcase.IssuesRequest; import io.metersphere.track.request.testcase.IssuesRequest;
import io.metersphere.track.service.TestCaseService; import io.metersphere.track.service.TestCaseService;
import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.StringUtils;
import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
import org.apache.http.conn.ssl.TrustStrategy;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.springframework.http.HttpHeaders; import org.springframework.http.HttpHeaders;
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
import org.springframework.web.client.RestTemplate;
import javax.net.ssl.SSLContext;
import java.security.cert.X509Certificate;
public abstract class AbstractIssuePlatform implements IssuesPlatform { public abstract class AbstractIssuePlatform implements IssuesPlatform {
private static RestTemplate restTemplate;
protected IntegrationService integrationService; protected IntegrationService integrationService;
protected TestCaseIssuesMapper testCaseIssuesMapper; protected TestCaseIssuesMapper testCaseIssuesMapper;
protected ProjectService projectService; protected ProjectService projectService;
@ -26,8 +38,29 @@ public abstract class AbstractIssuePlatform implements IssuesPlatform {
protected IssuesMapper issuesMapper; protected IssuesMapper issuesMapper;
protected ExtIssuesMapper extIssuesMapper; protected ExtIssuesMapper extIssuesMapper;
protected RestTemplate restTemplateIgnoreSSL;
protected String testCaseId; protected String testCaseId;
static {
try {
TrustStrategy acceptingTrustStrategy = (X509Certificate[] chain, String authType) -> true;
SSLContext sslContext = org.apache.http.ssl.SSLContexts.custom()
.loadTrustMaterial(null, acceptingTrustStrategy)
.build();
SSLConnectionSocketFactory csf = new SSLConnectionSocketFactory(sslContext);
CloseableHttpClient httpClient = HttpClients.custom()
.setSSLSocketFactory(csf)
.build();
HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory();
requestFactory.setHttpClient(httpClient);
restTemplate = new RestTemplate(requestFactory);
} catch (Exception e) {
LogUtil.error(e);
}
}
public AbstractIssuePlatform(IssuesRequest issuesRequest) { public AbstractIssuePlatform(IssuesRequest issuesRequest) {
this.integrationService = CommonBeanFactory.getBean(IntegrationService.class); this.integrationService = CommonBeanFactory.getBean(IntegrationService.class);
this.testCaseIssuesMapper = CommonBeanFactory.getBean(TestCaseIssuesMapper.class); this.testCaseIssuesMapper = CommonBeanFactory.getBean(TestCaseIssuesMapper.class);
@ -36,6 +69,8 @@ public abstract class AbstractIssuePlatform implements IssuesPlatform {
this.issuesMapper = CommonBeanFactory.getBean(IssuesMapper.class); this.issuesMapper = CommonBeanFactory.getBean(IssuesMapper.class);
this.extIssuesMapper = CommonBeanFactory.getBean(ExtIssuesMapper.class); this.extIssuesMapper = CommonBeanFactory.getBean(ExtIssuesMapper.class);
this.testCaseId = issuesRequest.getTestCaseId(); this.testCaseId = issuesRequest.getTestCaseId();
//
this.restTemplateIgnoreSSL = restTemplate;
} }
protected String getPlatformConfig(String platform) { protected String getPlatformConfig(String platform) {
@ -64,6 +99,7 @@ public abstract class AbstractIssuePlatform implements IssuesPlatform {
/** /**
* 获取平台与项目相关的属性 * 获取平台与项目相关的属性
*
* @return 其他平台和本地项目绑定的属性值 * @return 其他平台和本地项目绑定的属性值
*/ */
abstract String getProjectId(); abstract String getProjectId();

View File

@ -186,8 +186,8 @@ public class JiraPlatform extends AbstractIssuePlatform {
String url = object.getString("url"); String url = object.getString("url");
HttpHeaders headers = auth(account, password); HttpHeaders headers = auth(account, password);
HttpEntity<MultiValueMap> requestEntity = new HttpEntity<>(headers); HttpEntity<MultiValueMap> requestEntity = new HttpEntity<>(headers);
RestTemplate restTemplate = new RestTemplate(); // 忽略ssl
restTemplate.exchange(url + "rest/api/2/issue/createmeta", HttpMethod.GET, requestEntity, String.class); restTemplateIgnoreSSL.exchange(url + "rest/api/2/issue/createmeta", HttpMethod.GET, requestEntity, String.class);
} catch (Exception e) { } catch (Exception e) {
LogUtil.error(e.getMessage(), e); LogUtil.error(e.getMessage(), e);
MSException.throwException("验证失败!"); MSException.throwException("验证失败!");

View File

@ -203,7 +203,7 @@ public class TestCaseReviewService {
return extTestCaseReviewMapper.listByWorkspaceId(currentWorkspaceId, SessionUtils.getUserId(), SessionUtils.getCurrentProjectId()); return extTestCaseReviewMapper.listByWorkspaceId(currentWorkspaceId, SessionUtils.getUserId(), SessionUtils.getCurrentProjectId());
} }
public void editCaseReview(SaveTestCaseReviewRequest testCaseReview) { public String editCaseReview(SaveTestCaseReviewRequest testCaseReview) {
editCaseReviewer(testCaseReview); editCaseReviewer(testCaseReview);
testCaseReview.setUpdateTime(System.currentTimeMillis()); testCaseReview.setUpdateTime(System.currentTimeMillis());
checkCaseReviewExist(testCaseReview); checkCaseReviewExist(testCaseReview);
@ -221,6 +221,7 @@ public class TestCaseReviewService {
.event(NoticeConstants.Event.UPDATE) .event(NoticeConstants.Event.UPDATE)
.build(); .build();
noticeSendService.send(NoticeConstants.TaskType.REVIEW_TASK, noticeModel); noticeSendService.send(NoticeConstants.TaskType.REVIEW_TASK, noticeModel);
return testCaseReview.getId();
} }
private void editCaseReviewer(SaveTestCaseReviewRequest testCaseReview) { private void editCaseReviewer(SaveTestCaseReviewRequest testCaseReview) {

@ -1 +1 @@
Subproject commit 3f497f88ebbd312a3b7637c1b694a8e28c68c287 Subproject commit d2fc4b42117be97c679b4d15d6f979923e598f7f

View File

@ -33,7 +33,7 @@ alter table swagger_url_project
-- add_test_case -- add_test_case
alter table test_case alter table test_case
add demand_id varchar(50) null; add demand_id varchar(120) null;
alter table test_case alter table test_case
add demand_name varchar(999) null; add demand_name varchar(999) null;

View File

@ -1,15 +1,18 @@
<template> <template>
<el-card> <el-card>
<div class="card-content"> <div class="card-content">
<div class="ms-main-div" @click="showAll"> <div class="ms-main-div" @click="showAll" v-if="type!=='detail'">
<!--操作按钮--> <!--操作按钮-->
<div class="ms-opt-btn"> <div class="ms-opt-btn">
<el-button id="inputDelay" type="primary" size="small" @click="editScenario" title="ctrl + s">{{$t('commons.save')}}</el-button> <el-button id="inputDelay" type="primary" size="small" @click="editScenario" title="ctrl + s">
{{ $t('commons.save') }}
</el-button>
</div> </div>
<div class="tip">{{$t('test_track.plan_view.base_info')}}</div> <div class="tip">{{ $t('test_track.plan_view.base_info') }}</div>
<el-form :model="currentScenario" label-position="right" label-width="80px" size="small" :rules="rules" ref="currentScenario" style="margin-right: 20px"> <el-form :model="currentScenario" label-position="right" label-width="80px" size="small" :rules="rules"
ref="currentScenario" style="margin-right: 20px">
<!-- 基础信息 --> <!-- 基础信息 -->
<el-row> <el-row>
<el-col :span="7"> <el-col :span="7">
@ -99,7 +102,7 @@
<div class="ms-debug-div" @click="showAll"> <div class="ms-debug-div" @click="showAll">
<el-row style="margin: 5px"> <el-row style="margin: 5px">
<el-col :span="6" class="ms-col-one ms-font"> <el-col :span="6" class="ms-col-one ms-font">
{{currentScenario.name ===undefined || ''? $t('api_test.scenario.name') : currentScenario.name}} {{ currentScenario.name === undefined || '' ? $t('api_test.scenario.name') : currentScenario.name }}
</el-col> </el-col>
<el-col :span="3" class="ms-col-one ms-font"> <el-col :span="3" class="ms-col-one ms-font">
{{$t('api_test.automation.step_total')}}{{scenarioDefinition.length}} {{$t('api_test.automation.step_total')}}{{scenarioDefinition.length}}
@ -160,30 +163,35 @@
</div> </div>
<!--接口列表--> <!--接口列表-->
<scenario-api-relevance @save="pushApiOrCase" ref="scenarioApiRelevance"/> <scenario-api-relevance @save="pushApiOrCase" ref="scenarioApiRelevance" v-if="type!=='detail'"/>
<!--自定义接口--> <!--自定义接口-->
<el-drawer :visible.sync="customizeVisible" :destroy-on-close="true" direction="ltr" :withHeader="false" :title="$t('api_test.automation.customize_req')" style="overflow: auto" :modal="false" size="90%"> <el-drawer v-if="type!=='detail'" :visible.sync="customizeVisible" :destroy-on-close="true" direction="ltr"
:withHeader="false" :title="$t('api_test.automation.customize_req')" style="overflow: auto"
:modal="false" size="90%">
<ms-api-customize :request="customizeRequest" @addCustomizeApi="addCustomizeApi"/> <ms-api-customize :request="customizeRequest" @addCustomizeApi="addCustomizeApi"/>
</el-drawer> </el-drawer>
<!--场景导入 --> <!--场景导入 -->
<scenario-relevance @save="addScenario" ref="scenarioRelevance"/> <scenario-relevance v-if="type!=='detail'" @save="addScenario" ref="scenarioRelevance"/>
<!-- 环境 --> <!-- 环境 -->
<api-environment-config ref="environmentConfig" @close="environmentConfigClose"/> <api-environment-config v-if="type!=='detail'" ref="environmentConfig" @close="environmentConfigClose"/>
<!--执行组件--> <!--执行组件-->
<ms-run :debug="true" :environment="projectEnvMap" :reportId="reportId" :run-data="debugData" <ms-run :debug="true" v-if="type!=='detail'" :environment="projectEnvMap" :reportId="reportId"
:run-data="debugData"
@runRefresh="runRefresh" ref="runTest"/> @runRefresh="runRefresh" ref="runTest"/>
<!-- 调试结果 --> <!-- 调试结果 -->
<el-drawer :visible.sync="debugVisible" :destroy-on-close="true" direction="ltr" :withHeader="true" :modal="false" size="90%"> <el-drawer v-if="type!=='detail'" :visible.sync="debugVisible" :destroy-on-close="true" direction="ltr"
:withHeader="true" :modal="false" size="90%">
<ms-api-report-detail :report-id="reportId" :debug="true" :currentProjectId="projectId"/> <ms-api-report-detail :report-id="reportId" :debug="true" :currentProjectId="projectId"/>
</el-drawer> </el-drawer>
<!--场景公共参数--> <!--场景公共参数-->
<ms-variable-list @setVariables="setVariables" ref="scenarioParameters" class="ms-sc-variable-header"/> <ms-variable-list v-if="type!=='detail'" @setVariables="setVariables" ref="scenarioParameters"
class="ms-sc-variable-header"/>
<!--外部导入--> <!--外部导入-->
<api-import ref="apiImport" :saved="false" @refresh="apiImport"/> <api-import v-if="type!=='detail'" ref="apiImport" :saved="false" @refresh="apiImport"/>
</div> </div>
</el-card> </el-card>
</template> </template>
@ -222,6 +230,7 @@
props: { props: {
moduleOptions: Array, moduleOptions: Array,
currentScenario: {}, currentScenario: {},
type: String
}, },
components: { components: {
MsVariableList, MsVariableList,
@ -532,6 +541,9 @@
if (arr[i].type === ELEMENT_TYPE.LoopController && arr[i].hashTree && arr[i].hashTree.length > 1) { if (arr[i].type === ELEMENT_TYPE.LoopController && arr[i].hashTree && arr[i].hashTree.length > 1) {
arr[i].countController.proceed = true; arr[i].countController.proceed = true;
} }
if (!arr[i].projectId) {
arr[i].projectId = getCurrentProjectID();
}
if (arr[i].hashTree != undefined && arr[i].hashTree.length > 0) { if (arr[i].hashTree != undefined && arr[i].hashTree.length > 0) {
this.recursiveSorting(arr[i].hashTree); this.recursiveSorting(arr[i].hashTree);
} }
@ -544,6 +556,9 @@
&& this.scenarioDefinition[i].hashTree.length > 1) { && this.scenarioDefinition[i].hashTree.length > 1) {
this.scenarioDefinition[i].countController.proceed = true; this.scenarioDefinition[i].countController.proceed = true;
} }
if (!this.scenarioDefinition[i].projectId) {
this.scenarioDefinition.projectId = getCurrentProjectID();
}
if (this.scenarioDefinition[i].hashTree != undefined && this.scenarioDefinition[i].hashTree.length > 0) { if (this.scenarioDefinition[i].hashTree != undefined && this.scenarioDefinition[i].hashTree.length > 0) {
this.recursiveSorting(this.scenarioDefinition[i].hashTree); this.recursiveSorting(this.scenarioDefinition[i].hashTree);
} }
@ -919,14 +934,15 @@
} }
this.enableCookieShare = obj.enableCookieShare; this.enableCookieShare = obj.enableCookieShare;
this.scenarioDefinition = obj.hashTree; this.scenarioDefinition = obj.hashTree;
this.initProjectIds();
} }
} }
if (this.currentScenario.copy) { if (this.currentScenario.copy) {
this.path = "/api/automation/create"; this.path = "/api/automation/create";
} }
} }
this.getEnvironments(); this.sort();
this.initProjectIds();
// this.getEnvironments();
}) })
} }
}, },

View File

@ -251,16 +251,19 @@
this.reload(); this.reload();
}, },
run() { run() {
if (!this.envMap || this.envMap.size === 0) { if (this.isApiImport) {
this.$warning("请在环境配置中为该步骤所属项目选择运行环境!"); if (!this.envMap || this.envMap.size === 0) {
return false;
} else if (this.envMap && this.envMap.size > 0) {
const env = this.envMap.get(this.request.projectId);
if (!env) {
this.$warning("请在环境配置中为该步骤所属项目选择运行环境!"); this.$warning("请在环境配置中为该步骤所属项目选择运行环境!");
return false; return false;
} else if (this.envMap && this.envMap.size > 0) {
const env = this.envMap.get(this.request.projectId);
if (!env) {
this.$warning("请在环境配置中为该步骤所属项目选择运行环境!");
return false;
}
} }
} }
this.request.active = true; this.request.active = true;
this.loading = true; this.loading = true;
this.runData = []; this.runData = [];

View File

@ -108,6 +108,9 @@
recursive(arr) { recursive(arr) {
for (let i in arr) { for (let i in arr) {
arr[i].disabled = true; arr[i].disabled = true;
if (!arr[i].projectId) {
arr[i].projectId = getCurrentProjectID();
}
if (arr[i].hashTree != undefined && arr[i].hashTree.length > 0) { if (arr[i].hashTree != undefined && arr[i].hashTree.length > 0) {
this.recursive(arr[i].hashTree); this.recursive(arr[i].hashTree);
} }
@ -116,6 +119,9 @@
setDisabled(scenarioDefinition) { setDisabled(scenarioDefinition) {
for (let i in scenarioDefinition) { for (let i in scenarioDefinition) {
scenarioDefinition[i].disabled = true; scenarioDefinition[i].disabled = true;
if (!scenarioDefinition[i].projectId) {
scenarioDefinition[i].projectId = getCurrentProjectID();
}
if (scenarioDefinition[i].hashTree != undefined && scenarioDefinition[i].hashTree.length > 0) { if (scenarioDefinition[i].hashTree != undefined && scenarioDefinition[i].hashTree.length > 0) {
this.recursive(scenarioDefinition[i].hashTree); this.recursive(scenarioDefinition[i].hashTree);
} }

View File

@ -231,14 +231,6 @@ export default {
let data = response.data; let data = response.data;
this.total = data.itemCount; this.total = data.itemCount;
this.tableData = data.listObject; this.tableData = data.listObject;
for (let i = 0; i < this.tableData.length; i++) {
let path = "/test/plan/project";
this.$post(path, {planId: this.tableData[i].id}, res => {
let arr = res.data;
let projectIds = arr.map(data => data.id);
this.$set(this.tableData[i], "projectIds", projectIds);
})
}
}); });
}, },
buildPagePath(path) { buildPagePath(path) {

View File

@ -1,6 +1,6 @@
<template> <template>
<el-card style="margin-top: 5px" @click.native="selectTestCase(apiCase,$event)"> <el-card style="margin-top: 5px" @click.native="selectTestCase(apiCase,$event)">
<div @click="active(apiCase)"> <div @click="active(apiCase)" v-if="type!=='detail'">
<el-row> <el-row>
<el-col :span="5"> <el-col :span="5">
<el-row> <el-row>
@ -79,16 +79,14 @@
</div> </div>
</el-col> </el-col>
</el-row> </el-row>
<el-divider ></el-divider>
</div> </div>
<!-- 请求参数--> <!-- 请求参数-->
<el-collapse-transition> <el-collapse-transition>
<div v-if="apiCase.active"> <div v-if="apiCase.active||type==='detail'">
<el-divider></el-divider>
<p class="tip">{{ $t('api_test.definition.request.req_param') }} </p> <p class="tip">{{ $t('api_test.definition.request.req_param') }} </p>
<ms-api-request-form :isShowEnable="true" :showScript="true" :is-read-only="isReadOnly" :headers="apiCase.request.headers " :request="apiCase.request" v-if="api.protocol==='HTTP'"/>
<ms-api-request-form :isShowEnable="true" :showScript="true" :is-read-only="isReadOnly" :headers="apiCase.request.headers " :request="apiCase.request" v-if="api.protocol==='HTTP'"/>
<ms-tcp-basis-parameters :showScript="true" :request="apiCase.request" v-if="api.protocol==='TCP'"/> <ms-tcp-basis-parameters :showScript="true" :request="apiCase.request" v-if="api.protocol==='TCP'"/>
<ms-sql-basis-parameters :showScript="true" :request="apiCase.request" v-if="api.protocol==='SQL'"/> <ms-sql-basis-parameters :showScript="true" :request="apiCase.request" v-if="api.protocol==='SQL'"/>
<ms-dubbo-basis-parameters :showScript="true" :request="apiCase.request" v-if="api.protocol==='DUBBO'"/> <ms-dubbo-basis-parameters :showScript="true" :request="apiCase.request" v-if="api.protocol==='DUBBO'"/>
@ -99,12 +97,13 @@
<ms-jmx-step :request="apiCase.request" :response="apiCase.responseData"/> <ms-jmx-step :request="apiCase.request" :response="apiCase.responseData"/>
<!-- 保存操作 --> <!-- 保存操作 -->
<el-button type="primary" size="small" style="margin: 20px; float: right" @click="saveTestCase(apiCase)" v-tester> <el-button type="primary" size="small" style="margin: 20px; float: right" @click="saveTestCase(apiCase)" v-tester v-if="type!=='detail'">
{{ $t('commons.save') }} {{ $t('commons.save') }}
</el-button> </el-button>
</div> </div>
</el-collapse-transition> </el-collapse-transition>
</el-card> </el-card>
</template> </template>
<script> <script>
@ -183,6 +182,7 @@
return {} return {}
} }
}, },
type:String,
isCaseEdit: Boolean, isCaseEdit: Boolean,
}, },
watch: {}, watch: {},

View File

@ -132,7 +132,8 @@
isReadOnly: { isReadOnly: {
type: Boolean, type: Boolean,
default: false default: false
} },
type: String,
}, },
data() { data() {

View File

@ -63,11 +63,16 @@ export const Api_Case_List = [
export const Api_Scenario_List = [ export const Api_Scenario_List = [
{id: 'num', label: "ID"}, {id: 'num', label: "ID"},
{id: 'name', label: i18n.t('test_track.case.name')}, {id: 'name', label: i18n.t('test_track.case.name')},
{id: 'priority', label: i18n.t('test_track.case.priority')}, {id: 'level', label: i18n.t('api_test.automation.case_level')},
{id: 'path', label: i18n.t('api_test.definition.api_path')}, {id: 'status', label: i18n.t('test_track.plan.plan_status')},
{id: 'tags', label: i18n.t('commons.tag')}, {id: 'tags', label: i18n.t('commons.tag')},
{id: 'createUser', label: '创建人'}, {id: 'userId', label: i18n.t('api_test.automation.creator')},
{id: 'updateTime', label: i18n.t('api_test.definition.api_last_time')}, {id: 'updateTime', label: i18n.t('api_test.definition.api_last_time')},
{id: 'stepTotal', label: i18n.t('api_test.automation.step')},
{id: 'lastResult', label: i18n.t('api_test.automation.last_result')},
{id: 'passRate', label: i18n.t('api_test.automation.passing_rate')},
] ]
//测试评审-测试用例 //测试评审-测试用例
export const Test_Case_Review_Case_List = [ export const Test_Case_Review_Case_List = [

View File

@ -134,7 +134,6 @@ export default {
plannedStartTime: '', plannedStartTime: '',
plannedEndTime: '' plannedEndTime: ''
}, },
dbProjectIds: [],
rules: { rules: {
name: [ name: [
{required: true, message: this.$t('test_track.plan.input_plan_name'), trigger: 'blur'}, {required: true, message: this.$t('test_track.plan.input_plan_name'), trigger: 'blur'},
@ -160,7 +159,6 @@ export default {
let tmp = {}; let tmp = {};
Object.assign(tmp, testPlan); Object.assign(tmp, testPlan);
Object.assign(this.form, tmp); Object.assign(this.form, tmp);
this.dbProjectIds = JSON.parse(JSON.stringify(this.form.projectIds));
} }
listenGoBack(this.close); listenGoBack(this.close);
this.dialogFormVisible = true; this.dialogFormVisible = true;

View File

@ -279,14 +279,6 @@ export default {
let data = response.data; let data = response.data;
this.total = data.itemCount; this.total = data.itemCount;
this.tableData = data.listObject; this.tableData = data.listObject;
for (let i = 0; i < this.tableData.length; i++) {
let path = "/test/plan/project";
this.$post(path, {planId: this.tableData[i].id}, res => {
let arr = res.data;
let projectIds = arr.filter(d => d.id !== this.tableData[i].projectId).map(data => data.id);
this.$set(this.tableData[i], "projectIds", projectIds);
})
}
}); });
}, },
buildPagePath(path) { buildPagePath(path) {

View File

@ -85,7 +85,7 @@
<el-button type="primary" @click="saveReview"> <el-button type="primary" @click="saveReview">
{{ $t('test_track.confirm') }} {{ $t('test_track.confirm') }}
</el-button> </el-button>
<el-button type="primary" @click="reviewInfo('form')"> <el-button type="primary" @click="reviewInfo">
{{ $t('test_track.planning_execution') }} {{ $t('test_track.planning_execution') }}
</el-button> </el-button>
</div> </div>
@ -137,7 +137,23 @@ export default {
}; };
}, },
methods: { methods: {
reviewInfo(form) { openCaseReviewEditDialog(caseReview) {
this.resetForm();
this.setReviewerOptions();
this.operationType = 'save';
if (caseReview) {
//
this.operationType = 'edit';
let tmp = {};
Object.assign(tmp, caseReview);
Object.assign(this.form, tmp);
this.dbProjectIds = JSON.parse(JSON.stringify(this.form.projectIds));
}
listenGoBack(this.close);
this.dialogFormVisible = true;
},
reviewInfo() {
this.$refs['reviewForm'].validate((valid) => { this.$refs['reviewForm'].validate((valid) => {
if (valid) { if (valid) {
let param = {}; let param = {};
@ -155,7 +171,6 @@ export default {
if (!this.compareTime(new Date().getTime(), this.form.endTime)) { if (!this.compareTime(new Date().getTime(), this.form.endTime)) {
return false; return false;
} }
this.result = this.$post('/test/case/review/' + this.operationType, param, response => { this.result = this.$post('/test/case/review/' + this.operationType, param, response => {
this.dialogFormVisible = false; this.dialogFormVisible = false;
this.$router.push('/track/review/view/' + response.data); this.$router.push('/track/review/view/' + response.data);
@ -165,21 +180,7 @@ export default {
} }
}); });
}, },
openCaseReviewEditDialog(caseReview) {
this.resetForm();
this.setReviewerOptions();
this.operationType = 'save';
if (caseReview) {
//
this.operationType = 'edit';
let tmp = {};
Object.assign(tmp, caseReview);
Object.assign(this.form, tmp);
this.dbProjectIds = JSON.parse(JSON.stringify(this.form.projectIds));
}
listenGoBack(this.close);
this.dialogFormVisible = true;
},
saveReview() { saveReview() {
this.$refs['reviewForm'].validate((valid) => { this.$refs['reviewForm'].validate((valid) => {
if (valid) { if (valid) {

View File

@ -14,7 +14,7 @@
<el-menu-item index="functional">功能测试用例</el-menu-item> <el-menu-item index="functional">功能测试用例</el-menu-item>
<el-menu-item index="api">接口测试用例</el-menu-item> <el-menu-item index="api">接口测试用例</el-menu-item>
<el-menu-item index="load">性能测试用例</el-menu-item> <el-menu-item index="load">性能测试用例</el-menu-item>
<el-menu-item index="report">报告统计</el-menu-item> <!-- <el-menu-item index="report">报告统计</el-menu-item>-->
</el-menu> </el-menu>
</template> </template>
</ms-test-plan-header-bar> </ms-test-plan-header-bar>
@ -24,6 +24,7 @@
:review-id="reviewId"></test-review-api> :review-id="reviewId"></test-review-api>
<test-review-load v-if="activeIndex === 'load'" :redirectCharType="redirectCharType" :clickType="clickType" <test-review-load v-if="activeIndex === 'load'" :redirectCharType="redirectCharType" :clickType="clickType"
:review-id="reviewId"></test-review-load> :review-id="reviewId"></test-review-load>
</div> </div>
</template> </template>

View File

@ -1,55 +1,116 @@
<template> <template>
<ms-test-plan-common-component> <ms-test-plan-common-component>
<template v-slot:aside> <template v-slot:aside>
<ms-node-tree <ms-api-module
class="node-tree" v-if="model === 'api'"
:all-label="$t('commons.all_label.review')"
v-loading="result.loading"
@nodeSelectEvent="nodeChange" @nodeSelectEvent="nodeChange"
:tree-nodes="treeNodes" @protocolChange="handleProtocolChange"
ref="nodeTree"/> @refreshTable="refreshTable"
@setModuleOptions="setModuleOptions"
:review-id="reviewId"
:is-read-only="true"
:redirectCharType="redirectCharType"
ref="apiNodeTree"
>
<template v-slot:header>
<div class="model-change-radio">
<el-radio v-model="model" label="api">接口用例</el-radio>
<el-radio v-model="model" label="scenario">场景用例</el-radio>
</div>
</template>
</ms-api-module>
<ms-api-scenario-module
v-if="model === 'scenario'"
@nodeSelectEvent="nodeChange"
@refreshTable="refreshTable"
@setModuleOptions="setModuleOptions"
:is-read-only="true"
:review-id="reviewId"
ref="scenarioNodeTree">
<template v-slot:header>
<div class="model-change-radio">
<el-radio v-model="model" label="api">接口用例</el-radio>
<el-radio v-model="model" label="scenario">场景用例</el-radio>
</div>
</template>
</ms-api-scenario-module>
</template> </template>
<template v-slot:main> <template v-slot:main>
<test-review-test-case-list <test-plan-api-case-list
class="table-list" v-if="model === 'api'"
@openTestReviewRelevanceDialog="openTestReviewRelevanceDialog" :current-protocol="currentProtocol"
@refresh="refresh" :currentRow="currentRow"
:review-id="reviewId" :select-node-ids="selectNodeIds"
:trash-enable="trashEnable"
:is-case-relevance="true"
:model="'plan'"
:review-id="reviewId"
:clickType="clickType"
@refresh="refreshTree"
@relevanceCase="openTestCaseRelevanceDialog"
ref="apiCaseList"/>
<ms-test-plan-api-scenario-list
v-if="model === 'scenario'"
:select-node-ids="selectNodeIds" :select-node-ids="selectNodeIds"
:select-parent-nodes="selectParentNodes" :trash-enable="trashEnable"
:review-id="reviewId"
:clickType="clickType" :clickType="clickType"
ref="testPlanTestCaseList"/> @refresh="refreshTree"
@relevanceCase="openTestCaseRelevanceDialog"
ref="apiScenarioList"/>
</template> </template>
<test-review-relevance <test-case-api-relevance
@refresh="refresh" @refresh="refresh"
:review-id="reviewId" :review-id="reviewId"
ref="testReviewRelevance"/> :model="model"
ref="apiCaseRelevance"/>
<test-case-scenario-relevance
@refresh="refresh"
:review-id="reviewId"
:model="model"
ref="scenarioCaseRelevance"/>
</ms-test-plan-common-component> </ms-test-plan-common-component>
</template> </template>
<script> <script>
import MsTestPlanCommonComponent from "@/business/components/track/plan/view/comonents/base/TestPlanCommonComponent"; import MsTestPlanCommonComponent from "@/business/components/track/plan/view/comonents/base/TestPlanCommonComponent";
import FunctionalTestCaseList from "@/business/components/track/plan/view/comonents/functional/FunctionalTestCaseList"; import TestCaseScenarioRelevance from "@/business/components/track/plan/view/comonents/api/TestCaseScenarioRelevance";
import MsNodeTree from "@/business/components/track/common/NodeTree"; import MsTestPlanApiScenarioList from "@/business/components/track/plan/view/comonents/api/TestPlanApiScenarioList";
import TestReviewRelevance from "@/business/components/track/review/view/components/TestReviewRelevance"; import MsApiScenarioModule from "@/business/components/api/automation/scenario/ApiScenarioModule";
import TestReviewTestCaseList from "@/business/components/track/review/view/components/TestReviewTestCaseList"; import ApiCaseSimpleList from "@/business/components/api/definition/components/list/ApiCaseSimpleList";
import TestCaseApiRelevance from "@/business/components/track/plan/view/comonents/api/TestCaseApiRelevance";
import TestPlanApiCaseList from "@/business/components/track/plan/view/comonents/api/TestPlanApiCaseList";
import TestCaseRelevance from "@/business/components/track/plan/view/comonents/functional/TestCaseFunctionalRelevance";
import NodeTree from "@/business/components/track/common/NodeTree";
import MsApiModule from "../../../../api/definition/components/module/ApiModule"
export default { export default {
name: "TestReviewApi", name: "TestReviewApi",
components: { components: {
TestReviewTestCaseList, TestCaseScenarioRelevance,
TestReviewRelevance, MsNodeTree, FunctionalTestCaseList, MsTestPlanCommonComponent MsTestPlanApiScenarioList,
MsApiScenarioModule,
ApiCaseSimpleList,
TestCaseApiRelevance,
TestPlanApiCaseList,
MsTestPlanCommonComponent,
TestCaseRelevance,
NodeTree,
MsApiModule,
}, },
data() { data() {
return { return {
result: {}, result: {},
testReviews: [],
currentReview: {},
selectNodeIds: [],
selectParentNodes: [],
treeNodes: [], treeNodes: [],
isMenuShow: true, currentRow: "",
trashEnable: false,
currentProtocol: null,
currentModule: null,
selectNodeIds: [],
moduleOptions: {},
model: 'api'
} }
}, },
props: [ props: [
@ -58,32 +119,66 @@ export default {
'clickType' 'clickType'
], ],
mounted() { mounted() {
this.getNodeTreeByReviewId() this.checkRedirectCharType();
}, },
activated() { watch: {
this.getNodeTreeByReviewId() model() {
this.selectNodeIds = [];
this.moduleOptions = {};
},
redirectCharType(){
if(this.redirectCharType=='scenario'){
this.model = 'scenario';
}else{
this.model = 'api';
}
}
}, },
methods: { methods: {
refresh() { checkRedirectCharType(){
this.selectNodeIds = []; if(this.redirectCharType=='scenario'){
this.selectParentNodes = []; this.model = 'scenario';
this.$refs.testReviewRelevance.search(); }else{
this.getNodeTreeByReviewId(); this.model = 'api';
},
nodeChange(node, nodeIds, pNodes) {
this.selectNodeIds = nodeIds;
this.selectParentNodes = pNodes;
},
getNodeTreeByReviewId() {
if (this.reviewId) {
this.result = this.$get("/case/node/list/review/" + this.reviewId, response => {
this.treeNodes = response.data;
});
} }
}, },
openTestReviewRelevanceDialog() { refresh() {
this.$refs.testReviewRelevance.openTestReviewRelevanceDialog(); this.refreshTree();
this.refreshTable();
},
refreshTable() {
if (this.$refs.apiCaseList) {
this.$refs.apiCaseList.initTable();
}
if (this.$refs.apiScenarioList) {
this.$refs.apiScenarioList.search();
}
},
refreshTree() {
if (this.$refs.apiNodeTree) {
this.$refs.apiNodeTree.list();
}
if (this.$refs.scenarioNodeTree) {
this.$refs.scenarioNodeTree.list();
}
},
nodeChange(node, nodeIds, pNodes) {
this.selectNodeIds = nodeIds;
},
handleProtocolChange(protocol) {
this.currentProtocol = protocol;
},
setModuleOptions(data) {
this.moduleOptions = data;
},
openTestCaseRelevanceDialog(model) {
if (model === 'scenario') {
this.$refs.scenarioCaseRelevance.open();
} else {
this.$refs.apiCaseRelevance.open();
}
}, },
} }
} }

View File

@ -1,55 +1,52 @@
<template> <template>
<ms-test-plan-common-component> <ms-test-plan-common-component>
<template v-slot:aside> <template v-slot:aside>
<ms-node-tree <node-tree
class="node-tree" class="node-tree"
:all-label="$t('commons.all_label.review')"
v-loading="result.loading" v-loading="result.loading"
@nodeSelectEvent="nodeChange" @nodeSelectEvent="nodeChange"
:tree-nodes="treeNodes" :tree-nodes="treeNodes"
ref="nodeTree"/> ref="nodeTree"/>
</template> </template>
<template v-slot:main> <template v-slot:main>
<test-review-test-case-list <test-plan-load-case-list
class="table-list" class="table-list"
@openTestReviewRelevanceDialog="openTestReviewRelevanceDialog"
@refresh="refresh" @refresh="refresh"
:review-id="reviewId" :review-id="reviewId"
:select-node-ids="selectNodeIds"
:select-parent-nodes="selectParentNodes"
:clickType="clickType" :clickType="clickType"
ref="testPlanTestCaseList"/> :select-project-id="selectProjectId"
:select-parent-nodes="selectParentNodes"
@relevanceCase="openTestCaseRelevanceDialog"
ref="testPlanLoadCaseList"/>
</template> </template>
<test-review-relevance <test-case-load-relevance
@refresh="refresh" @refresh="refresh"
:review-id="reviewId" :review-id="reviewId"
ref="testReviewRelevance"/> ref="testCaseLoadRelevance"/>
</ms-test-plan-common-component> </ms-test-plan-common-component>
</template> </template>
<script> <script>
import MsTestPlanCommonComponent from "@/business/components/track/plan/view/comonents/base/TestPlanCommonComponent"; import MsTestPlanCommonComponent from "@/business/components/track/plan/view/comonents/base/TestPlanCommonComponent";
import FunctionalTestCaseList from "@/business/components/track/plan/view/comonents/functional/FunctionalTestCaseList"; import NodeTree from "@/business/components/track/common/NodeTree";
import MsNodeTree from "@/business/components/track/common/NodeTree"; import TestPlanLoadCaseList from "@/business/components/track/plan/view/comonents/load/TestPlanLoadCaseList";
import TestReviewRelevance from "@/business/components/track/review/view/components/TestReviewRelevance"; import TestCaseLoadRelevance from "@/business/components/track/plan/view/comonents/load/TestCaseLoadRelevance";
import TestReviewTestCaseList from "@/business/components/track/review/view/components/TestReviewTestCaseList";
export default { export default {
name: "TestReviewLoad", name: "TestReviewLoad",
components: { components: {
TestReviewTestCaseList, MsTestPlanCommonComponent,
TestReviewRelevance, MsNodeTree, FunctionalTestCaseList, MsTestPlanCommonComponent NodeTree,
TestPlanLoadCaseList,
TestCaseLoadRelevance,
}, },
data() { data() {
return { return {
result: {}, result: {},
testReviews: [],
currentReview: {},
selectNodeIds: [], selectNodeIds: [],
selectParentNodes: [], selectParentNodes: [],
selectProjectId: "",
treeNodes: [], treeNodes: [],
isMenuShow: true,
} }
}, },
props: [ props: [
@ -57,34 +54,42 @@ export default {
'redirectCharType', 'redirectCharType',
'clickType' 'clickType'
], ],
mounted() { watch: {
this.getNodeTreeByReviewId() planId() {
this.initData();
}
}, },
activated() { mounted() {
this.getNodeTreeByReviewId() this.initData();
}, },
methods: { methods: {
refresh() { refresh() {
this.selectNodeIds = []; this.selectProjectId = '';
this.selectParentNodes = []; this.selectParentNodes = [];
this.$refs.testReviewRelevance.search(); this.$refs.testPlanLoadCaseList.initTable();
this.getNodeTreeByReviewId(); this.getNodeTreeByPlanId();
},
initData() {
this.getNodeTreeByPlanId();
},
openTestCaseRelevanceDialog() {
this.$refs.testCaseLoadRelevance.open();
}, },
nodeChange(node, nodeIds, pNodes) { nodeChange(node, nodeIds, pNodes) {
this.selectNodeIds = nodeIds; this.selectProjectId = node.key;
this.selectParentNodes = pNodes; // node
this.$refs.testPlanLoadCaseList.currentPage = 1;
this.$refs.testPlanLoadCaseList.pageSize = 10;
}, },
getNodeTreeByReviewId() { getNodeTreeByPlanId() {
if (this.reviewId) { if (this.planId) {
this.result = this.$get("/case/node/list/review/" + this.reviewId, response => { this.result = this.$get("/case/node/list/plan/" + this.planId, response => {
this.treeNodes = response.data; this.treeNodes = response.data;
//
this.treeNodes.map(node => node.children = null);
}); });
} }
}, },
openTestReviewRelevanceDialog() {
this.$refs.testReviewRelevance.openTestReviewRelevanceDialog();
},
} }
} }
</script> </script>

View File

@ -39,11 +39,14 @@
<el-divider direction="vertical"></el-divider> <el-divider direction="vertical"></el-divider>
<el-button type="success" size="mini" <el-button type="success" size="mini"
:disabled="isReadOnly" :icon="testCase.reviewStatus === 'Pass' ? 'el-icon-check' : ''" @click="saveCase('Pass')"> :disabled="isReadOnly" :icon="testCase.reviewStatus === 'Pass' ? 'el-icon-check' : ''"
@click="saveCase('Pass')">
{{ $t('test_track.review.pass') }} {{ $t('test_track.review.pass') }}
</el-button> </el-button>
<el-button type="danger" size="mini" <el-button type="danger" size="mini"
:disabled="isReadOnly" :icon="testCase.reviewStatus === 'UnPass' ? 'el-icon-check' : ''" @click="saveCase('UnPass')"> :disabled="isReadOnly"
:icon="testCase.reviewStatus === 'UnPass' ? 'el-icon-check' : ''"
@click="saveCase('UnPass')">
{{ $t('test_track.review.un_pass') }} {{ $t('test_track.review.un_pass') }}
</el-button> </el-button>
</el-col> </el-col>
@ -66,29 +69,22 @@
</el-col> </el-col>
<el-col :span="5"> <el-col :span="5">
<span class="cast_label">{{ $t('test_track.case.case_type') }}</span> <span class="cast_label">{{ $t('test_track.case.case_type') }}</span>
<span class="cast_item" v-if="testCase.type === 'functional'">{{ <span class="cast_item" v-if="testCase.type === 'automation'">
$t('commons.functional') 场景用例
}}</span> </span>
<span class="cast_item" <span class="cast_item"
v-if="testCase.type === 'performance'">{{ $t('commons.performance') }}</span> v-if="testCase.type === 'performance'">{{ $t('commons.performance') }}</span>
<span class="cast_item" v-if="testCase.type === 'api'">{{ $t('commons.api') }}</span> <span class="cast_item" v-if="testCase.type === 'api'">{{ $t('commons.api') }}</span>
<span class="cast_item" v-if="testCase.type === 'testcase'">接口用例</span>
</el-col> </el-col>
</el-row> </el-row>
<el-row> <el-row>
<el-col :span="4" :offset="1"> <el-col :offset="1">
<span class="cast_label">{{ $t('test_track.case.method') }}</span>
<span v-if="testCase.method === 'manual'">{{ $t('test_track.case.manual') }}</span>
<span v-if="testCase.method === 'auto'">{{ $t('test_track.case.auto') }}</span>
</el-col>
<el-col :span="5">
<span class="cast_label">{{ $t('test_track.case.module') }}</span> <span class="cast_label">{{ $t('test_track.case.module') }}</span>
<span class="cast_item">{{ testCase.nodePath }}</span> <span class="cast_item">{{ testCase.nodePath }}</span>
</el-col> </el-col>
<el-col :span="4" :offset="1">
<span class="cast_label">{{ $t('test_track.plan.plan_project') }}</span>
<span class="cast_item">{{ testCase.projectName }}</span>
</el-col>
</el-row> </el-row>
<el-row> <el-row>
@ -98,7 +94,7 @@
</el-col> </el-col>
</el-row> </el-row>
<el-row v-if="testCase.method === 'auto' && testCase.testId"> <el-row>
<el-col class="test-detail" :span="20" :offset="1"> <el-col class="test-detail" :span="20" :offset="1">
<el-tabs v-model="activeTab" type="border-card"> <el-tabs v-model="activeTab" type="border-card">
<el-tab-pane name="detail" :label="$t('test_track.plan_view.test_detail')"> <el-tab-pane name="detail" :label="$t('test_track.plan_view.test_detail')">
@ -108,12 +104,17 @@
:is-read-only="true" :is-read-only="true"
:id="testCase.testId" :id="testCase.testId"
ref="performanceTestDetail"/> ref="performanceTestDetail"/>
<api-case-item :type="mark" :api="api" :api-case="apiCase" v-if="testCase.type==='testcase'"
ref="apiCaseConfig"/>
<ms-edit-api-scenario :type="mark" v-if="testCase.type==='automation'" :currentScenario="currentScenario"
ref="autoScenarioConfig"></ms-edit-api-scenario>
</el-tab-pane> </el-tab-pane>
</el-tabs> </el-tabs>
</el-col> </el-col>
</el-row> </el-row>
<el-row v-if="testCase.method && testCase.method !== 'auto'"> <el-row v-if="testCase.type === 'function'">
<el-col :span="20" :offset="1"> <el-col :span="20" :offset="1">
<div> <div>
<span class="cast_label">{{ $t('test_track.case.steps') }}</span> <span class="cast_label">{{ $t('test_track.case.steps') }}</span>
@ -229,7 +230,8 @@
<i class="el-icon-refresh" @click="getComments(testCase)" <i class="el-icon-refresh" @click="getComments(testCase)"
style="margin-left:10px;font-size: 14px; cursor: pointer"/> style="margin-left:10px;font-size: 14px; cursor: pointer"/>
</template> </template>
<review-comment :comments="comments" :case-id="testCase.caseId" :review-id="testCase.reviewId" @getComments="getComments"/> <review-comment :comments="comments" :case-id="testCase.caseId" :review-id="testCase.reviewId"
@getComments="getComments"/>
</el-card> </el-card>
</el-col> </el-col>
</div> </div>
@ -248,9 +250,11 @@ import PerformanceTestDetail from "../../../plan/view/comonents/test/Performance
import ApiTestResult from "../../../plan/view/comonents/test/ApiTestResult"; import ApiTestResult from "../../../plan/view/comonents/test/ApiTestResult";
import ApiTestDetail from "../../../plan/view/comonents/test/ApiTestDetail"; import ApiTestDetail from "../../../plan/view/comonents/test/ApiTestDetail";
import TestPlanTestCaseStatusButton from "../../../plan/common/TestPlanTestCaseStatusButton"; import TestPlanTestCaseStatusButton from "../../../plan/common/TestPlanTestCaseStatusButton";
import {listenGoBack, removeGoBackListener} from "@/common/js/utils"; import {getCurrentProjectID, listenGoBack, removeGoBackListener} from "@/common/js/utils";
import ReviewComment from "../../commom/ReviewComment"; import ReviewComment from "../../commom/ReviewComment";
import TestCaseAttachment from "@/business/components/track/case/components/TestCaseAttachment"; import TestCaseAttachment from "@/business/components/track/case/components/TestCaseAttachment";
import ApiCaseItem from "@/business/components/api/definition/components/case/ApiCaseItem";
import MsEditApiScenario from "@/business/components/api/automation/scenario/EditApiScenario"
export default { export default {
name: "TestReviewTestCaseEdit", name: "TestReviewTestCaseEdit",
@ -261,7 +265,10 @@ export default {
ApiTestDetail, ApiTestDetail,
TestPlanTestCaseStatusButton, TestPlanTestCaseStatusButton,
ReviewComment, ReviewComment,
TestCaseAttachment TestCaseAttachment,
ApiCaseItem,
MsEditApiScenario
}, },
data() { data() {
return { return {
@ -277,7 +284,11 @@ export default {
users: [], users: [],
activeName: 'comment', activeName: 'comment',
comments: [], comments: [],
tableData: [] tableData: [],
currentScenario: {},
mark: 'detail',
api: {},
apiCase: {},
}; };
}, },
props: { props: {
@ -378,16 +389,55 @@ export default {
listenGoBack(this.handleClose); listenGoBack(this.handleClose);
this.initData(testCase); this.initData(testCase);
this.getComments(testCase); this.getComments(testCase);
this.getApiTestCase(testCase);
this.getCurrentScenario(testCase)
},
getApiTestCase(testCase) {
let param = {}
param.projectId = getCurrentProjectID();
param.id = testCase.testId;
this.result = this.$post("/api/testcase/list", param, response => {
let apiCaseList = []
this.apiCaseList = response.data;
this.apiCaseList.forEach(apiCase => {
if (apiCase.tags && apiCase.tags.length > 0) {
apiCase.tags = JSON.parse(apiCase.tags);
this.$set(apiCase, 'selected', false);
}
if (Object.prototype.toString.call(apiCase.request).match(/\[object (\w+)\]/)[1].toLowerCase() != 'object') {
apiCase.request = JSON.parse(apiCase.request);
}
if (!apiCase.request.hashTree) {
apiCase.request.hashTree = [];
}
this.apiCase = apiCase
this.handleTestCase(apiCase)
})
});
},
getCurrentScenario(testCase) {
this.result = this.$get("/api/automation/getApiScenario/" + testCase.testId, response => {
this.currentScenario=response.data
});
},
handleTestCase(testCase) {
this.$get('/api/definition/get/' + testCase.apiDefinitionId, (response) => {
this.api = response.data;
});
}, },
initTest() { initTest() {
this.$nextTick(() => { this.$nextTick(() => {
if (this.testCase.testId && this.testCase.testId !== 'other') { if (this.testCase.testId && this.testCase.testId !== 'other') {
if (this.testCase.method === 'auto') { if (this.$refs.apiTestDetail && this.testCase.type === 'api') {
if (this.$refs.apiTestDetail && this.testCase.type === 'api') { this.$refs.apiTestDetail.init();
this.$refs.apiTestDetail.init(); } else if (this.testCase.type === 'performance') {
} else if (this.testCase.type === 'performance') { this.$refs.performanceTestDetail.init();
this.$refs.performanceTestDetail.init(); } else if (this.testCase.type === 'testcase') {
} this.$refs.apiCaseConfig.active(this.api);
} else if (this.testCase.type === 'automation') {
this.$refs.autoScenarioConfig.showAll();
} }
} }
}); });

View File

@ -84,7 +84,7 @@
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <!-- <el-table-column
v-if="item.id=='method'" v-if="item.id=='method'"
prop="method" prop="method"
:filters="methodFilters" :filters="methodFilters"
@ -95,7 +95,7 @@
<template v-slot:default="scope"> <template v-slot:default="scope">
<method-table-item :value="scope.row.method"/> <method-table-item :value="scope.row.method"/>
</template> </template>
</el-table-column> </el-table-column>-->
<el-table-column <el-table-column
v-if="item.id=='nodePath'" v-if="item.id=='nodePath'"

@ -1 +1 @@
Subproject commit e4422e1dc640704febfbf469185cd85b78d2147a Subproject commit 360d7214d15951ae11b3973add795305a5c3d035

View File

@ -187,7 +187,7 @@ export default {
<style scoped> <style scoped>
.container { .container {
width: 1440px; width: 1200px;
height: 810px; height: 810px;
margin: calc((100vh - 810px) / 2) auto 0; margin: calc((100vh - 810px) / 2) auto 0;
background-color: #FFFFFF; background-color: #FFFFFF;
@ -294,6 +294,7 @@ body {
color: #2B415C; color: #2B415C;
-webkit-font-smoothing: antialiased; -webkit-font-smoothing: antialiased;
margin: 0; margin: 0;
height: auto;
} }
.form .el-input > .el-input__inner { .form .el-input > .el-input__inner {