From e33e92c6d7694ffc8bbfddd8e6e8b12a4d5f99a2 Mon Sep 17 00:00:00 2001 From: "Captain.B" Date: Mon, 26 Oct 2020 16:13:54 +0800 Subject: [PATCH 1/7] =?UTF-8?q?refactor(=E6=B6=88=E6=81=AF=E9=80=9A?= =?UTF-8?q?=E7=9F=A5):=20=E6=8B=86=E5=88=86=E7=BB=84=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../organization/TaskNotification.vue | 613 +----------------- .../components/DefectTaskNotification.vue | 220 +++++++ .../components/JenkinsNotification.vue | 221 +++++++ .../components/TestPlanTaskNotification.vue | 242 +++++++ .../components/TestReviewNotification.vue | 248 +++++++ 5 files changed, 955 insertions(+), 589 deletions(-) create mode 100644 frontend/src/business/components/settings/organization/components/DefectTaskNotification.vue create mode 100644 frontend/src/business/components/settings/organization/components/JenkinsNotification.vue create mode 100644 frontend/src/business/components/settings/organization/components/TestPlanTaskNotification.vue create mode 100644 frontend/src/business/components/settings/organization/components/TestReviewNotification.vue diff --git a/frontend/src/business/components/settings/organization/TaskNotification.vue b/frontend/src/business/components/settings/organization/TaskNotification.vue index 058d4c9b51..7b6838795e 100644 --- a/frontend/src/business/components/settings/organization/TaskNotification.vue +++ b/frontend/src/business/components/settings/organization/TaskNotification.vue @@ -1,488 +1,48 @@ + + diff --git a/frontend/src/business/components/settings/organization/components/JenkinsNotification.vue b/frontend/src/business/components/settings/organization/components/JenkinsNotification.vue new file mode 100644 index 0000000000..c1bb5da081 --- /dev/null +++ b/frontend/src/business/components/settings/organization/components/JenkinsNotification.vue @@ -0,0 +1,221 @@ + + + + + diff --git a/frontend/src/business/components/settings/organization/components/TestPlanTaskNotification.vue b/frontend/src/business/components/settings/organization/components/TestPlanTaskNotification.vue new file mode 100644 index 0000000000..630f769e87 --- /dev/null +++ b/frontend/src/business/components/settings/organization/components/TestPlanTaskNotification.vue @@ -0,0 +1,242 @@ + + + + + diff --git a/frontend/src/business/components/settings/organization/components/TestReviewNotification.vue b/frontend/src/business/components/settings/organization/components/TestReviewNotification.vue new file mode 100644 index 0000000000..91aaadb4a6 --- /dev/null +++ b/frontend/src/business/components/settings/organization/components/TestReviewNotification.vue @@ -0,0 +1,248 @@ + + + + + From ad97428e1de47dc03369d46cfc124bee8e788799 Mon Sep 17 00:00:00 2001 From: "Captain.B" Date: Mon, 26 Oct 2020 18:11:30 +0800 Subject: [PATCH 2/7] =?UTF-8?q?feat:=20=E6=B5=8B=E8=AF=95=E7=94=A8?= =?UTF-8?q?=E4=BE=8B=E5=89=8D=E7=BD=AE=E6=9D=A1=E4=BB=B6=E6=8D=A2=E8=A1=8C?= =?UTF-8?q?=E6=98=BE=E7=A4=BA?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../track/plan/view/comonents/TestPlanTestCaseEdit.vue | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/frontend/src/business/components/track/plan/view/comonents/TestPlanTestCaseEdit.vue b/frontend/src/business/components/track/plan/view/comonents/TestPlanTestCaseEdit.vue index b3840820da..61937b1282 100644 --- a/frontend/src/business/components/track/plan/view/comonents/TestPlanTestCaseEdit.vue +++ b/frontend/src/business/components/track/plan/view/comonents/TestPlanTestCaseEdit.vue @@ -101,7 +101,7 @@ {{ $t('test_track.case.prerequisite') }}: - {{ testCase.prerequisite }} +

{{ testCase.prerequisite }}

