]> source.dussan.org Git - sonarqube.git/blob
0734c07c9fbec97d512dfe9a2b922b093a84b81c
[sonarqube.git] /
1 /*
2  * SonarQube
3  * Copyright (C) 2009-2017 SonarSource SA
4  * mailto:info AT sonarsource DOT com
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 3 of the License, or (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public License
17  * along with this program; if not, write to the Free Software Foundation,
18  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
19  */
20 package org.sonar.server.computation.task.projectanalysis.step;
21
22 import com.google.common.collect.ImmutableMap;
23 import java.util.List;
24 import java.util.Locale;
25 import java.util.Map;
26 import java.util.Optional;
27 import java.util.Set;
28 import org.sonar.api.i18n.I18n;
29 import org.sonar.db.DbClient;
30 import org.sonar.db.DbSession;
31 import org.sonar.db.component.ComponentLinkDto;
32 import org.sonar.scanner.protocol.output.ScannerReport;
33 import org.sonar.scanner.protocol.output.ScannerReport.ComponentLink.ComponentLinkType;
34 import org.sonar.server.computation.task.projectanalysis.batch.BatchReportReader;
35 import org.sonar.server.computation.task.projectanalysis.component.Component;
36 import org.sonar.server.computation.task.projectanalysis.component.CrawlerDepthLimit;
37 import org.sonar.server.computation.task.projectanalysis.component.DepthTraversalTypeAwareCrawler;
38 import org.sonar.server.computation.task.projectanalysis.component.TreeRootHolder;
39 import org.sonar.server.computation.task.projectanalysis.component.TypeAwareVisitorAdapter;
40 import org.sonar.server.computation.task.step.ComputationStep;
41
42 import static com.google.common.collect.Sets.newHashSet;
43 import static org.sonar.server.computation.task.projectanalysis.component.ComponentVisitor.Order.PRE_ORDER;
44
45 /**
46  * Persist project and module links
47  */
48 public class PersistProjectLinksStep implements ComputationStep {
49
50   private final DbClient dbClient;
51   private final I18n i18n;
52   private final TreeRootHolder treeRootHolder;
53   private final BatchReportReader reportReader;
54
55   private static final Map<ComponentLinkType, String> typesConverter = ImmutableMap.of(
56     ComponentLinkType.HOME, ComponentLinkDto.TYPE_HOME_PAGE,
57     ComponentLinkType.SCM, ComponentLinkDto.TYPE_SOURCES,
58     ComponentLinkType.SCM_DEV, ComponentLinkDto.TYPE_SOURCES_DEV,
59     ComponentLinkType.CI, ComponentLinkDto.TYPE_CI,
60     ComponentLinkType.ISSUE, ComponentLinkDto.TYPE_ISSUE_TRACKER);
61
62   public PersistProjectLinksStep(DbClient dbClient, I18n i18n, TreeRootHolder treeRootHolder, BatchReportReader reportReader) {
63     this.dbClient = dbClient;
64     this.i18n = i18n;
65     this.treeRootHolder = treeRootHolder;
66     this.reportReader = reportReader;
67   }
68
69   @Override
70   public void execute() {
71     try (DbSession session = dbClient.openSession(false)) {
72       new DepthTraversalTypeAwareCrawler(new ProjectLinkVisitor(session))
73         .visit(treeRootHolder.getRoot());
74       session.commit();
75     }
76   }
77
78   private class ProjectLinkVisitor extends TypeAwareVisitorAdapter {
79
80     private final DbSession session;
81
82     private ProjectLinkVisitor(DbSession session) {
83       super(CrawlerDepthLimit.FILE, PRE_ORDER);
84       this.session = session;
85     }
86
87     @Override
88     public void visitProject(Component project) {
89       processComponent(project);
90     }
91
92     @Override
93     public void visitModule(Component module) {
94       processComponent(module);
95     }
96
97     private void processComponent(Component component) {
98       ScannerReport.Component batchComponent = reportReader.readComponent(component.getReportAttributes().getRef());
99       processLinks(component.getUuid(), batchComponent.getLinkList());
100     }
101
102     private void processLinks(String componentUuid, List<ScannerReport.ComponentLink> links) {
103       List<ComponentLinkDto> previousLinks = dbClient.componentLinkDao().selectByComponentUuid(session, componentUuid);
104       mergeLinks(session, componentUuid, links, previousLinks);
105     }
106
107     private void mergeLinks(DbSession session, String componentUuid, List<ScannerReport.ComponentLink> links, List<ComponentLinkDto> previousLinks) {
108       Set<String> linkType = newHashSet();
109       for (final ScannerReport.ComponentLink link : links) {
110         String type = convertType(link.getType());
111         if (!linkType.contains(type)) {
112           linkType.add(type);
113         } else {
114           throw new IllegalArgumentException(String.format("Link of type '%s' has already been declared on component '%s'", type, componentUuid));
115         }
116
117         Optional<ComponentLinkDto> previousLink = previousLinks.stream()
118           .filter(input -> input != null && input.getType().equals(convertType(link.getType())))
119           .findFirst();
120         if (previousLink.isPresent()) {
121           previousLink.get().setHref(link.getHref());
122           dbClient.componentLinkDao().update(session, previousLink.get());
123         } else {
124           dbClient.componentLinkDao().insert(session,
125             new ComponentLinkDto()
126               .setComponentUuid(componentUuid)
127               .setType(type)
128               .setName(i18n.message(Locale.ENGLISH, "project_links." + type, null))
129               .setHref(link.getHref()));
130         }
131       }
132
133       for (ComponentLinkDto dto : previousLinks) {
134         if (!linkType.contains(dto.getType()) && ComponentLinkDto.PROVIDED_TYPES.contains(dto.getType())) {
135           dbClient.componentLinkDao().delete(session, dto.getId());
136         }
137       }
138     }
139
140     private String convertType(ComponentLinkType reportType) {
141       String type = typesConverter.get(reportType);
142       if (type != null) {
143         return type;
144       } else {
145         throw new IllegalArgumentException(String.format("Unsupported type %s", reportType.name()));
146       }
147     }
148   }
149
150   @Override
151   public String getDescription() {
152     return "Persist project links";
153   }
154 }