Skip to content
This repository has been archived by the owner on Jan 7, 2021. It is now read-only.

Commit

Permalink
implement resetting of overview comments
Browse files Browse the repository at this point in the history
  • Loading branch information
t-8ch committed Nov 3, 2017
1 parent ab72441 commit 7d50a97
Show file tree
Hide file tree
Showing 6 changed files with 113 additions and 57 deletions.
6 changes: 6 additions & 0 deletions src/main/java/org/sonar/plugins/stash/StashPluginUtils.java
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,9 @@
import com.google.common.base.CharMatcher;
import java.io.IOException;
import java.util.Collection;
import java.util.Optional;
import java.util.Properties;
import java.util.stream.Stream;
import org.sonar.api.batch.fs.InputComponent;
import org.sonar.api.batch.fs.InputModule;
import org.sonar.api.batch.postjob.issue.PostJobIssue;
Expand Down Expand Up @@ -72,4 +74,8 @@ public static String removeEnd(String s, String suffix) {
}
return s;
}

public static <T> Stream<T> removeEmpty(Optional<T> v) {
return v.map(Stream::of).orElse(Stream.empty());
}
}
52 changes: 27 additions & 25 deletions src/main/java/org/sonar/plugins/stash/StashRequestFacade.java
Original file line number Diff line number Diff line change
Expand Up @@ -398,39 +398,41 @@ public void resetComments(PullRequestRef pr,
StashUser sonarUser,
StashClient stashClient) {
try {
// Let's call this "diffRep_loop"
for (StashComment comment : diffReport.getComments()) {

// delete comment only if published by the current SQ user
if (sonarUser.getId() != comment.getAuthor().getId()) {
continue;
// Next element in "diffRep_loop"

// comment contains tasks which cannot be deleted => do nothing
} else if (comment.containsPermanentTasks()) {
LOGGER.debug("Comment \"{}\" (path:\"{}\", line:\"{}\")"
+ "CANNOT be deleted because one of its tasks is not deletable.", comment.getId(),
comment.getPath(),
comment.getLine());
continue; // Next element in "diffRep_loop"
}

// delete tasks linked to the current comment
for (StashTask task : comment.getTasks()) {
stashClient.deleteTaskOnComment(task);
}

stashClient.deletePullRequestComment(pr, comment);
}
// FIXME delete tasks on file-wide comments
// resetComments(diffReport.getComments(), pr, sonarUser, stashClient);
resetComments(stashClient.getPullRequestOverviewComments(pr), pr, sonarUser, stashClient);

LOGGER.info("SonarQube issues reported to Stash by user \"{}\" have been reset",
sonarUser.getName());

} catch (StashClientException e) {
LOGGER.error("Unable to reset comment list", e);
}
}

private void resetComments(Collection<StashComment> comments, PullRequestRef pr, StashUser sonarUser, StashClient stashClient)
throws StashClientException {
for (StashComment comment : comments) {
if (sonarUser.getId() != comment.getAuthor().getId()) {
continue;
}

if (comment.containsPermanentTasks()) {
LOGGER.debug("Comment \"{}\" (path:\"{}\", line:\"{}\")"
+ "CANNOT be deleted because one of its tasks is not deletable.", comment.getId(),
comment.getPath(),
comment.getLine());
continue; // Next element in "diffRep_loop"
}

// delete tasks linked to the current comment
for (StashTask task : comment.getTasks()) {
stashClient.deleteTaskOnComment(task);
}

stashClient.deletePullRequestComment(pr, comment);
}
}

