Skip to content

Commit

Permalink
SONAR-21259 Increase range of issue status tracked as issue fixed and…
Browse files Browse the repository at this point in the history
… refactor code
  • Loading branch information
leo-geoffroy-sonarsource authored and sonartech committed Jan 17, 2024
1 parent 529b1f2 commit fdd609d
Show file tree
Hide file tree
Showing 8 changed files with 78 additions and 37 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@
import org.sonar.core.issue.DefaultIssue;
import org.sonar.core.issue.FieldDiffs;
import org.sonar.core.issue.IssueChangeContext;
import org.sonar.core.issue.tracking.Input;
import org.sonar.core.issue.tracking.Tracker;
import org.sonar.db.DbClient;
import org.sonar.db.DbTester;
Expand Down Expand Up @@ -157,14 +158,14 @@ public void setUp() throws Exception {
ClosedIssuesInputFactory closedIssuesInputFactory = new ClosedIssuesInputFactory(issuesLoader, dbClient, movedFilesRepository);
TrackerExecution tracker = new TrackerExecution(baseInputFactory, closedIssuesInputFactory, new Tracker<>(), issuesLoader, analysisMetadataHolder);
ReferenceBranchTrackerExecution mergeBranchTracker = new ReferenceBranchTrackerExecution(mergeInputFactory, new Tracker<>());
PullRequestTrackerExecution prBranchTracker = new PullRequestTrackerExecution(baseInputFactory, targetInputFactory, new Tracker<>(), newLinesRepository);
PullRequestTrackerExecution prBranchTracker = new PullRequestTrackerExecution(baseInputFactory, new Tracker<>(), newLinesRepository);
IssueTrackingDelegator trackingDelegator = new IssueTrackingDelegator(prBranchTracker, mergeBranchTracker, tracker, analysisMetadataHolder);
treeRootHolder.setRoot(PROJECT);
protoIssueCache = new ProtoIssueCache(temp.newFile(), System2.INSTANCE);
when(issueFilter.accept(any(DefaultIssue.class), eq(FILE))).thenReturn(true);
when(issueChangeContext.date()).thenReturn(new Date());
underTest = new IntegrateIssuesVisitor(protoIssueCache, rawInputFactory, baseInputFactory, issueLifecycle, issueVisitors, trackingDelegator, issueStatusCopier,
referenceBranchComponentUuids, mock(PullRequestSourceBranchMerger.class), fileStatuses, analysisMetadataHolder);
referenceBranchComponentUuids, mock(PullRequestSourceBranchMerger.class), fileStatuses, analysisMetadataHolder, targetInputFactory);
}

@Test
Expand Down Expand Up @@ -229,6 +230,28 @@ public void execute_issue_visitors() {
assertThat(defaultIssueCaptor.getValue().ruleKey().rule()).isEqualTo("x1");
}

@Test
public void visitAny_whenIsPullRequest_shouldCallExpectedVisitorsRawIssues() {
when(analysisMetadataHolder.isPullRequest()).thenReturn(true);
when(targetBranchComponentUuids.hasTargetBranchAnalysis()).thenReturn(true);

ruleRepositoryRule.add(RuleTesting.XOO_X1);
ScannerReport.Issue reportIssue = getReportIssue(RuleTesting.XOO_X1);
reportReader.putIssues(FILE_REF, singletonList(reportIssue));

IssueDto otherBranchIssueDto = addBaseIssueOnBranch(RuleTesting.XOO_X1);
when(targetBranchComponentUuids.getTargetBranchComponentUuid(FILE.getKey())).thenReturn(otherBranchIssueDto.getComponentUuid());

underTest.visitAny(FILE);

ArgumentCaptor<Input<DefaultIssue>> targetInputCaptor = ArgumentCaptor.forClass(Input.class);
ArgumentCaptor<Input<DefaultIssue>> rawInputCaptor = ArgumentCaptor.forClass(Input.class);
verify(issueVisitor).onRawIssues(eq(FILE), rawInputCaptor.capture(), targetInputCaptor.capture());
assertThat(rawInputCaptor.getValue().getIssues()).extracting(i -> i.ruleKey().rule()).containsExactly("x1");
assertThat(targetInputCaptor.getValue().getIssues()).extracting(DefaultIssue::key).containsExactly(otherBranchIssueDto.getKee());

}

