123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252 |
- /*
- * SonarQube, open source software quality management tool.
- * Copyright (C) 2008-2014 SonarSource
- * mailto:contact AT sonarsource DOT com
- *
- * SonarQube is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 3 of the License, or (at your option) any later version.
- *
- * SonarQube is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public License
- * along with this program; if not, write to the Free Software Foundation,
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
-
- package org.sonar.server.computation.step;
-
- import com.google.common.base.Joiner;
- import com.google.common.collect.ArrayListMultimap;
- import com.google.common.collect.HashBasedTable;
- import com.google.common.collect.ImmutableMap;
- import com.google.common.collect.Multimap;
- import com.google.common.collect.Table;
- import java.util.ArrayList;
- import java.util.Collection;
- import java.util.HashMap;
- import java.util.HashSet;
- import java.util.List;
- import java.util.Map;
- import java.util.Set;
- import org.apache.ibatis.session.ResultContext;
- import org.apache.ibatis.session.ResultHandler;
- import org.sonar.api.utils.System2;
- import org.sonar.api.utils.internal.Uuids;
- import org.sonar.api.utils.log.Logger;
- import org.sonar.api.utils.log.Loggers;
- import org.sonar.batch.protocol.output.BatchReport;
- import org.sonar.db.DbSession;
- import org.sonar.db.MyBatis;
- import org.sonar.db.source.FileSourceDto;
- import org.sonar.db.source.FileSourceDto.Type;
- import org.sonar.server.computation.batch.BatchReportReader;
- import org.sonar.server.computation.component.Component;
- import org.sonar.server.computation.component.DepthTraversalTypeAwareVisitor;
- import org.sonar.server.computation.component.TreeRootHolder;
- import org.sonar.server.db.DbClient;
- import org.sonar.server.source.db.FileSourceDb;
- import org.sonar.server.source.db.FileSourceDb.Test.TestStatus;
- import org.sonar.server.util.CloseableIterator;
-
- public class PersistTestsStep implements ComputationStep {
-
- private static final Logger LOG = Loggers.get(PersistTestsStep.class);
-
- private final DbClient dbClient;
- private final System2 system;
- private final BatchReportReader reportReader;
- private final TreeRootHolder treeRootHolder;
-
- public PersistTestsStep(DbClient dbClient, System2 system, BatchReportReader reportReader, TreeRootHolder treeRootHolder) {
- this.dbClient = dbClient;
- this.system = system;
- this.reportReader = reportReader;
- this.treeRootHolder = treeRootHolder;
- }
-
- @Override
- public void execute() {
- DbSession session = dbClient.openSession(true);
- try {
- TestDepthTraversalTypeAwareVisitor visitor = new TestDepthTraversalTypeAwareVisitor(session);
- visitor.visit(treeRootHolder.getRoot());
- session.commit();
- if (visitor.hasUnprocessedCoverageDetails) {
- LOG.warn("Some coverage tests are not taken into account during analysis of project '{}'", visitor.getProjectKey());
- }
- } finally {
- MyBatis.closeQuietly(session);
- }
- }
-
- @Override
- public String getDescription() {
- return "Persist tests";
- }
-
- private class TestDepthTraversalTypeAwareVisitor extends DepthTraversalTypeAwareVisitor {
- final DbSession session;
- final Map<String, FileSourceDto> existingFileSourcesByUuid;
- final String projectUuid;
- final String projectKey;
- boolean hasUnprocessedCoverageDetails = false;
-
- public TestDepthTraversalTypeAwareVisitor(DbSession session) {
- super(Component.Type.FILE, Order.PRE_ORDER);
- this.session = session;
- this.existingFileSourcesByUuid = new HashMap<>();
- this.projectUuid = treeRootHolder.getRoot().getUuid();
- this.projectKey = treeRootHolder.getRoot().getKey();
- session.select("org.sonar.db.source.FileSourceMapper.selectHashesForProject",
- ImmutableMap.of("projectUuid", treeRootHolder.getRoot().getUuid(), "dataType", Type.TEST),
- new ResultHandler() {
- @Override
- public void handleResult(ResultContext context) {
- FileSourceDto dto = (FileSourceDto) context.getResultObject();
- existingFileSourcesByUuid.put(dto.getFileUuid(), dto);
- }
- });
- }
-
- @Override
- public void visitFile(Component file) {
- if (file.getFileAttributes().isUnitTest()) {
- persistTestResults(file);
- }
- }
-
- private void persistTestResults(Component component) {
- Multimap<String, FileSourceDb.Test.Builder> testsByName = buildDbTests(component.getRef());
- Table<String, String, FileSourceDb.Test.CoveredFile.Builder> coveredFilesByName = loadCoverageDetails(component.getRef());
- List<FileSourceDb.Test> tests = addCoveredFilesToTests(testsByName, coveredFilesByName);
- if (checkIfThereAreUnprocessedCoverageDetails(testsByName, coveredFilesByName, component.getKey())) {
- hasUnprocessedCoverageDetails = true;
- }
-
- if (tests.isEmpty()) {
- return;
- }
-
- String componentUuid = getUuid(component.getRef());
- FileSourceDto existingDto = existingFileSourcesByUuid.get(componentUuid);
- long now = system.now();
- if (existingDto != null) {
- // update
- existingDto
- .setTestData(tests)
- .setUpdatedAt(now);
- dbClient.fileSourceDao().update(session, existingDto);
- } else {
- // insert
- FileSourceDto newDto = new FileSourceDto()
- .setTestData(tests)
- .setFileUuid(componentUuid)
- .setProjectUuid(projectUuid)
- .setDataType(Type.TEST)
- .setCreatedAt(now)
- .setUpdatedAt(now);
- dbClient.fileSourceDao().insert(session, newDto);
- }
- }
-
- private boolean checkIfThereAreUnprocessedCoverageDetails(Multimap<String, FileSourceDb.Test.Builder> testsByName,
- Table<String, String, FileSourceDb.Test.CoveredFile.Builder> coveredFilesByName, String componentKey) {
- Set<String> unprocessedCoverageDetailNames = new HashSet<>(coveredFilesByName.rowKeySet());
- unprocessedCoverageDetailNames.removeAll(testsByName.keySet());
- boolean hasUnprocessedCoverage = !unprocessedCoverageDetailNames.isEmpty();
- if (hasUnprocessedCoverage) {
- LOG.trace("The following test coverages for file '{}' have not been taken into account: {}", componentKey, Joiner.on(", ").join(unprocessedCoverageDetailNames));
- }
- return hasUnprocessedCoverage;
- }
-
- private List<FileSourceDb.Test> addCoveredFilesToTests(Multimap<String, FileSourceDb.Test.Builder> testsByName,
- Table<String, String, FileSourceDb.Test.CoveredFile.Builder> coveredFilesByName) {
- List<FileSourceDb.Test> tests = new ArrayList<>();
- for (FileSourceDb.Test.Builder test : testsByName.values()) {
- Collection<FileSourceDb.Test.CoveredFile.Builder> coveredFiles = coveredFilesByName.row(test.getName()).values();
- if (!coveredFiles.isEmpty()) {
- for (FileSourceDb.Test.CoveredFile.Builder coveredFile : coveredFiles) {
- test.addCoveredFile(coveredFile);
- }
- }
- tests.add(test.build());
- }
-
- return tests;
- }
-
- private Multimap<String, FileSourceDb.Test.Builder> buildDbTests(int componentRed) {
- Multimap<String, FileSourceDb.Test.Builder> tests = ArrayListMultimap.create();
-
- try (CloseableIterator<BatchReport.Test> testIterator = reportReader.readTests(componentRed)) {
- while (testIterator.hasNext()) {
- BatchReport.Test batchTest = testIterator.next();
- FileSourceDb.Test.Builder dbTest = FileSourceDb.Test.newBuilder();
- dbTest.setUuid(Uuids.create());
- dbTest.setName(batchTest.getName());
- if (batchTest.hasStacktrace()) {
- dbTest.setStacktrace(batchTest.getStacktrace());
- }
- if (batchTest.hasStatus()) {
- dbTest.setStatus(TestStatus.valueOf(batchTest.getStatus().name()));
- }
- if (batchTest.hasMsg()) {
- dbTest.setMsg(batchTest.getMsg());
- }
- if (batchTest.hasDurationInMs()) {
- dbTest.setExecutionTimeMs(batchTest.getDurationInMs());
- }
-
- tests.put(dbTest.getName(), dbTest);
- }
- }
-
- return tests;
- }
-
- /**
- * returns a Table of (test name, main file uuid, covered file)
- */
- private Table<String, String, FileSourceDb.Test.CoveredFile.Builder> loadCoverageDetails(int testFileRef) {
- Table<String, String, FileSourceDb.Test.CoveredFile.Builder> nameToCoveredFiles = HashBasedTable.create();
-
- try (CloseableIterator<BatchReport.CoverageDetail> coverageIterator = reportReader.readCoverageDetails(testFileRef)) {
- while (coverageIterator.hasNext()) {
- BatchReport.CoverageDetail batchCoverageDetail = coverageIterator.next();
- for (BatchReport.CoverageDetail.CoveredFile batchCoveredFile : batchCoverageDetail.getCoveredFileList()) {
- String testName = batchCoverageDetail.getTestName();
- String mainFileUuid = getUuid(batchCoveredFile.getFileRef());
- FileSourceDb.Test.CoveredFile.Builder existingDbCoveredFile = nameToCoveredFiles.get(testName, mainFileUuid);
- List<Integer> batchCoveredLines = batchCoveredFile.getCoveredLineList();
- if (existingDbCoveredFile == null) {
- FileSourceDb.Test.CoveredFile.Builder dbCoveredFile = FileSourceDb.Test.CoveredFile.newBuilder()
- .setFileUuid(getUuid(batchCoveredFile.getFileRef()))
- .addAllCoveredLine(batchCoveredLines);
- nameToCoveredFiles.put(testName, mainFileUuid, dbCoveredFile);
- } else {
- List<Integer> remainingBatchCoveredLines = new ArrayList<>(batchCoveredLines);
- remainingBatchCoveredLines.removeAll(existingDbCoveredFile.getCoveredLineList());
- existingDbCoveredFile.addAllCoveredLine(batchCoveredLines);
- }
- }
- }
- }
- return nameToCoveredFiles;
- }
-
- private String getUuid(int fileRef) {
- return treeRootHolder.getComponentByRef(fileRef).getUuid();
- }
-
- public String getProjectKey() {
- return projectKey;
- }
- }
-
- }
|