@Override
public String getIssuePath(PostJobIssue issue) {
InputComponent ip = issue.inputComponent();
Expand Down
74 changes: 53 additions & 21 deletions src/main/java/org/sonar/plugins/stash/client/StashClient.java
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
package org.sonar.plugins.stash.client;

import java.util.Collection;
import java.util.Optional;
import java.util.stream.Collectors;
import org.asynchttpclient.AsyncHttpClient;
import org.asynchttpclient.BoundRequestBuilder;
import org.asynchttpclient.DefaultAsyncHttpClient;
Expand All @@ -19,7 +21,6 @@
import org.sonar.plugins.stash.PeekableInputStream;
import org.sonar.plugins.stash.PluginInfo;
import org.sonar.plugins.stash.PullRequestRef;
import org.sonar.plugins.stash.StashPlugin;
import org.sonar.plugins.stash.StashPlugin.IssueType;
import org.sonar.plugins.stash.StashPluginUtils;
import org.sonar.plugins.stash.exceptions.StashClientException;
Expand Down Expand Up @@ -64,7 +65,7 @@ public class StashClient implements AutoCloseable {
private static final String API_ONE_PR_ALL_COMMENTS = API_ONE_PR + "/comments";
private static final String API_ONE_PR_DIFF = API_ONE_PR + "/diff?withComments=true";
private static final String API_ONE_PR_APPROVAL = API_ONE_PR + "/approve";
private static final String API_ONE_PR_COMMENT_PATH = API_ONE_PR + "/comments?path={4}&start={5,number,#}";
private static final String API_ONE_PR_COMMENT_PATH = API_ONE_PR + "/comments?path={4}";

private static final String API_ONE_PR_ONE_COMMENT = API_ONE_PR_ALL_COMMENTS + "/{4}?version={5}";

Expand Down Expand Up @@ -111,31 +112,29 @@ public void postCommentOnPullRequest(PullRequestRef pr, String report)
postCreate(request, json, MessageFormat.format(COMMENT_POST_ERROR_MESSAGE, pr.repository(), pr.pullRequestId()));
}

public Collection<StashComment> getPullRequestOverviewComments(PullRequestRef pr) throws StashClientException {
return getPaged(
MessageFormat.format(API_ONE_PR + "/activities", baseUrl, pr.project(), pr.repository(), pr.pullRequestId()),
"Error!"
).stream().map(StashCollector::extractCommentFromActivity).flatMap(StashPluginUtils::removeEmpty).collect(Collectors.toList());
}

public StashCommentReport getPullRequestComments(PullRequestRef pr, String path)
throws StashClientException {
StashCommentReport result = new StashCommentReport();

long start = 0;
boolean isLastPage = false;
String url = MessageFormat.format(API_ONE_PR_COMMENT_PATH,
baseUrl,
pr.project(),
pr.repository(),
pr.pullRequestId(),
path
);

while (!isLastPage) {
for (JsonObject jsonComment: getPaged(url,
MessageFormat.format(COMMENT_GET_ERROR_MESSAGE, pr.repository(), pr.pullRequestId()))) {
try {
String request = MessageFormat.format(API_ONE_PR_COMMENT_PATH,
baseUrl,
pr.project(),
pr.repository(),
pr.pullRequestId(),
path,
start);
JsonObject jsonComments = get(request,
MessageFormat.format(COMMENT_GET_ERROR_MESSAGE,
pr.repository(),
pr.pullRequestId()));
result.add(StashCollector.extractComments(jsonComments));

// Stash pagination: check if you get all comments linked to the pull-request
isLastPage = StashCollector.isLastPage(jsonComments);
start = StashCollector.getNextPageStart(jsonComments);
result.add(StashCollector.extractComment(jsonComment));
} catch (StashReportExtractionException e) {
throw new StashClientException(e);
}
Expand Down Expand Up @@ -310,6 +309,10 @@ private JsonObject get(String url, String errorMessage) throws StashClientExcept
return performRequest(httpClient.prepareGet(url), null, HttpURLConnection.HTTP_OK, errorMessage);
}

private Collection<JsonObject> getPaged(String url, String errorMessage) throws StashClientException {
return performPagedRequest(httpClient.prepareGet(url), null, HttpURLConnection.HTTP_OK, errorMessage);
}

private JsonObject post(String url, JsonObject body, String errorMessage) throws StashClientException {
return performRequest(httpClient.preparePost(url), body, HttpURLConnection.HTTP_OK, errorMessage);
}
Expand Down Expand Up @@ -456,4 +459,33 @@ AsyncHttpClient createHttpClient(String sonarQubeVersion) {
.build()
);
}

private Collection<JsonObject> performPagedRequest(BoundRequestBuilder requestBuilder,
JsonObject body,
int expectedStatusCode,
String errorMessage) throws StashClientException {

Collection<JsonObject> result = new ArrayList<>();

boolean doneLastPage = false;
int nextPageStart = 0;
// FIXME size/limit support

while (!doneLastPage) {
if (nextPageStart > 0) {
requestBuilder.addQueryParam("start", String.valueOf(nextPageStart));
}

JsonObject res = performRequest(requestBuilder, body, expectedStatusCode, errorMessage);
JsonArray values = ((JsonArray) res.get("values"));
if (values == null) {
throw new StashClientException("Paged response did not include values");
}
values.asCollection(result);
doneLastPage = res.getBooleanOrDefault("isLastPage", true);
nextPageStart = res.getIntegerOrDefault("nextPageStart", -1);
}

return result;
}
}
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
package org.sonar.plugins.stash.issue.collector;

import java.math.BigDecimal;
import java.util.Objects;
import java.util.Optional;
import org.json.simple.JsonArray;
import org.json.simple.JsonObject;
import org.sonar.plugins.stash.PullRequestRef;
Expand All @@ -20,9 +22,7 @@ public final class StashCollector {
private static final String AUTHOR = "author";
private static final String VERSION = "version";

private StashCollector() {
// Hiding implicit public constructor with an explicit private one (squid:S1118)
}
private StashCollector() {}

public static StashCommentReport extractComments(JsonObject jsonComments) throws StashReportExtractionException {
StashCommentReport result = new StashCommentReport();
Expand All @@ -41,8 +41,11 @@ public static StashCommentReport extractComments(JsonObject jsonComments) throws
return result;
}

public static StashComment extractComment(JsonObject jsonComment, String path, Long line) {
public static Optional<StashComment> extractCommentFromActivity(JsonObject json) {
return Optional.ofNullable((JsonObject) json.get("comment")).filter(Objects::nonNull).map(j -> StashCollector.extractComment(j, null, null));
}

public static StashComment extractComment(JsonObject jsonComment, String path, Long line) {
long id = getLong(jsonComment, "id");
String message = (String)jsonComment.get("text");

Expand All @@ -51,7 +54,10 @@ public static StashComment extractComment(JsonObject jsonComment, String path, L
JsonObject jsonAuthor = (JsonObject)jsonComment.get(AUTHOR);
StashUser stashUser = extractUser(jsonAuthor);

return new StashComment(id, message, path, line, stashUser, version);
StashComment result = new StashComment(id, message, path, line, stashUser, version);
// FIXME do this at some central place
updateCommentTasks(result, (JsonArray) jsonComment.get("tasks"));
return result;
}

public static StashComment extractComment(JsonObject jsonComment) throws StashReportExtractionException {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -246,6 +246,7 @@ public void setUp() throws Exception {
comments.add(comment3);

when(diffReport.getComments()).thenReturn(comments);
when(stashClient.getPullRequestOverviewComments(pr)).thenReturn(comments);

stashCommentsReport1 = mock(StashCommentReport.class);
when(stashCommentsReport1.getComments()).thenReturn(comments);
Expand Down Expand Up @@ -713,6 +714,7 @@ public void testResetCommentsWithDifferentStashUsers() throws Exception {
comments.add(comment);

when(diffReport.getComments()).thenReturn(comments);
when(stashClient.getPullRequestOverviewComments(eq(pr))).thenReturn(comments);

myFacade.resetComments(pr, diffReport, stashUser, stashClient);

Expand All @@ -721,7 +723,8 @@ public void testResetCommentsWithDifferentStashUsers() throws Exception {

@Test
public void testResetCommentsWithoutAnyComments() throws Exception {
when(diffReport.getComments()).thenReturn(new ArrayList<StashComment>());
when(diffReport.getComments()).thenReturn(new ArrayList<>());
when(stashClient.getPullRequestOverviewComments(eq(pr))).thenReturn(new ArrayList<>());

myFacade.resetComments(pr, diffReport, stashUser, stashClient);

Expand Down
17 changes: 12 additions & 5 deletions src/test/java/org/sonar/plugins/stash/client/StashClientTest.java
Original file line number Diff line number Diff line change
Expand Up @@ -150,8 +150,7 @@ public void testGetPullRequestCommentsWithNextPage() throws Exception {
+ "\"author\": {\"id\":1, \"name\":\"SonarQube\", \"slug\":\"sonarqube\", \"email\":\"[email protected]\"}, \"version\": 0}], \"isLastPage\": true}";

wireMock.stubFor(get(
urlPathEqualTo("/rest/api/1.0/projects/Project/repos/Repository/pull-requests/1/comments"))
.withQueryParam("start", equalTo(String.valueOf(0))).willReturn(
urlPathEqualTo("/rest/api/1.0/projects/Project/repos/Repository/pull-requests/1/comments")).willReturn(
aJsonResponse().withStatus(HttpURLConnection.HTTP_OK).withBody(stashJsonComment1)
));

Expand All @@ -177,7 +176,7 @@ public void testGetPullRequestCommentsWithNoNextPage() throws Exception {
"{\"values\": [{\"id\":4321, \"text\":\"message2\", \"anchor\": {\"path\":\"path\", \"line\":10},"
+ "\"author\": {\"id\":1, \"name\":\"SonarQube\", \"slug\":\"sonarqube\", \"email\":\"[email protected]\"}, \"version\": 0}], \"isLastPage\": true}";

wireMock.stubFor(get(anyUrl()).withQueryParam("start", equalTo(String.valueOf(0))).willReturn(
wireMock.stubFor(get(anyUrl()).willReturn(
aJsonResponse().withStatus(HttpURLConnection.HTTP_OK).withBody(stashJsonComment1)));

wireMock.stubFor(get(anyUrl()).withQueryParam("start", equalTo(String.valueOf(1))).willReturn(
Expand Down Expand Up @@ -371,7 +370,7 @@ public void testPullRequestHugePullRequestId() throws Exception {
// See https://github.com/AmadeusITGroup/sonar-stash/issues/98
int hugePullRequestId = 1234567890;

wireMock.stubFor(any(anyUrl()).willReturn(aJsonResponse()));
wireMock.stubFor(any(anyUrl()).willReturn(aPagedJsonResponse()));

PullRequestRef pr = PullRequestRef.builder()
.setProject("Project")
Expand Down Expand Up @@ -424,7 +423,15 @@ private void addErrorResponse(MappingBuilder mapping, int statusCode) {
}

public static ResponseDefinitionBuilder aJsonResponse() {
return aResponse().withHeader("Content-Type", "application/json").withBody("{}");
return aJsonResponse("{}");
}

private ResponseDefinitionBuilder aPagedJsonResponse() {
return aJsonResponse("{\"values\":[]}");
}

public static ResponseDefinitionBuilder aJsonResponse(String body) {
return aResponse().withHeader("Content-Type", "application/json").withBody(body);
}

public static ResponseDefinitionBuilder aXMLResponse() {
Expand Down

0 comments on commit 7d50a97

Please sign in to comment.