@Test
public void close_unmatched_base_issue() {
RuleKey ruleKey = RuleTesting.XOO_X1;
Expand Down Expand Up @@ -334,7 +357,7 @@ private void addBaseIssue(RuleKey ruleKey) {
dbTester.getSession().commit();
}

private void addBaseIssueOnBranch(RuleKey ruleKey) {
private IssueDto addBaseIssueOnBranch(RuleKey ruleKey) {
ComponentDto project = ComponentTesting.newPrivateProjectDto(PROJECT_UUID_ON_BRANCH).setKey(PROJECT_KEY);
ComponentDto file = ComponentTesting.newFileDto(project, null, FILE_UUID_ON_BRANCH).setKey(FILE_KEY);
dbTester.components().insertComponents(project, file);
Expand All @@ -349,6 +372,7 @@ private void addBaseIssueOnBranch(RuleKey ruleKey) {
.setChecksum(null);
dbTester.getDbClient().issueDao().insert(dbTester.getSession(), issue);
dbTester.getSession().commit();
return issue;
}

@NotNull
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,8 @@
import java.util.Objects;
import java.util.Optional;
import java.util.stream.Stream;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.sonar.api.rule.RuleKey;
import org.sonar.ce.task.projectanalysis.analysis.Analysis;
import org.sonar.ce.task.projectanalysis.analysis.AnalysisMetadataHolder;
Expand Down Expand Up @@ -54,6 +56,8 @@ public class IntegrateIssuesVisitor extends TypeAwareVisitorAdapter {
private final PullRequestSourceBranchMerger pullRequestSourceBranchMerger;
private final FileStatuses fileStatuses;
private final AnalysisMetadataHolder analysisMetadataHolder;
private final TrackerTargetBranchInputFactory targetInputFactory;


public IntegrateIssuesVisitor(
ProtoIssueCache protoIssueCache,
Expand All @@ -66,7 +70,9 @@ public IntegrateIssuesVisitor(
ReferenceBranchComponentUuids referenceBranchComponentUuids,
PullRequestSourceBranchMerger pullRequestSourceBranchMerger,
FileStatuses fileStatuses,
AnalysisMetadataHolder analysisMetadataHolder) {
AnalysisMetadataHolder analysisMetadataHolder,
TrackerTargetBranchInputFactory targetInputFactory
) {

super(CrawlerDepthLimit.FILE, POST_ORDER);
this.protoIssueCache = protoIssueCache;
Expand All @@ -80,16 +86,18 @@ public IntegrateIssuesVisitor(
this.pullRequestSourceBranchMerger = pullRequestSourceBranchMerger;
this.fileStatuses = fileStatuses;
this.analysisMetadataHolder = analysisMetadataHolder;
this.targetInputFactory = targetInputFactory;
}

@Override
public void visitAny(Component component) {
try (CacheAppender<DefaultIssue> cacheAppender = protoIssueCache.newAppender()) {
issueVisitors.beforeComponent(component);
Input<DefaultIssue> rawInput = rawInputFactory.create(component);
List<DefaultIssue> issues = getIssues(rawInput, component);
Input<DefaultIssue> targetInput = createTargetInputIfExist(component);
List<DefaultIssue> issues = getIssues(rawInput, targetInput, component);

issueVisitors.onRawIssues(component, rawInput);
issueVisitors.onRawIssues(component, rawInput, targetInput);
processIssues(component, issues);

issueVisitors.beforeCaching(component);
Expand All @@ -100,21 +108,29 @@ public void visitAny(Component component) {
}
}

private List<DefaultIssue> getIssues(Input<DefaultIssue> rawInput, Component component) {
@CheckForNull
private Input<DefaultIssue> createTargetInputIfExist(Component component) {
if (targetInputFactory.hasTargetBranchAnalysis()) {
return targetInputFactory.createForTargetBranch(component);
}
return null;
}

private List<DefaultIssue> getIssues(Input<DefaultIssue> rawInput, @Nullable Input<DefaultIssue> targetInput, Component component) {
if (fileStatuses.isDataUnchanged(component)) {
// we assume there's a previous analysis of the same branch
return getIssuesForUnchangedFile(rawInput, component);
return getIssuesForUnchangedFile(rawInput, targetInput, component);
} else {
return getRawIssues(rawInput, component);
return getRawIssues(rawInput, targetInput, component);
}
}

private List<DefaultIssue> getIssuesForUnchangedFile(Input<DefaultIssue> rawInput, Component component) {
private List<DefaultIssue> getIssuesForUnchangedFile(Input<DefaultIssue> rawInput, @Nullable Input<DefaultIssue> targetInput, Component component) {
Input<DefaultIssue> baseIssues = baseInputFactory.create(component);
Collection<DefaultIssue> issues = baseIssues.getIssues();
//In case of plugin update, issue impacts are potentially updated. We want to avoid incremental analysis in this case.
return hasAnyInvolvedPluginChangedSinceLastAnalysis(issues)
? getRawIssues(rawInput, component)
? getRawIssues(rawInput, targetInput, component)
: new LinkedList<>(issues);
}

Expand All @@ -134,8 +150,8 @@ private List<ScannerPlugin> getInvolvedPlugins(Collection<DefaultIssue> issues)
.toList();
}

private List<DefaultIssue> getRawIssues(Input<DefaultIssue> rawInput, Component component) {
TrackingResult tracking = issueTracking.track(component, rawInput);
private List<DefaultIssue> getRawIssues(Input<DefaultIssue> rawInput, @Nullable Input<DefaultIssue> targetInput, Component component) {
TrackingResult tracking = issueTracking.track(component, rawInput, targetInput);
var newOpenIssues = fillNewOpenIssues(component, tracking.newIssues(), rawInput);
var existingOpenIssues = fillExistingOpenIssues(tracking.issuesToMerge());
var closedIssues = closeIssues(tracking.issuesToClose());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
*/
package org.sonar.ce.task.projectanalysis.issue;

import javax.annotation.Nullable;
import org.sonar.ce.task.projectanalysis.analysis.AnalysisMetadataHolder;
import org.sonar.ce.task.projectanalysis.component.Component;
import org.sonar.core.issue.DefaultIssue;
Expand All @@ -42,9 +43,9 @@ public IssueTrackingDelegator(PullRequestTrackerExecution pullRequestTracker, Re
this.analysisMetadataHolder = analysisMetadataHolder;
}

public TrackingResult track(Component component, Input<DefaultIssue> rawInput) {
public TrackingResult track(Component component, Input<DefaultIssue> rawInput, @Nullable Input<DefaultIssue> targetInput) {
if (analysisMetadataHolder.isPullRequest()) {
return standardResult(pullRequestTracker.track(component, rawInput));
return standardResult(pullRequestTracker.track(component, rawInput, targetInput));
}

if (isFirstAnalysisSecondaryBranch()) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
*/
package org.sonar.ce.task.projectanalysis.issue;

import javax.annotation.Nullable;
import org.sonar.ce.task.projectanalysis.component.Component;
import org.sonar.core.issue.DefaultIssue;
import org.sonar.core.issue.tracking.Input;
Expand Down Expand Up @@ -48,7 +49,7 @@ public void onIssue(Component component, DefaultIssue issue) {
/**
* This method is called for all raw issues of a component before tracking is done.
*/
public void onRawIssues(Component component, Input<DefaultIssue> rawIssues) {
public void onRawIssues(Component component, Input<DefaultIssue> rawIssues, @Nullable Input<DefaultIssue> baseIssues) {

}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
*/
package org.sonar.ce.task.projectanalysis.issue;

import javax.annotation.Nullable;
import org.sonar.ce.task.projectanalysis.component.Component;
import org.sonar.core.issue.DefaultIssue;
import org.sonar.core.issue.tracking.Input;
Expand Down Expand Up @@ -55,9 +56,9 @@ public void beforeCaching(Component component) {
}
}

public void onRawIssues(Component component, Input<DefaultIssue> rawIssues) {
public void onRawIssues(Component component, Input<DefaultIssue> rawIssues, @Nullable Input<DefaultIssue> targetIssues) {
for (IssueVisitor visitor : visitors) {
visitor.onRawIssues(component, rawIssues);
visitor.onRawIssues(component, rawIssues, targetIssues);
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
import java.util.List;
import java.util.Optional;
import java.util.Set;
import javax.annotation.Nullable;
import org.sonar.api.issue.Issue;
import org.sonar.ce.task.projectanalysis.component.Component;
import org.sonar.ce.task.projectanalysis.source.NewLinesRepository;
Expand All @@ -36,25 +37,22 @@ public class PullRequestTrackerExecution {
private final TrackerBaseInputFactory baseInputFactory;
private final Tracker<DefaultIssue, DefaultIssue> tracker;
private final NewLinesRepository newLinesRepository;
private final TrackerTargetBranchInputFactory targetInputFactory;

public PullRequestTrackerExecution(TrackerBaseInputFactory baseInputFactory, TrackerTargetBranchInputFactory targetInputFactory,
public PullRequestTrackerExecution(TrackerBaseInputFactory baseInputFactory,
Tracker<DefaultIssue, DefaultIssue> tracker, NewLinesRepository newLinesRepository) {
this.baseInputFactory = baseInputFactory;
this.targetInputFactory = targetInputFactory;
this.tracker = tracker;
this.newLinesRepository = newLinesRepository;
}

public Tracking<DefaultIssue, DefaultIssue> track(Component component, Input<DefaultIssue> rawInput) {
public Tracking<DefaultIssue, DefaultIssue> track(Component component, Input<DefaultIssue> rawInput, @Nullable Input<DefaultIssue> targetInput) {
// Step 1: only keep issues on changed lines
List<DefaultIssue> filteredRaws = keepIssuesHavingAtLeastOneLocationOnChangedLines(component, rawInput.getIssues());
Input<DefaultIssue> unmatchedRawsAfterChangedLineFiltering = createInput(rawInput, filteredRaws);

// Step 2: remove issues that are resolved in the target branch
Input<DefaultIssue> unmatchedRawsAfterTargetResolvedTracking;
if (targetInputFactory.hasTargetBranchAnalysis()) {
Input<DefaultIssue> targetInput = targetInputFactory.createForTargetBranch(component);
if (targetInput != null) {
List<DefaultIssue> resolvedTargetIssues = targetInput.getIssues().stream().filter(i -> Issue.STATUS_RESOLVED.equals(i.status())).toList();
Input<DefaultIssue> resolvedTargetInput = createInput(targetInput, resolvedTargetIssues);
Tracking<DefaultIssue, DefaultIssue> prResolvedTracking = tracker.trackNonClosed(unmatchedRawsAfterChangedLineFiltering, resolvedTargetInput);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,9 @@ public class IssueTrackingDelegatorTest {
@Mock
private Input<DefaultIssue> rawInput;

@Mock
private Input<DefaultIssue> targetInput;

private IssueTrackingDelegator underTest;

@Before
Expand All @@ -60,14 +63,14 @@ public void setUp() {
underTest = new IssueTrackingDelegator(prBranchTracker, mergeBranchTracker, tracker, analysisMetadataHolder);
when(tracker.track(component, rawInput)).thenReturn(trackingResult);
when(mergeBranchTracker.track(component, rawInput)).thenReturn(trackingResult);
when(prBranchTracker.track(component, rawInput)).thenReturn(trackingResult);
when(prBranchTracker.track(component, rawInput, targetInput)).thenReturn(trackingResult);
}

@Test
public void delegate_regular_tracker() {
when(analysisMetadataHolder.getBranch()).thenReturn(mock(Branch.class));

underTest.track(component, rawInput);
underTest.track(component, rawInput, targetInput);

verify(tracker).track(component, rawInput);
verifyNoInteractions(prBranchTracker);
Expand All @@ -82,7 +85,7 @@ public void delegate_merge_tracker() {
when(analysisMetadataHolder.getBranch()).thenReturn(branch);
when(analysisMetadataHolder.isFirstAnalysis()).thenReturn(true);

underTest.track(component, rawInput);
underTest.track(component, rawInput, targetInput);

verify(mergeBranchTracker).track(component, rawInput);
verifyNoInteractions(tracker);
Expand All @@ -97,9 +100,9 @@ public void delegate_pull_request_tracker() {
when(analysisMetadataHolder.getBranch()).thenReturn(mock(Branch.class));
when(analysisMetadataHolder.isPullRequest()).thenReturn(true);

underTest.track(component, rawInput);
underTest.track(component, rawInput, targetInput);

verify(prBranchTracker).track(component, rawInput);
verify(prBranchTracker).track(component, rawInput, targetInput);
verifyNoInteractions(tracker);
verifyNoInteractions(mergeBranchTracker);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -65,15 +65,13 @@ public class PullRequestTrackerExecutionTest {

private PullRequestTrackerExecution underTest;

private TrackerTargetBranchInputFactory targetFactory = mock(TrackerTargetBranchInputFactory.class);

@Before
public void setUp() {
when(baseFactory.create(FILE)).thenReturn(createInput(baseIssues));
when(targetFactory.createForTargetBranch(FILE)).thenReturn(createInput(targetIssues));

Tracker<DefaultIssue, DefaultIssue> tracker = new Tracker<>();
underTest = new PullRequestTrackerExecution(baseFactory, targetFactory, tracker, newLinesRepository);
underTest = new PullRequestTrackerExecution(baseFactory, tracker, newLinesRepository);
}

@Test
Expand All @@ -84,7 +82,7 @@ public void simple_tracking_keep_only_issues_having_location_on_changed_lines()

when(newLinesRepository.getNewLines(FILE)).thenReturn(Optional.of(new HashSet<>(Arrays.asList(1, 3))));

Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues));
Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues), createInput(targetIssues));

assertThat(tracking.getUnmatchedBases()).isEmpty();
assertThat(tracking.getMatchedRaws()).isEmpty();
Expand Down Expand Up @@ -120,7 +118,7 @@ public void simple_tracking_keep_also_issues_having_secondary_locations_on_chang

when(newLinesRepository.getNewLines(FILE)).thenReturn(Optional.of(new HashSet<>(Arrays.asList(7, 10))));

Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues));
Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues), createInput(targetIssues));

assertThat(tracking.getUnmatchedBases()).isEmpty();
assertThat(tracking.getMatchedRaws()).isEmpty();
Expand All @@ -135,7 +133,6 @@ public void track_and_ignore_resolved_issues_from_target_branch() {
rawIssues.add(createIssue(2, RuleTesting.XOO_X2));
rawIssues.add(createIssue(3, RuleTesting.XOO_X3));

when(targetFactory.hasTargetBranchAnalysis()).thenReturn(true);
DefaultIssue resolvedIssue = createIssue(1, RuleTesting.XOO_X1).setStatus(Issue.STATUS_RESOLVED).setResolution(Issue.RESOLUTION_FALSE_POSITIVE);
// will cause rawIssue0 to be ignored
targetIssues.add(resolvedIssue);
Expand All @@ -146,7 +143,7 @@ public void track_and_ignore_resolved_issues_from_target_branch() {
// should be matched
baseIssues.add(rawIssues.get(1));

Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues));
Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues), createInput(targetIssues));
assertThat(tracking.getMatchedRaws()).isEqualTo(Collections.singletonMap(rawIssues.get(1), rawIssues.get(1)));
assertThat(tracking.getUnmatchedRaws()).containsOnly(rawIssues.get(2));
}
Expand All @@ -160,7 +157,7 @@ public void track_and_ignore_issues_from_previous_analysis() {
rawIssues.add(createIssue(3, RuleTesting.XOO_X3));
baseIssues.add(rawIssues.get(0));

Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues));
Tracking<DefaultIssue, DefaultIssue> tracking = underTest.track(FILE, createInput(rawIssues), createInput(targetIssues));
assertThat(tracking.getMatchedRaws()).isEqualTo(Collections.singletonMap(rawIssues.get(0), rawIssues.get(0)));
assertThat(tracking.getUnmatchedRaws()).containsOnly(rawIssues.get(2));
}
Expand Down

0 comments on commit fdd609d

Please sign in to comment.