From 7d2b2a4cc956f148e93ae80dd42c30358712d605 Mon Sep 17 00:00:00 2001 From: BugKing Date: Mon, 21 Sep 2020 10:28:24 +0800 Subject: [PATCH 01/16] =?UTF-8?q?refactor:=20=E4=BF=AE=E6=94=B9=E7=89=88?= =?UTF-8?q?=E6=9C=AC=E5=8F=B7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Dockerfile | 4 ++-- backend/pom.xml | 2 +- frontend/pom.xml | 2 +- pom.xml | 2 +- 4 files changed, 5 insertions(+), 5 deletions(-) diff --git a/Dockerfile b/Dockerfile index 705770d659..8b0f86e7e0 100644 --- a/Dockerfile +++ b/Dockerfile @@ -6,11 +6,11 @@ ARG MS_VERSION=dev RUN mkdir -p /opt/apps && mkdir -p /opt/jmeter -ADD backend/target/backend-1.1.jar /opt/apps +ADD backend/target/backend-1.3.jar /opt/apps ADD backend/target/classes/jmeter/ /opt/jmeter/ -ENV JAVA_APP_JAR=/opt/apps/backend-1.1.jar +ENV JAVA_APP_JAR=/opt/apps/backend-1.3.jar ENV AB_OFF=true diff --git a/backend/pom.xml b/backend/pom.xml index ef2af712b3..a1772fb09f 100644 --- a/backend/pom.xml +++ b/backend/pom.xml @@ -7,7 +7,7 @@ metersphere-server io.metersphere - 1.1 + 1.3 4.0.0 diff --git a/frontend/pom.xml b/frontend/pom.xml index fad8d9c5c0..e22139f987 100644 --- a/frontend/pom.xml +++ b/frontend/pom.xml @@ -7,7 +7,7 @@ metersphere-server io.metersphere - 1.1 + 1.3 4.0.0 diff --git a/pom.xml b/pom.xml index d6a5c89a3e..f43ee92c7c 100644 --- a/pom.xml +++ b/pom.xml @@ -4,7 +4,7 @@ 4.0.0 io.metersphere metersphere-server - 1.1 + 1.3 pom From 01cf93bf45e758eb8738f6c815043f0cb22bddf6 Mon Sep 17 00:00:00 2001 From: "Captain.B" Date: Mon, 21 Sep 2020 17:50:32 +0800 Subject: [PATCH 02/16] =?UTF-8?q?refactor(=E6=80=A7=E8=83=BD=E6=B5=8B?= =?UTF-8?q?=E8=AF=95):=20=E5=93=8D=E5=BA=94=E6=97=B6=E9=97=B4=E9=BB=98?= =?UTF-8?q?=E8=AE=A4=E4=B8=BA=E7=A9=BA?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../performance/test/components/PerformanceAdvancedConfig.vue | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/frontend/src/business/components/performance/test/components/PerformanceAdvancedConfig.vue b/frontend/src/business/components/performance/test/components/PerformanceAdvancedConfig.vue index 5429c37056..4459d8ef45 100644 --- a/frontend/src/business/components/performance/test/components/PerformanceAdvancedConfig.vue +++ b/frontend/src/business/components/performance/test/components/PerformanceAdvancedConfig.vue @@ -96,8 +96,7 @@
{{ $t('load_test.response_timeout') }}
- + ms From bebc97d867705832390bc8daa36ea90af54b37bd Mon Sep 17 00:00:00 2001 From: shiziyuan9527 Date: Tue, 22 Sep 2020 10:30:03 +0800 Subject: [PATCH 03/16] =?UTF-8?q?fix(=E7=94=A8=E4=BE=8B=E8=AF=84=E5=AE=A1)?= =?UTF-8?q?:=20=E4=BF=AE=E6=94=B9=E8=AF=84=E5=AE=A1=E4=BA=BA=E4=B8=BA?= =?UTF-8?q?=E5=BF=85=E5=A1=AB=E9=A1=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../components/track/review/components/TestCaseReviewEdit.vue | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue b/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue index a0941ffb98..ecb6fa1811 100644 --- a/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue +++ b/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue @@ -44,7 +44,7 @@ - + Date: Tue, 22 Sep 2020 10:35:59 +0800 Subject: [PATCH 04/16] =?UTF-8?q?feat(=E6=B5=8B=E8=AF=95=E8=B7=9F=E8=B8=AA?= =?UTF-8?q?):=20=E6=B5=8B=E8=AF=95=E8=AE=A1=E5=88=92=E6=8A=A5=E5=91=8A?= =?UTF-8?q?=E5=AF=BC=E5=87=BA?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/src/main/java/io/metersphere/xpack | 2 +- frontend/package.json | 4 +- .../TemplateComponent/TemplateComponent.vue | 49 ++++++++++++++-- .../comonents/report/TestCaseReportView.vue | 56 +++++++------------ frontend/src/business/components/xpack | 2 +- frontend/src/common/js/utils.js | 55 ++++++++++++++++++ 6 files changed, 124 insertions(+), 44 deletions(-) diff --git a/backend/src/main/java/io/metersphere/xpack b/backend/src/main/java/io/metersphere/xpack index 321c869938..d5b4969642 160000 --- a/backend/src/main/java/io/metersphere/xpack +++ b/backend/src/main/java/io/metersphere/xpack @@ -1 +1 @@ -Subproject commit 321c869938357e8c2253e5bd86c963828664ae23 +Subproject commit d5b4969642fd8d10cc2f949d7377e0a0e5217a3a diff --git a/frontend/package.json b/frontend/package.json index cf5672e833..fa0ad64992 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -30,7 +30,9 @@ "md5": "^2.3.0", "sha.js": "^2.4.11", "js-base64": "^3.4.4", - "json-bigint": "^1.0.0" + "json-bigint": "^1.0.0", + "html2canvas": "^1.0.0-rc.7", + "jspdf": "^2.1.1" }, "devDependencies": { "@vue/cli-plugin-babel": "^4.1.0", diff --git a/frontend/src/business/components/track/plan/view/comonents/report/TemplateComponent/TemplateComponent.vue b/frontend/src/business/components/track/plan/view/comonents/report/TemplateComponent/TemplateComponent.vue index d2b48f293a..47da205f32 100644 --- a/frontend/src/business/components/track/plan/view/comonents/report/TemplateComponent/TemplateComponent.vue +++ b/frontend/src/business/components/track/plan/view/comonents/report/TemplateComponent/TemplateComponent.vue @@ -13,12 +13,12 @@
- - - - - - + + + + + +
@@ -31,6 +31,8 @@ import RichTextComponent from "./RichTextComponent"; import FailureResultComponent from "./FailureResultComponent"; import DefectListComponent from "./DefectListComponent"; + import html2canvas from 'html2canvas'; + export default { name: "TemplateComponent", components: { @@ -51,6 +53,41 @@ type: Boolean, default: true }, + index: { + type: Number, + default: 0 + }, + }, + methods: { + getCanvas(canvasList) { + let index = this.index; + let componentId = this.getComponentId(); + return new Promise(function(resolve, reject) { + html2canvas(document.getElementById(componentId), { + scale: 2 + }).then(function(canvas) { + //排序 + canvasList.splice(index, 0, canvas); + resolve('success'); + }); + }); + }, + getComponentId() { + switch (this.preview.id) { + case 1: + return "baseInfoComponent"; + case 2: + return "testResultComponent"; + case 3: + return "resultChartComponent"; + case 4: + return "failureResultComponent"; + case 5: + return "defectListComponent"; + default: + return "richTextComponent"; + } + }, } } diff --git a/frontend/src/business/components/track/plan/view/comonents/report/TestCaseReportView.vue b/frontend/src/business/components/track/plan/view/comonents/report/TestCaseReportView.vue index 7277e73c68..ce08b348c4 100644 --- a/frontend/src/business/components/track/plan/view/comonents/report/TestCaseReportView.vue +++ b/frontend/src/business/components/track/plan/view/comonents/report/TestCaseReportView.vue @@ -25,38 +25,33 @@ {{$t('test_track.plan_view.edit_component')}} - +
-
- +
+
-
diff --git a/frontend/src/business/components/xpack b/frontend/src/business/components/xpack index f2d5a342c8..0a375848d0 160000 --- a/frontend/src/business/components/xpack +++ b/frontend/src/business/components/xpack @@ -1 +1 @@ -Subproject commit f2d5a342c82e629f510550d5778d752bb73bf5e7 +Subproject commit 0a375848d034d20eaf05caf11769e1c75c39235c diff --git a/frontend/src/common/js/utils.js b/frontend/src/common/js/utils.js index 10a28eb25b..6a344e4a0c 100644 --- a/frontend/src/common/js/utils.js +++ b/frontend/src/common/js/utils.js @@ -9,6 +9,7 @@ import { LicenseKey } from "./constants"; import axios from "axios"; +import {jsPDF} from "jspdf"; export function hasRole(role) { let user = getCurrentUser(); @@ -203,3 +204,57 @@ export function getUUID() { } +export function exportPdf(canvasList) { + + let pdf = new jsPDF('', 'pt', 'a4'); + + // 当前页面的当前高度 + let currentHeight = 0; + for (let canvas of canvasList) { + if (canvas) { + let contentWidth = canvas.width; + let contentHeight = canvas.height; + + //a4纸的尺寸[595.28,841.89] + let a4Width = 592.28; + let a4Height = 841.89; + + // html页面生成的canvas在pdf中图片的宽高 + let imgWidth = a4Width; + let imgHeight = a4Width/contentWidth * contentHeight; + + let pageData = canvas.toDataURL('image/jpeg', 1.0); + + // 当前图片的剩余高度 + let leftHeight = imgHeight; + + // 当前页面的剩余高度 + let blankHeight = a4Height - currentHeight; + + if (leftHeight > blankHeight) { + //页面偏移 + let position = 0; + while(leftHeight > 0) { + // 本次添加占用的高度 + let occupation = a4Height - currentHeight; + pdf.addImage(pageData, 'JPEG', 0, position + currentHeight, imgWidth, imgHeight); + currentHeight = leftHeight; + leftHeight -= occupation; + position -= occupation; + //避免添加空白页 + if(leftHeight > 0) { + pdf.addPage(); + currentHeight = 0; + } + } + } else { + pdf.addImage(pageData, 'JPEG', 0, currentHeight, imgWidth, imgHeight); + currentHeight += imgHeight; + } + } + } + + pdf.save('stone.pdf'); + +} + From d9a9d9634291cf30694f76d2906c98211536f40a Mon Sep 17 00:00:00 2001 From: shiziyuan9527 Date: Tue, 22 Sep 2020 10:50:16 +0800 Subject: [PATCH 05/16] =?UTF-8?q?fix(=E7=94=A8=E4=BE=8B=E8=AF=84=E5=AE=A1)?= =?UTF-8?q?:=20=E7=94=A8=E4=BE=8B=E8=AF=84=E5=AE=A1=E5=88=97=E8=A1=A8?= =?UTF-8?q?=E5=8F=91=E8=B5=B7=E4=BA=BA=E6=98=BE=E7=A4=BA=E7=94=A8=E6=88=B7?= =?UTF-8?q?=E5=90=8D?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../base/mapper/ext/ExtTestCaseReviewMapper.xml | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseReviewMapper.xml b/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseReviewMapper.xml index 96a18c6767..0dfdc87839 100644 --- a/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseReviewMapper.xml +++ b/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseReviewMapper.xml @@ -2,12 +2,16 @@ - + select distinct test_case_review.id, test_case_review.name, user.name as creator, test_case_review.status, + test_case_review.create_time, test_case_review.update_time, test_case_review.end_time, + test_case_review.description + from test_case_review, project, test_case_review_project, user test_case_review.id = test_case_review_project.review_id and test_case_review_project.project_id = project.id + and user.id = test_case_review.creator and test_case_review.name like CONCAT('%', #{request.name},'%') @@ -21,7 +25,8 @@ - select distinct test_case_review.* from test_case_review, project, test_case_review_project where test_case_review.id = test_case_review_project.review_id From 1e7a1aa83f1521a652acadd74c5a735a9ed5844b Mon Sep 17 00:00:00 2001 From: fit2-zhao Date: Tue, 22 Sep 2020 11:03:14 +0800 Subject: [PATCH 06/16] =?UTF-8?q?refactor(=E6=B5=8B=E8=AF=95=E8=B7=9F?= =?UTF-8?q?=E8=B8=AA):=20=E9=87=8D=E6=9E=84xmind=20=E5=AF=BC=E5=85=A5?= =?UTF-8?q?=E6=96=B9=E6=B3=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../track/service/TestCaseService.java | 13 +- ...stCaseParser.java => XmindCaseParser.java} | 104 +++------- .../metersphere/xmind/parser/XmindParser.java | 178 +++++++++--------- .../xmind/parser/domain/Attached.java | 19 -- .../xmind/parser/domain/Children.java | 12 -- .../xmind/parser/domain/Comments.java | 12 -- .../xmind/parser/domain/JsonRootBean.java | 12 -- .../xmind/parser/domain/Notes.java | 10 - .../xmind/parser/domain/RootTopic.java | 16 -- .../template/{testcase.xml => xmind.xml} | Bin .../io/metersphere/xmind/utils/FileUtil.java | 61 ++++++ 11 files changed, 190 insertions(+), 247 deletions(-) rename backend/src/main/java/io/metersphere/xmind/{XmindToTestCaseParser.java => XmindCaseParser.java} (76%) delete mode 100755 backend/src/main/java/io/metersphere/xmind/parser/domain/Attached.java delete mode 100755 backend/src/main/java/io/metersphere/xmind/parser/domain/Children.java delete mode 100755 backend/src/main/java/io/metersphere/xmind/parser/domain/Comments.java delete mode 100755 backend/src/main/java/io/metersphere/xmind/parser/domain/JsonRootBean.java delete mode 100755 backend/src/main/java/io/metersphere/xmind/parser/domain/Notes.java delete mode 100755 backend/src/main/java/io/metersphere/xmind/parser/domain/RootTopic.java rename backend/src/main/java/io/metersphere/xmind/template/{testcase.xml => xmind.xml} (100%) create mode 100644 backend/src/main/java/io/metersphere/xmind/utils/FileUtil.java diff --git a/backend/src/main/java/io/metersphere/track/service/TestCaseService.java b/backend/src/main/java/io/metersphere/track/service/TestCaseService.java index 2239515a26..777e82404f 100644 --- a/backend/src/main/java/io/metersphere/track/service/TestCaseService.java +++ b/backend/src/main/java/io/metersphere/track/service/TestCaseService.java @@ -27,7 +27,7 @@ import io.metersphere.i18n.Translator; import io.metersphere.track.dto.TestCaseDTO; import io.metersphere.track.request.testcase.QueryTestCaseRequest; import io.metersphere.track.request.testcase.TestCaseBatchRequest; -import io.metersphere.xmind.XmindToTestCaseParser; +import io.metersphere.xmind.XmindCaseParser; import org.apache.commons.lang3.StringUtils; import org.apache.ibatis.session.ExecutorType; import org.apache.ibatis.session.SqlSession; @@ -273,13 +273,18 @@ public class TestCaseService { if (multipartFile.getOriginalFilename().endsWith(".xmind")) { try { errList = new ArrayList<>(); - String processLog = new XmindToTestCaseParser(this, userId, projectId, testCaseNames).importXmind(multipartFile); + XmindCaseParser xmindParser = new XmindCaseParser(this, userId, projectId, testCaseNames); + String processLog = xmindParser.parse(multipartFile); if (!StringUtils.isEmpty(processLog)) { excelResponse.setSuccess(false); - ExcelErrData excelErrData = new ExcelErrData(null, 1, Translator.get("upload_fail")+":"+ processLog); + ExcelErrData excelErrData = new ExcelErrData(null, 1, Translator.get("upload_fail") + ":" + processLog); errList.add(excelErrData); excelResponse.setErrList(errList); } else { + if (!xmindParser.testCases.isEmpty()) { + this.saveImportData(xmindParser.testCases, projectId); + xmindParser.clear(); + } excelResponse.setSuccess(true); } } catch (Exception e) { @@ -345,7 +350,7 @@ public class TestCaseService { // 发送给客户端的数据 byte[] buff = new byte[1024]; try (OutputStream outputStream = res.getOutputStream(); - BufferedInputStream bis = new BufferedInputStream(TestCaseService.class.getResourceAsStream("/io/metersphere/xmind/template/testcase.xml"));) { + BufferedInputStream bis = new BufferedInputStream(TestCaseService.class.getResourceAsStream("/io/metersphere/xmind/template/xmind.xml"));) { int i = bis.read(buff); while (i != -1) { outputStream.write(buff, 0, buff.length); diff --git a/backend/src/main/java/io/metersphere/xmind/XmindToTestCaseParser.java b/backend/src/main/java/io/metersphere/xmind/XmindCaseParser.java similarity index 76% rename from backend/src/main/java/io/metersphere/xmind/XmindToTestCaseParser.java rename to backend/src/main/java/io/metersphere/xmind/XmindCaseParser.java index 699948c904..4b2b849f38 100644 --- a/backend/src/main/java/io/metersphere/xmind/XmindToTestCaseParser.java +++ b/backend/src/main/java/io/metersphere/xmind/XmindCaseParser.java @@ -11,15 +11,11 @@ import io.metersphere.excel.domain.TestCaseExcelData; import io.metersphere.i18n.Translator; import io.metersphere.track.service.TestCaseService; import io.metersphere.xmind.parser.XmindParser; -import io.metersphere.xmind.parser.domain.Attached; -import io.metersphere.xmind.parser.domain.JsonRootBean; +import io.metersphere.xmind.parser.pojo.Attached; +import io.metersphere.xmind.parser.pojo.JsonRootBean; import org.springframework.util.StringUtils; import org.springframework.web.multipart.MultipartFile; -import java.io.File; -import java.io.FileOutputStream; -import java.io.InputStream; -import java.io.OutputStream; import java.util.*; import java.util.regex.Matcher; import java.util.regex.Pattern; @@ -27,33 +23,40 @@ import java.util.regex.Pattern; /** * 数据转换 * 1 解析Xmind文件 XmindParser.parseJson - * 2 解析后的JSON 转成测试用例 + * 2 解析后的JSON this.parse 转成测试用例 */ -public class XmindToTestCaseParser { +public class XmindCaseParser { private TestCaseService testCaseService; private String maintainer; private String projectId; private StringBuffer process; // 过程校验记录 + // 已存在用例名称 private Set testCaseNames; + // 案例详情重写了hashCode方法去重用 + public List testCases; + // 用于重复对比 + private List xmindDataList; - public XmindToTestCaseParser(TestCaseService testCaseService, String userId, String projectId, Set testCaseNames) { + public XmindCaseParser(TestCaseService testCaseService, String userId, String projectId, Set testCaseNames) { this.testCaseService = testCaseService; this.maintainer = userId; this.projectId = projectId; this.testCaseNames = testCaseNames; - testCaseWithBLOBs = new LinkedList<>(); + testCases = new LinkedList<>(); xmindDataList = new ArrayList<>(); process = new StringBuffer(); } - // 案例详情 - private List testCaseWithBLOBs; - // 用于重复对比 - protected List xmindDataList; + // 这里清理是为了 加快jvm 回收 + public void clear() { + xmindDataList.clear(); + testCases.clear(); + testCaseNames.clear(); + } // 递归处理案例数据 - private void makeXmind(StringBuffer processBuffer, Attached parent, int level, String nodePath, List attacheds) { + private void recursion(StringBuffer processBuffer, Attached parent, int level, String nodePath, List attacheds) { for (Attached item : attacheds) { if (isBlack(item.getTitle(), "(?:tc:|tc:|tc)")) { // 用例 item.setParent(parent); @@ -63,7 +66,7 @@ public class XmindToTestCaseParser { item.setPath(nodePath); if (item.getChildren() != null && !item.getChildren().getAttached().isEmpty()) { item.setParent(parent); - makeXmind(processBuffer, item, level + 1, nodePath, item.getChildren().getAttached()); + recursion(processBuffer, item, level + 1, nodePath, item.getChildren().getAttached()); } } } @@ -88,7 +91,7 @@ public class XmindToTestCaseParser { } // 获取步骤数据 - public String getSteps(List attacheds) { + private String getSteps(List attacheds) { JSONArray jsonArray = new JSONArray(); for (int i = 0; i < attacheds.size(); i++) { // 保持插入顺序,判断用例是否有相同的steps @@ -177,43 +180,13 @@ public class XmindToTestCaseParser { testCase.setId(UUID.randomUUID().toString()); testCase.setCreateTime(System.currentTimeMillis()); testCase.setUpdateTime(System.currentTimeMillis()); - testCaseWithBLOBs.add(testCase); + testCases.add(testCase); } xmindDataList.add(compartData); } - //获取流文件 - private static void inputStreamToFile(InputStream ins, File file) { - try (OutputStream os = new FileOutputStream(file);) { - int bytesRead = 0; - byte[] buffer = new byte[8192]; - while ((bytesRead = ins.read(buffer, 0, 8192)) != -1) { - os.write(buffer, 0, bytesRead); - } - } catch (Exception e) { - LogUtil.error(e.getMessage()); - } - } - - /** - * MultipartFile 转 File - * - * @param file - * @throws Exception - */ - private File multipartFileToFile(MultipartFile file) throws Exception { - if (file != null && file.getSize() > 0) { - try (InputStream ins = file.getInputStream();) { - File toFile = new File(file.getOriginalFilename()); - inputStreamToFile(ins, toFile); - return toFile; - } - } - return null; - } - - - public boolean validate(TestCaseWithBLOBs data) { + // 验证合法性 + private boolean validate(TestCaseWithBLOBs data) { String nodePath = data.getNodePath(); StringBuilder stringBuilder = new StringBuilder(); @@ -237,8 +210,6 @@ public class XmindToTestCaseParser { if (testCaseNames.contains(data.getName())) { boolean dbExist = testCaseService.exist(data); - boolean excelExist = false; - if (dbExist) { // db exist stringBuilder.append(Translator.get("test_case_already_exists_excel") + ":" + data.getName() + "; "); @@ -255,22 +226,11 @@ public class XmindToTestCaseParser { } // 导入思维导图处理 - public String importXmind(MultipartFile multipartFile) { + public String parse(MultipartFile multipartFile) { StringBuffer processBuffer = new StringBuffer(); - File file = null; try { - file = multipartFileToFile(multipartFile); - if (file == null || !file.exists()) - return Translator.get("incorrect_format"); - // 获取思维导图内容 - String content = XmindParser.parseJson(file); - if (StringUtils.isEmpty(content) || content.split("(?:tc:|tc:|TC:|TC:|tc|TC)").length == 1) { - return Translator.get("import_xmind_not_found"); - } - if (!StringUtils.isEmpty(content) && content.split("(?:tc:|tc:|TC:|TC:|tc|TC)").length > 500) { - return Translator.get("import_xmind_count_error"); - } + String content = XmindParser.parseJson(multipartFile); JsonRootBean root = JSON.parseObject(content, JsonRootBean.class); if (root != null && root.getRootTopic() != null && root.getRootTopic().getChildren() != null) { @@ -282,22 +242,18 @@ public class XmindToTestCaseParser { item.setPath(item.getTitle()); if (item.getChildren() != null && !item.getChildren().getAttached().isEmpty()) { item.setPath(item.getTitle()); - makeXmind(processBuffer, item, 1, item.getPath(), item.getChildren().getAttached()); + recursion(processBuffer, item, 1, item.getPath(), item.getChildren().getAttached()); } } } } - if (StringUtils.isEmpty(process.toString()) && !testCaseWithBLOBs.isEmpty()) { - testCaseService.saveImportData(testCaseWithBLOBs, projectId); - } + //if (StringUtils.isEmpty(process.toString()) && !testCaseWithBLOBs.isEmpty()) { + // testCaseService.saveImportData(testCaseWithBLOBs, projectId); + //} } catch (Exception ex) { processBuffer.append(Translator.get("incorrect_format")); LogUtil.error(ex.getMessage()); - ex.printStackTrace(); - } finally { - if (file != null) - file.delete(); - testCaseWithBLOBs.clear(); + return ex.getMessage(); } return process.toString(); } diff --git a/backend/src/main/java/io/metersphere/xmind/parser/XmindParser.java b/backend/src/main/java/io/metersphere/xmind/parser/XmindParser.java index 44e1cd5055..d61a28a9d7 100644 --- a/backend/src/main/java/io/metersphere/xmind/parser/XmindParser.java +++ b/backend/src/main/java/io/metersphere/xmind/parser/XmindParser.java @@ -1,9 +1,14 @@ package io.metersphere.xmind.parser; import com.alibaba.fastjson.JSON; -import io.metersphere.xmind.parser.domain.JsonRootBean; +import io.metersphere.commons.exception.MSException; +import io.metersphere.i18n.Translator; +import io.metersphere.xmind.parser.pojo.JsonRootBean; +import io.metersphere.xmind.utils.FileUtil; import org.apache.commons.compress.archivers.ArchiveException; import org.dom4j.DocumentException; +import org.springframework.util.StringUtils; +import org.springframework.web.multipart.MultipartFile; import java.io.File; import java.io.IOException; @@ -16,101 +21,98 @@ import java.util.Objects; * @Description 解析主体 */ public class XmindParser { - public static final String xmindZenJson = "content.json"; - public static final String xmindLegacyContent = "content.xml"; - public static final String xmindLegacyComments = "comments.xml"; + public static final String xmindZenJson = "content.json"; + public static final String xmindLegacyContent = "content.xml"; + public static final String xmindLegacyComments = "comments.xml"; - /** - * 解析脑图文件,返回content整合后的内容 - * - * @param file - * @return - * @throws IOException - * @throws ArchiveException - * @throws DocumentException - */ - public static String parseJson(File file) throws IOException, ArchiveException, DocumentException { - String res = ZipUtils.extract(file); + /** + * 解析脑图文件,返回content整合后的内容 + * + * @param multipartFile + * @return + * @throws IOException + * @throws ArchiveException + * @throws DocumentException + */ + public static String parseJson(MultipartFile multipartFile) throws IOException, ArchiveException, DocumentException { - String content = null; - if (isXmindZen(res, file)) { - content = getXmindZenContent(file, res); - } else { - content = getXmindLegacyContent(file, res); - } + File file = FileUtil.multipartFileToFile(multipartFile); + if (file == null || !file.exists()) + MSException.throwException(Translator.get("incorrect_format")); - // 删除生成的文件夹 - File dir = new File(res); - boolean flag = deleteDir(dir); - if (flag) { - // do something - } - JsonRootBean jsonRootBean = JSON.parseObject(content, JsonRootBean.class); - return (JSON.toJSONString(jsonRootBean, false)); - } + String res = ZipUtils.extract(file); + String content = null; + if (isXmindZen(res, file)) { + content = getXmindZenContent(file, res); + } else { + content = getXmindLegacyContent(file, res); + } - public static JsonRootBean parseObject(File file) throws DocumentException, ArchiveException, IOException { - String content = parseJson(file); - JsonRootBean jsonRootBean = JSON.parseObject(content, JsonRootBean.class); - return jsonRootBean; - } + // 删除生成的文件夹 + File dir = new File(res); + FileUtil.deleteDir(dir); + JsonRootBean jsonRootBean = JSON.parseObject(content, JsonRootBean.class); + // 删除零时文件 + if (file != null) + file.delete(); + String json = (JSON.toJSONString(jsonRootBean, false)); - public static boolean deleteDir(File dir) { - if (dir.isDirectory()) { - String[] children = dir.list(); - // 递归删除目录中的子目录下 - for (int i = 0; i < children.length; i++) { - boolean success = deleteDir(new File(dir, children[i])); - if (!success) { - return false; - } - } - } - // 目录此时为空,可以删除 - return dir.delete(); - } + if (StringUtils.isEmpty(content) || content.split("(?:tc:|tc:|TC:|TC:|tc|TC)").length == 1) { + MSException.throwException(Translator.get("import_xmind_not_found")); + } + if (!StringUtils.isEmpty(content) && content.split("(?:tc:|tc:|TC:|TC:|tc|TC)").length > 500) { + MSException.throwException(Translator.get("import_xmind_count_error")); + } + return json; + } - /** - * @return - */ - public static String getXmindZenContent(File file, String extractFileDir) - throws IOException, ArchiveException { - List keys = new ArrayList<>(); - keys.add(xmindZenJson); - Map map = ZipUtils.getContents(keys, file, extractFileDir); - String content = map.get(xmindZenJson); - content = XmindZen.getContent(content); - return content; - } + public static JsonRootBean parseObject(MultipartFile multipartFile) throws DocumentException, ArchiveException, IOException { + String content = parseJson(multipartFile); + JsonRootBean jsonRootBean = JSON.parseObject(content, JsonRootBean.class); + return jsonRootBean; + } - /** - * @return - */ - public static String getXmindLegacyContent(File file, String extractFileDir) - throws IOException, ArchiveException, DocumentException { - List keys = new ArrayList<>(); - keys.add(xmindLegacyContent); - keys.add(xmindLegacyComments); - Map map = ZipUtils.getContents(keys, file, extractFileDir); + /** + * @return + */ + public static String getXmindZenContent(File file, String extractFileDir) + throws IOException, ArchiveException { + List keys = new ArrayList<>(); + keys.add(xmindZenJson); + Map map = ZipUtils.getContents(keys, file, extractFileDir); + String content = map.get(xmindZenJson); + content = XmindZen.getContent(content); + return content; + } - String contentXml = map.get(xmindLegacyContent); - String commentsXml = map.get(xmindLegacyComments); - String xmlContent = XmindLegacy.getContent(contentXml, commentsXml); + /** + * @return + */ + public static String getXmindLegacyContent(File file, String extractFileDir) + throws IOException, ArchiveException, DocumentException { + List keys = new ArrayList<>(); + keys.add(xmindLegacyContent); + keys.add(xmindLegacyComments); + Map map = ZipUtils.getContents(keys, file, extractFileDir); - return xmlContent; - } + String contentXml = map.get(xmindLegacyContent); + String commentsXml = map.get(xmindLegacyComments); + String xmlContent = XmindLegacy.getContent(contentXml, commentsXml); - private static boolean isXmindZen(String res, File file) throws IOException, ArchiveException { - // 解压 - File parent = new File(res); - if (parent.isDirectory()) { - String[] files = parent.list(new ZipUtils.FileFilter()); - for (int i = 0; i < Objects.requireNonNull(files).length; i++) { - if (files[i].equals(xmindZenJson)) { - return true; - } - } - } - return false; - } + return xmlContent; + } + + private static boolean isXmindZen(String res, File file) throws IOException, ArchiveException { + // 解压 + File parent = new File(res); + if (parent.isDirectory()) { + String[] files = parent.list(new ZipUtils.FileFilter()); + for (int i = 0; i < Objects.requireNonNull(files).length; i++) { + if (files[i].equals(xmindZenJson)) { + return true; + } + } + } + return false; + } } diff --git a/backend/src/main/java/io/metersphere/xmind/parser/domain/Attached.java b/backend/src/main/java/io/metersphere/xmind/parser/domain/Attached.java deleted file mode 100755 index df7932a26d..0000000000 --- a/backend/src/main/java/io/metersphere/xmind/parser/domain/Attached.java +++ /dev/null @@ -1,19 +0,0 @@ - -package io.metersphere.xmind.parser.domain; - -import lombok.Data; - -import java.util.List; - -@Data -public class Attached { - - private String id; - private String title; - private Notes notes; - private String path; - private Attached parent; - private List comments; - private Children children; - -} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/domain/Children.java b/backend/src/main/java/io/metersphere/xmind/parser/domain/Children.java deleted file mode 100755 index 761a5fd9ce..0000000000 --- a/backend/src/main/java/io/metersphere/xmind/parser/domain/Children.java +++ /dev/null @@ -1,12 +0,0 @@ -package io.metersphere.xmind.parser.domain; - -import lombok.Data; - -import java.util.List; - -@Data -public class Children { - - private List attached; - -} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/domain/Comments.java b/backend/src/main/java/io/metersphere/xmind/parser/domain/Comments.java deleted file mode 100755 index e96dda1065..0000000000 --- a/backend/src/main/java/io/metersphere/xmind/parser/domain/Comments.java +++ /dev/null @@ -1,12 +0,0 @@ -package io.metersphere.xmind.parser.domain; - -import lombok.Data; - -@Data -public class Comments { - - private long creationTime; - private String author; - private String content; - -} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/domain/JsonRootBean.java b/backend/src/main/java/io/metersphere/xmind/parser/domain/JsonRootBean.java deleted file mode 100755 index f69dc6d9d4..0000000000 --- a/backend/src/main/java/io/metersphere/xmind/parser/domain/JsonRootBean.java +++ /dev/null @@ -1,12 +0,0 @@ -package io.metersphere.xmind.parser.domain; - -import lombok.Data; - -@Data -public class JsonRootBean { - - private String id; - private String title; - private RootTopic rootTopic; - -} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/domain/Notes.java b/backend/src/main/java/io/metersphere/xmind/parser/domain/Notes.java deleted file mode 100755 index 882ee5682d..0000000000 --- a/backend/src/main/java/io/metersphere/xmind/parser/domain/Notes.java +++ /dev/null @@ -1,10 +0,0 @@ -package io.metersphere.xmind.parser.domain; - -import lombok.Data; - -@Data -public class Notes { - - private String content; - -} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/domain/RootTopic.java b/backend/src/main/java/io/metersphere/xmind/parser/domain/RootTopic.java deleted file mode 100755 index 74deb34695..0000000000 --- a/backend/src/main/java/io/metersphere/xmind/parser/domain/RootTopic.java +++ /dev/null @@ -1,16 +0,0 @@ -package io.metersphere.xmind.parser.domain; - -import lombok.Data; - -import java.util.List; - -@Data -public class RootTopic { - - private String id; - private String title; - private Notes notes; - private List comments; - private Children children; - -} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/template/testcase.xml b/backend/src/main/java/io/metersphere/xmind/template/xmind.xml similarity index 100% rename from backend/src/main/java/io/metersphere/xmind/template/testcase.xml rename to backend/src/main/java/io/metersphere/xmind/template/xmind.xml diff --git a/backend/src/main/java/io/metersphere/xmind/utils/FileUtil.java b/backend/src/main/java/io/metersphere/xmind/utils/FileUtil.java new file mode 100644 index 0000000000..60ad9cf53f --- /dev/null +++ b/backend/src/main/java/io/metersphere/xmind/utils/FileUtil.java @@ -0,0 +1,61 @@ +package io.metersphere.xmind.utils; + +import io.metersphere.commons.utils.LogUtil; +import org.springframework.web.multipart.MultipartFile; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.InputStream; +import java.io.OutputStream; + +public class FileUtil { + + //获取流文件 + private static void inputStreamToFile(InputStream ins, File file) { + try (OutputStream os = new FileOutputStream(file);) { + int bytesRead = 0; + byte[] buffer = new byte[8192]; + while ((bytesRead = ins.read(buffer, 0, 8192)) != -1) { + os.write(buffer, 0, bytesRead); + } + } catch (Exception e) { + LogUtil.error(e.getMessage()); + } + } + + /** + * MultipartFile 转 File + * + * @param file + * @throws Exception + */ + public static File multipartFileToFile(MultipartFile file) { + if (file != null && file.getSize() > 0) { + try (InputStream ins = file.getInputStream();) { + File toFile = new File(file.getOriginalFilename()); + inputStreamToFile(ins, toFile); + return toFile; + } catch (Exception e) { + LogUtil.error(e.getMessage()); + } + } + return null; + } + + public static boolean deleteDir(File dir) { + if (dir.isDirectory()) { + String[] children = dir.list(); + // 递归删除目录中的子目录下 + for (int i = 0; i < children.length; i++) { + boolean success = deleteDir(new File(dir, children[i])); + if (!success) { + return false; + } + } + } + // 目录此时为空,可以删除 + return dir.delete(); + } + + +} From 421b522cc62bfefa3fe9e531345c62e4bbfc479a Mon Sep 17 00:00:00 2001 From: fit2-zhao Date: Tue, 22 Sep 2020 11:04:27 +0800 Subject: [PATCH 07/16] =?UTF-8?q?refactor(=E6=B5=8B=E8=AF=95=E8=B7=9F?= =?UTF-8?q?=E8=B8=AA):=20=E9=87=8D=E6=9E=84pojo?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../xmind/parser/pojo/Attached.java | 19 +++++++++++++++++++ .../xmind/parser/pojo/Children.java | 12 ++++++++++++ .../xmind/parser/pojo/Comments.java | 12 ++++++++++++ .../xmind/parser/pojo/JsonRootBean.java | 12 ++++++++++++ .../metersphere/xmind/parser/pojo/Notes.java | 10 ++++++++++ .../xmind/parser/pojo/RootTopic.java | 16 ++++++++++++++++ 6 files changed, 81 insertions(+) create mode 100755 backend/src/main/java/io/metersphere/xmind/parser/pojo/Attached.java create mode 100755 backend/src/main/java/io/metersphere/xmind/parser/pojo/Children.java create mode 100755 backend/src/main/java/io/metersphere/xmind/parser/pojo/Comments.java create mode 100755 backend/src/main/java/io/metersphere/xmind/parser/pojo/JsonRootBean.java create mode 100755 backend/src/main/java/io/metersphere/xmind/parser/pojo/Notes.java create mode 100755 backend/src/main/java/io/metersphere/xmind/parser/pojo/RootTopic.java diff --git a/backend/src/main/java/io/metersphere/xmind/parser/pojo/Attached.java b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Attached.java new file mode 100755 index 0000000000..5781da5500 --- /dev/null +++ b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Attached.java @@ -0,0 +1,19 @@ + +package io.metersphere.xmind.parser.pojo; + +import lombok.Data; + +import java.util.List; + +@Data +public class Attached { + + private String id; + private String title; + private Notes notes; + private String path; + private Attached parent; + private List comments; + private Children children; + +} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/pojo/Children.java b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Children.java new file mode 100755 index 0000000000..c24b4cdd3d --- /dev/null +++ b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Children.java @@ -0,0 +1,12 @@ +package io.metersphere.xmind.parser.pojo; + +import lombok.Data; + +import java.util.List; + +@Data +public class Children { + + private List attached; + +} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/pojo/Comments.java b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Comments.java new file mode 100755 index 0000000000..a5204e1f97 --- /dev/null +++ b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Comments.java @@ -0,0 +1,12 @@ +package io.metersphere.xmind.parser.pojo; + +import lombok.Data; + +@Data +public class Comments { + + private long creationTime; + private String author; + private String content; + +} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/pojo/JsonRootBean.java b/backend/src/main/java/io/metersphere/xmind/parser/pojo/JsonRootBean.java new file mode 100755 index 0000000000..93d3bf0008 --- /dev/null +++ b/backend/src/main/java/io/metersphere/xmind/parser/pojo/JsonRootBean.java @@ -0,0 +1,12 @@ +package io.metersphere.xmind.parser.pojo; + +import lombok.Data; + +@Data +public class JsonRootBean { + + private String id; + private String title; + private RootTopic rootTopic; + +} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/pojo/Notes.java b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Notes.java new file mode 100755 index 0000000000..d3bfe93e24 --- /dev/null +++ b/backend/src/main/java/io/metersphere/xmind/parser/pojo/Notes.java @@ -0,0 +1,10 @@ +package io.metersphere.xmind.parser.pojo; + +import lombok.Data; + +@Data +public class Notes { + + private String content; + +} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/xmind/parser/pojo/RootTopic.java b/backend/src/main/java/io/metersphere/xmind/parser/pojo/RootTopic.java new file mode 100755 index 0000000000..eebaa0b43b --- /dev/null +++ b/backend/src/main/java/io/metersphere/xmind/parser/pojo/RootTopic.java @@ -0,0 +1,16 @@ +package io.metersphere.xmind.parser.pojo; + +import lombok.Data; + +import java.util.List; + +@Data +public class RootTopic { + + private String id; + private String title; + private Notes notes; + private List comments; + private Children children; + +} \ No newline at end of file From 74878cd8060c64fbae041d6b5049f568c9ba117b Mon Sep 17 00:00:00 2001 From: shiziyuan9527 Date: Tue, 22 Sep 2020 11:45:20 +0800 Subject: [PATCH 08/16] refactor: i18n --- backend/src/main/java/io/metersphere/xpack | 2 +- .../track/head/TrackHeaderMenus.vue | 6 +-- .../review/components/TestCaseReviewEdit.vue | 18 ++++---- .../review/components/TestCaseReviewList.vue | 18 ++++---- .../track/review/view/TestCaseReviewView.vue | 2 +- .../view/components/TestReviewRelevance.vue | 2 +- .../components/TestReviewTestCaseList.vue | 45 +++---------------- frontend/src/business/components/xpack | 2 +- frontend/src/i18n/en-US.js | 23 ++++++++++ frontend/src/i18n/zh-CN.js | 23 ++++++++++ frontend/src/i18n/zh-TW.js | 23 ++++++++++ 11 files changed, 101 insertions(+), 63 deletions(-) diff --git a/backend/src/main/java/io/metersphere/xpack b/backend/src/main/java/io/metersphere/xpack index d5b4969642..321c869938 160000 --- a/backend/src/main/java/io/metersphere/xpack +++ b/backend/src/main/java/io/metersphere/xpack @@ -1 +1 @@ -Subproject commit d5b4969642fd8d10cc2f949d7377e0a0e5217a3a +Subproject commit 321c869938357e8c2253e5bd86c963828664ae23 diff --git a/frontend/src/business/components/track/head/TrackHeaderMenus.vue b/frontend/src/business/components/track/head/TrackHeaderMenus.vue index a49833f7d2..341bb490bd 100644 --- a/frontend/src/business/components/track/head/TrackHeaderMenus.vue +++ b/frontend/src/business/components/track/head/TrackHeaderMenus.vue @@ -32,12 +32,12 @@ - + - + @@ -94,7 +94,7 @@ export default { } }, reviewRecent: { - title: "最近的评审", + title: this.$t('test_track.recent_review'), url: "/test/case/review/recent/5", index: function (item) { return '/track/review/view/' + item.id; diff --git a/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue b/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue index ecb6fa1811..9c411e16e0 100644 --- a/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue +++ b/frontend/src/business/components/track/review/components/TestCaseReviewEdit.vue @@ -3,7 +3,7 @@
@@ -23,10 +23,10 @@ - + - + @@ -62,7 +62,7 @@ - + @@ -84,7 +84,7 @@ - + diff --git a/frontend/src/business/components/track/review/components/TestCaseReviewList.vue b/frontend/src/business/components/track/review/components/TestCaseReviewList.vue index 6ac142939a..87651c0f24 100644 --- a/frontend/src/business/components/track/review/components/TestCaseReviewList.vue +++ b/frontend/src/business/components/track/review/components/TestCaseReviewList.vue @@ -3,8 +3,8 @@