@@ -684,4 +684,9 @@ export default { .el-switch >>> .el-switch__label.is-active { color: #409EFF; } + +p { + white-space: pre-line; + line-height: 20px; +} From 3989c53edaa242983928fc8d1307b041bf4fdbdd Mon Sep 17 00:00:00 2001 From: "Captain.B" Date: Mon, 26 Oct 2020 18:24:35 +0800 Subject: [PATCH 3/7] =?UTF-8?q?refactor:=20=E6=B5=8B=E8=AF=95=E8=B7=9F?= =?UTF-8?q?=E8=B8=AA=E9=A6=96=E9=A1=B5=E4=BF=AE=E6=94=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../business/components/track/home/TrackHome.vue | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/frontend/src/business/components/track/home/TrackHome.vue b/frontend/src/business/components/track/home/TrackHome.vue index dd90b15cd5..7416f3a6ef 100644 --- a/frontend/src/business/components/track/home/TrackHome.vue +++ b/frontend/src/business/components/track/home/TrackHome.vue @@ -3,18 +3,17 @@ - + + + + + + -
- - - - - @@ -52,4 +51,7 @@ export default { cursor: pointer; } +.el-row { + padding-bottom: 20px; +} From 7add174a3258b630d5151019bd52d73b6bf85c46 Mon Sep 17 00:00:00 2001 From: shiziyuan9527 Date: Tue, 27 Oct 2020 11:26:34 +0800 Subject: [PATCH 4/7] =?UTF-8?q?feat(=E6=B5=8B=E8=AF=95=E8=B7=9F=E8=B8=AA):?= =?UTF-8?q?=20=E7=94=A8=E4=BE=8B=E8=AF=84=E5=AE=A1=E8=AF=84=E8=AE=BA?= =?UTF-8?q?=E6=94=AF=E6=8C=81=E7=BC=96=E8=BE=91=E5=92=8C=E5=88=A0=E9=99=A4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../mapper/ext/ExtTestCaseCommentMapper.java | 17 +++++ .../mapper/ext/ExtTestCaseCommentMapper.xml | 14 ++++ .../controller/TestCaseCommentController.java | 18 +++-- .../controller/TestCaseReviewController.java | 2 +- .../track/dto/TestCaseCommentDTO.java | 9 +++ .../track/service/TestCaseCommentService.java | 58 ++++++++------- .../track/service/TestCaseService.java | 2 +- .../resources/i18n/messages_en_US.properties | 1 + .../resources/i18n/messages_zh_CN.properties | 1 + .../resources/i18n/messages_zh_TW.properties | 1 + .../track/review/commom/ReviewComment.vue | 10 ++- .../track/review/commom/ReviewCommentItem.vue | 70 ++++++++++++++++++- .../components/TestReviewTestCaseList.vue | 2 +- frontend/src/i18n/en-US.js | 2 + frontend/src/i18n/zh-CN.js | 2 + frontend/src/i18n/zh-TW.js | 2 + 16 files changed, 174 insertions(+), 37 deletions(-) create mode 100644 backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.java create mode 100644 backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.xml create mode 100644 backend/src/main/java/io/metersphere/track/dto/TestCaseCommentDTO.java diff --git a/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.java b/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.java new file mode 100644 index 0000000000..60f44b7202 --- /dev/null +++ b/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.java @@ -0,0 +1,17 @@ +package io.metersphere.base.mapper.ext; + +import io.metersphere.track.dto.TestCaseCommentDTO; +import org.springframework.data.repository.query.Param; + +import java.util.List; + +public interface ExtTestCaseCommentMapper { + + /** + * 获取用例的评论 + * @param caseId + * @return + */ + List getCaseComments(@Param("caseId") String caseId); + +} diff --git a/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.xml b/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.xml new file mode 100644 index 0000000000..9a92b100cd --- /dev/null +++ b/backend/src/main/java/io/metersphere/base/mapper/ext/ExtTestCaseCommentMapper.xml @@ -0,0 +1,14 @@ + + + + + + + + + \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/track/controller/TestCaseCommentController.java b/backend/src/main/java/io/metersphere/track/controller/TestCaseCommentController.java index 204c3fac59..5456ee976c 100644 --- a/backend/src/main/java/io/metersphere/track/controller/TestCaseCommentController.java +++ b/backend/src/main/java/io/metersphere/track/controller/TestCaseCommentController.java @@ -1,6 +1,6 @@ package io.metersphere.track.controller; -import io.metersphere.base.domain.TestCaseComment; +import io.metersphere.track.dto.TestCaseCommentDTO; import io.metersphere.track.request.testreview.SaveCommentRequest; import io.metersphere.track.service.TestCaseCommentService; import org.springframework.web.bind.annotation.*; @@ -13,7 +13,7 @@ import java.util.List; public class TestCaseCommentController { @Resource - TestCaseCommentService testCaseCommentService; + private TestCaseCommentService testCaseCommentService; @PostMapping("/save") public void saveComment(@RequestBody SaveCommentRequest request) { @@ -21,7 +21,17 @@ public class TestCaseCommentController { } @GetMapping("/list/{caseId}") - public List getComments(@PathVariable String caseId) { - return testCaseCommentService.getComments(caseId); + public List getCaseComments(@PathVariable String caseId) { + return testCaseCommentService.getCaseComments(caseId); + } + + @GetMapping("/delete/{commentId}") + public void deleteComment(@PathVariable String commentId) { + testCaseCommentService.delete(commentId); + } + + @PostMapping("/edit") + public void editComment(@RequestBody SaveCommentRequest request) { + testCaseCommentService.edit(request); } } diff --git a/backend/src/main/java/io/metersphere/track/controller/TestCaseReviewController.java b/backend/src/main/java/io/metersphere/track/controller/TestCaseReviewController.java index 6236c1f20b..0a89b86c11 100644 --- a/backend/src/main/java/io/metersphere/track/controller/TestCaseReviewController.java +++ b/backend/src/main/java/io/metersphere/track/controller/TestCaseReviewController.java @@ -106,7 +106,7 @@ public class TestCaseReviewController { } - @PostMapping("/get/{reviewId}") + @GetMapping("/get/{reviewId}") public TestCaseReview getTestReview(@PathVariable String reviewId) { checkOwnerService.checkTestReviewOwner(reviewId); return testCaseReviewService.getTestReview(reviewId); diff --git a/backend/src/main/java/io/metersphere/track/dto/TestCaseCommentDTO.java b/backend/src/main/java/io/metersphere/track/dto/TestCaseCommentDTO.java new file mode 100644 index 0000000000..3cd40001e9 --- /dev/null +++ b/backend/src/main/java/io/metersphere/track/dto/TestCaseCommentDTO.java @@ -0,0 +1,9 @@ +package io.metersphere.track.dto; + +import io.metersphere.base.domain.TestCaseComment; +import lombok.Data; + +@Data +public class TestCaseCommentDTO extends TestCaseComment { + private String authorName; +} diff --git a/backend/src/main/java/io/metersphere/track/service/TestCaseCommentService.java b/backend/src/main/java/io/metersphere/track/service/TestCaseCommentService.java index b3aeaef768..f7d0f28af4 100644 --- a/backend/src/main/java/io/metersphere/track/service/TestCaseCommentService.java +++ b/backend/src/main/java/io/metersphere/track/service/TestCaseCommentService.java @@ -3,21 +3,23 @@ package io.metersphere.track.service; import io.metersphere.base.domain.TestCaseComment; import io.metersphere.base.domain.TestCaseCommentExample; import io.metersphere.base.domain.TestCaseWithBLOBs; -import io.metersphere.base.domain.User; import io.metersphere.base.mapper.TestCaseCommentMapper; import io.metersphere.base.mapper.TestCaseMapper; -import io.metersphere.base.mapper.TestCaseReviewMapper; -import io.metersphere.base.mapper.UserMapper; +import io.metersphere.base.mapper.ext.ExtTestCaseCommentMapper; import io.metersphere.commons.constants.NoticeConstants; +import io.metersphere.commons.exception.MSException; import io.metersphere.commons.utils.LogUtil; import io.metersphere.commons.utils.SessionUtils; +import io.metersphere.i18n.Translator; import io.metersphere.notice.domain.MessageDetail; import io.metersphere.notice.domain.MessageSettingDetail; import io.metersphere.notice.service.DingTaskService; import io.metersphere.notice.service.MailService; import io.metersphere.notice.service.NoticeService; import io.metersphere.notice.service.WxChatTaskService; +import io.metersphere.track.dto.TestCaseCommentDTO; import io.metersphere.track.request.testreview.SaveCommentRequest; +import org.apache.commons.lang3.StringUtils; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; @@ -33,21 +35,19 @@ import java.util.UUID; public class TestCaseCommentService { @Resource - TestCaseCommentMapper testCaseCommentMapper; + private TestCaseCommentMapper testCaseCommentMapper; @Resource - private TestCaseReviewMapper testCaseReviewMapper; + private MailService mailService; @Resource - UserMapper userMapper; + private TestCaseMapper testCaseMapper; @Resource - MailService mailService; + private DingTaskService dingTaskService; @Resource - TestCaseMapper testCaseMapper; + private WxChatTaskService wxChatTaskService; @Resource - DingTaskService dingTaskService; + private NoticeService noticeService; @Resource - WxChatTaskService wxChatTaskService; - @Resource - NoticeService noticeService; + private ExtTestCaseCommentMapper extTestCaseCommentMapper; public void saveComment(SaveCommentRequest request) { @@ -86,21 +86,11 @@ public class TestCaseCommentService { } - public List getComments(String caseId) { - TestCaseCommentExample testCaseCommentExample = new TestCaseCommentExample(); - testCaseCommentExample.setOrderByClause("update_time desc"); - testCaseCommentExample.createCriteria().andCaseIdEqualTo(caseId); - List testCaseComments = testCaseCommentMapper.selectByExampleWithBLOBs(testCaseCommentExample); - testCaseComments.forEach(testCaseComment -> { - String authorId = testCaseComment.getAuthor(); - User user = userMapper.selectByPrimaryKey(authorId); - String author = user == null ? authorId : user.getName(); - testCaseComment.setAuthor(author); - }); - return testCaseComments; + public List getCaseComments(String caseId) { + return extTestCaseCommentMapper.getCaseComments(caseId); } - public void deleteComment(String caseId) { + public void deleteCaseComment(String caseId) { TestCaseCommentExample testCaseCommentExample = new TestCaseCommentExample(); testCaseCommentExample.createCriteria().andCaseIdEqualTo(caseId); testCaseCommentMapper.deleteByExample(testCaseCommentExample); @@ -118,4 +108,22 @@ public class TestCaseCommentService { context = "测试评审任务通知:" + testCaseComment.getAuthor() + "在" + start + "为" + "'" + testCaseWithBLOBs.getName() + "'" + "添加评论:" + testCaseComment.getDescription(); return context; } + + public void delete(String commentId) { + checkCommentOwner(commentId); + testCaseCommentMapper.deleteByPrimaryKey(commentId); + } + + public void edit(SaveCommentRequest request) { + checkCommentOwner(request.getId()); + testCaseCommentMapper.updateByPrimaryKeySelective(request); + } + + private void checkCommentOwner(String commentId) { + TestCaseComment comment = testCaseCommentMapper.selectByPrimaryKey(commentId); + if (!StringUtils.equals(comment.getAuthor(), SessionUtils.getUser().getId())) { + MSException.throwException(Translator.get("check_owner_comment")); + } + } + } 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 4113a6f23e..a0fe4f05b6 100644 --- a/backend/src/main/java/io/metersphere/track/service/TestCaseService.java +++ b/backend/src/main/java/io/metersphere/track/service/TestCaseService.java @@ -170,7 +170,7 @@ public class TestCaseService { example.createCriteria().andCaseIdEqualTo(testCaseId); testPlanTestCaseMapper.deleteByExample(example); testCaseIssueService.delTestCaseIssues(testCaseId); - testCaseCommentService.deleteComment(testCaseId); + testCaseCommentService.deleteCaseComment(testCaseId); return testCaseMapper.deleteByPrimaryKey(testCaseId); } diff --git a/backend/src/main/resources/i18n/messages_en_US.properties b/backend/src/main/resources/i18n/messages_en_US.properties index bd58c4e206..d3ff598c78 100644 --- a/backend/src/main/resources/i18n/messages_en_US.properties +++ b/backend/src/main/resources/i18n/messages_en_US.properties @@ -164,6 +164,7 @@ check_owner_test=The current user does not have permission to operate this test check_owner_case=The current user does not have permission to operate this use case check_owner_plan=The current user does not have permission to operate this plan check_owner_review=The current user does not have permission to operate this review +check_owner_comment=The current user does not have permission to manipulate this comment upload_content_is_null=Imported content is empty test_plan_notification=Test plan notification task_defect_notification=Task defect notification diff --git a/backend/src/main/resources/i18n/messages_zh_CN.properties b/backend/src/main/resources/i18n/messages_zh_CN.properties index 91d5ffa4b6..9b172e75dd 100644 --- a/backend/src/main/resources/i18n/messages_zh_CN.properties +++ b/backend/src/main/resources/i18n/messages_zh_CN.properties @@ -164,6 +164,7 @@ check_owner_test=当前用户没有操作此测试的权限 check_owner_case=当前用户没有操作此用例的权限 check_owner_plan=当前用户没有操作此计划的权限 check_owner_review=当前用户没有操作此评审的权限 +check_owner_comment=当前用户没有操作此评论的权限 upload_content_is_null=导入内容为空 test_plan_notification=测试计划通知 task_defect_notification=缺陷任务通知 diff --git a/backend/src/main/resources/i18n/messages_zh_TW.properties b/backend/src/main/resources/i18n/messages_zh_TW.properties index a34799eac5..b46f36cf88 100644 --- a/backend/src/main/resources/i18n/messages_zh_TW.properties +++ b/backend/src/main/resources/i18n/messages_zh_TW.properties @@ -165,6 +165,7 @@ check_owner_test=當前用戶沒有操作此測試的權限 check_owner_case=當前用戶沒有操作此用例的權限 check_owner_plan=當前用戶沒有操作此計劃的權限 check_owner_review=當前用戶沒有操作此評審的權限 +check_owner_comment=當前用戶沒有操作此評論的權限 upload_content_is_null=導入內容為空 test_plan_notification=測試計畫通知 task_defect_notification=缺陷任務通知 diff --git a/frontend/src/business/components/track/review/commom/ReviewComment.vue b/frontend/src/business/components/track/review/commom/ReviewComment.vue index 0733a0235e..62e0b40ad2 100644 --- a/frontend/src/business/components/track/review/commom/ReviewComment.vue +++ b/frontend/src/business/components/track/review/commom/ReviewComment.vue @@ -1,7 +1,10 @@ @@ -72,4 +130,10 @@ export default { pre { margin: 0 0; } + +.comment-delete { + float: right; + margin-right: 5px; + cursor: pointer; +} diff --git a/frontend/src/business/components/track/review/view/components/TestReviewTestCaseList.vue b/frontend/src/business/components/track/review/view/components/TestReviewTestCaseList.vue index 6d0da9dfb0..122ab4ef4f 100644 --- a/frontend/src/business/components/track/review/view/components/TestReviewTestCaseList.vue +++ b/frontend/src/business/components/track/review/view/components/TestReviewTestCaseList.vue @@ -365,7 +365,7 @@ export default { }, getTestReviewById() { if (this.reviewId) { - this.$post('/test/case/review/get/' + this.reviewId, {}, response => { + this.$get('/test/case/review/get/' + this.reviewId, response => { this.testReview = response.data; this.refreshTestReviewRecent(); }); diff --git a/frontend/src/i18n/en-US.js b/frontend/src/i18n/en-US.js index f2518841ad..afdb78b19f 100644 --- a/frontend/src/i18n/en-US.js +++ b/frontend/src/i18n/en-US.js @@ -848,6 +848,8 @@ export default { relevance_case: "Relevance Case", last_page: "It's the end!", execute_result: "Result", + cannot_edit: "Cannot edit this comment!", + cannot_delete: "Cannot delete this comment!", }, module: { search: "Search module", diff --git a/frontend/src/i18n/zh-CN.js b/frontend/src/i18n/zh-CN.js index 4f369a478b..cf63c0527e 100644 --- a/frontend/src/i18n/zh-CN.js +++ b/frontend/src/i18n/zh-CN.js @@ -841,6 +841,8 @@ export default { send: "发送", description_is_null: "评论内容不能为空!", send_success: "评论成功!", + cannot_edit: "无法编辑此评论!", + cannot_delete: "无法删除此评论!", }, review_view: { review: "评审", diff --git a/frontend/src/i18n/zh-TW.js b/frontend/src/i18n/zh-TW.js index 23fec07053..49ccd7ec0d 100644 --- a/frontend/src/i18n/zh-TW.js +++ b/frontend/src/i18n/zh-TW.js @@ -846,6 +846,8 @@ export default { send: "發送", description_is_null: "評論內容不能為空!", send_success: "評論成功!", + cannot_edit: "無法編輯此評論!", + cannot_delete: "無法刪除此評論!", }, review_view: { review: "評審", From 200c3979506f1851ebdd27756c82d42312a4dfef Mon Sep 17 00:00:00 2001 From: fit2-zhao Date: Tue, 27 Oct 2020 11:29:05 +0800 Subject: [PATCH 5/7] =?UTF-8?q?feat(=E6=B5=8B=E8=AF=95=E8=B7=9F=E8=B8=AA):?= =?UTF-8?q?=20=E6=B7=BB=E5=8A=A0=E7=B4=A2=E5=BC=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/src/main/resources/db/migration/V37__add_plan_index.sql | 2 ++ 1 file changed, 2 insertions(+) create mode 100644 backend/src/main/resources/db/migration/V37__add_plan_index.sql diff --git a/backend/src/main/resources/db/migration/V37__add_plan_index.sql b/backend/src/main/resources/db/migration/V37__add_plan_index.sql new file mode 100644 index 0000000000..762780f3e6 --- /dev/null +++ b/backend/src/main/resources/db/migration/V37__add_plan_index.sql @@ -0,0 +1,2 @@ +ALTER TABLE `test_plan_test_case` ADD INDEX index_name ( `case_id` ); +ALTER TABLE `test_case_review_test_case` ADD INDEX index_name ( `case_id` ); \ No newline at end of file From 8988059e54f1a6d55ff14a1c3c45b379a5d30227 Mon Sep 17 00:00:00 2001 From: "Captain.B" Date: Tue, 27 Oct 2020 17:43:50 +0800 Subject: [PATCH 6/7] =?UTF-8?q?feat(LDAP):=20=E6=94=AF=E6=8C=81LDAPS?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../ldap/service/CustomSSLSocketFactory.java | 84 +++++++++++++++++++ .../metersphere/ldap/service/LdapService.java | 8 +- .../ldap/service/SSLLdapContextSource.java | 16 ++++ 3 files changed, 106 insertions(+), 2 deletions(-) create mode 100644 backend/src/main/java/io/metersphere/ldap/service/CustomSSLSocketFactory.java create mode 100644 backend/src/main/java/io/metersphere/ldap/service/SSLLdapContextSource.java diff --git a/backend/src/main/java/io/metersphere/ldap/service/CustomSSLSocketFactory.java b/backend/src/main/java/io/metersphere/ldap/service/CustomSSLSocketFactory.java new file mode 100644 index 0000000000..8a25c8c47a --- /dev/null +++ b/backend/src/main/java/io/metersphere/ldap/service/CustomSSLSocketFactory.java @@ -0,0 +1,84 @@ +package io.metersphere.ldap.service; + +import javax.net.SocketFactory; +import javax.net.ssl.SSLContext; +import javax.net.ssl.SSLSocketFactory; +import javax.net.ssl.TrustManager; +import javax.net.ssl.X509TrustManager; +import java.io.IOException; +import java.net.InetAddress; +import java.net.Socket; +import java.net.UnknownHostException; +import java.security.SecureRandom; +import java.security.cert.CertificateException; +import java.security.cert.X509Certificate; + +public class CustomSSLSocketFactory extends SSLSocketFactory { + private SSLSocketFactory socketFactory; + + public CustomSSLSocketFactory() { + try { + SSLContext ctx = SSLContext.getInstance("TLS"); + ctx.init(null, new TrustManager[]{new DummyTrustmanager()}, new SecureRandom()); + socketFactory = ctx.getSocketFactory(); + } catch (Exception ex) { + ex.printStackTrace(System.err); + } + } + + public static SocketFactory getDefault() { + return new CustomSSLSocketFactory(); + } + + @Override + public String[] getDefaultCipherSuites() { + return socketFactory.getDefaultCipherSuites(); + } + + @Override + public String[] getSupportedCipherSuites() { + return socketFactory.getSupportedCipherSuites(); + } + + @Override + public Socket createSocket(Socket socket, String string, int num, boolean bool) throws IOException { + return socketFactory.createSocket(socket, string, num, bool); + } + + @Override + public Socket createSocket(String string, int num) throws IOException, UnknownHostException { + return socketFactory.createSocket(string, num); + } + + @Override + public Socket createSocket(String string, int num, InetAddress netAdd, int i) throws IOException, UnknownHostException { + return socketFactory.createSocket(string, num, netAdd, i); + } + + @Override + public Socket createSocket(InetAddress netAdd, int num) throws IOException { + return socketFactory.createSocket(netAdd, num); + } + + @Override + public Socket createSocket(InetAddress netAdd1, int num, InetAddress netAdd2, int i) throws IOException { + return socketFactory.createSocket(netAdd1, num, netAdd2, i); + } + + + /** + * 证书 + */ + public static class DummyTrustmanager implements X509TrustManager { + public void checkClientTrusted(X509Certificate[] cert, String string) throws CertificateException { + } + + public void checkServerTrusted(X509Certificate[] cert, String string) throws CertificateException { + } + + public X509Certificate[] getAcceptedIssuers() { + return new java.security.cert.X509Certificate[0]; + } + + } +} \ No newline at end of file diff --git a/backend/src/main/java/io/metersphere/ldap/service/LdapService.java b/backend/src/main/java/io/metersphere/ldap/service/LdapService.java index 03436c179a..60a1ce0145 100644 --- a/backend/src/main/java/io/metersphere/ldap/service/LdapService.java +++ b/backend/src/main/java/io/metersphere/ldap/service/LdapService.java @@ -18,7 +18,6 @@ import org.springframework.ldap.core.DirContextOperations; import org.springframework.ldap.core.LdapTemplate; import org.springframework.ldap.core.support.AbstractContextMapper; import org.springframework.ldap.core.support.DefaultDirObjectFactory; -import org.springframework.ldap.core.support.LdapContextSource; import org.springframework.ldap.query.SearchScope; import org.springframework.stereotype.Service; @@ -147,11 +146,16 @@ public class LdapService { String credentials = EncryptUtils.aesDecrypt(password).toString(); - LdapContextSource sourceLdapCtx = new LdapContextSource(); + SSLLdapContextSource sourceLdapCtx = new SSLLdapContextSource(); sourceLdapCtx.setUrl(url); sourceLdapCtx.setUserDn(dn); sourceLdapCtx.setPassword(credentials); sourceLdapCtx.setDirObjectFactory(DefaultDirObjectFactory.class); + // todo 这里加上strategy 会报错 +// DefaultTlsDirContextAuthenticationStrategy strategy = new DefaultTlsDirContextAuthenticationStrategy(); +// strategy.setShutdownTlsGracefully(true); +// strategy.setHostnameVerifier((hostname, session) -> true); +// sourceLdapCtx.setAuthenticationStrategy(strategy); sourceLdapCtx.afterPropertiesSet(); LdapTemplate ldapTemplate = new LdapTemplate(sourceLdapCtx); ldapTemplate.setIgnorePartialResultException(true); diff --git a/backend/src/main/java/io/metersphere/ldap/service/SSLLdapContextSource.java b/backend/src/main/java/io/metersphere/ldap/service/SSLLdapContextSource.java new file mode 100644 index 0000000000..05f302d02b --- /dev/null +++ b/backend/src/main/java/io/metersphere/ldap/service/SSLLdapContextSource.java @@ -0,0 +1,16 @@ +package io.metersphere.ldap.service; + +import org.springframework.ldap.core.support.LdapContextSource; + +import javax.naming.Context; +import java.util.Hashtable; + +public class SSLLdapContextSource extends LdapContextSource { + public Hashtable getAnonymousEnv() { + Hashtable anonymousEnv = super.getAnonymousEnv(); + anonymousEnv.put("java.naming.security.protocol", "ssl"); + anonymousEnv.put("java.naming.ldap.factory.socket", CustomSSLSocketFactory.class.getName()); + anonymousEnv.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.ldap.LdapCtxFactory"); + return anonymousEnv; + } +} \ No newline at end of file From c16bc0ca8e064b96cad6d3cc3670f5ef6b21a54a Mon Sep 17 00:00:00 2001 From: "Captain.B" Date: Tue, 27 Oct 2020 17:53:48 +0800 Subject: [PATCH 7/7] =?UTF-8?q?feat(LDAP):=20=E6=94=AF=E6=8C=81LDAPS,=20?= =?UTF-8?q?=E5=8C=BA=E5=88=86ldap=E5=92=8Cldaps?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../metersphere/ldap/service/LdapService.java | 19 ++++++++++++------- 1 file changed, 12 insertions(+), 7 deletions(-) diff --git a/backend/src/main/java/io/metersphere/ldap/service/LdapService.java b/backend/src/main/java/io/metersphere/ldap/service/LdapService.java index 60a1ce0145..44aa6bc84a 100644 --- a/backend/src/main/java/io/metersphere/ldap/service/LdapService.java +++ b/backend/src/main/java/io/metersphere/ldap/service/LdapService.java @@ -18,6 +18,7 @@ import org.springframework.ldap.core.DirContextOperations; import org.springframework.ldap.core.LdapTemplate; import org.springframework.ldap.core.support.AbstractContextMapper; import org.springframework.ldap.core.support.DefaultDirObjectFactory; +import org.springframework.ldap.core.support.LdapContextSource; import org.springframework.ldap.query.SearchScope; import org.springframework.stereotype.Service; @@ -145,17 +146,21 @@ public class LdapService { preConnect(url, dn, password); String credentials = EncryptUtils.aesDecrypt(password).toString(); - - SSLLdapContextSource sourceLdapCtx = new SSLLdapContextSource(); - sourceLdapCtx.setUrl(url); - sourceLdapCtx.setUserDn(dn); - sourceLdapCtx.setPassword(credentials); - sourceLdapCtx.setDirObjectFactory(DefaultDirObjectFactory.class); - // todo 这里加上strategy 会报错 + LdapContextSource sourceLdapCtx; + if (StringUtils.startsWith(url, "ldaps://")) { + sourceLdapCtx = new SSLLdapContextSource(); + // todo 这里加上strategy 会报错 // DefaultTlsDirContextAuthenticationStrategy strategy = new DefaultTlsDirContextAuthenticationStrategy(); // strategy.setShutdownTlsGracefully(true); // strategy.setHostnameVerifier((hostname, session) -> true); // sourceLdapCtx.setAuthenticationStrategy(strategy); + } else { + sourceLdapCtx = new LdapContextSource(); + } + sourceLdapCtx.setUrl(url); + sourceLdapCtx.setUserDn(dn); + sourceLdapCtx.setPassword(credentials); + sourceLdapCtx.setDirObjectFactory(DefaultDirObjectFactory.class); sourceLdapCtx.afterPropertiesSet(); LdapTemplate ldapTemplate = new LdapTemplate(sourceLdapCtx); ldapTemplate.setIgnorePartialResultException(true);