You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

BulkChangeAction.java 21KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2020 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.issue.ws;
  21. import java.util.Collection;
  22. import java.util.Date;
  23. import java.util.HashMap;
  24. import java.util.HashSet;
  25. import java.util.List;
  26. import java.util.Map;
  27. import java.util.Objects;
  28. import java.util.Optional;
  29. import java.util.Set;
  30. import java.util.function.Consumer;
  31. import java.util.function.Predicate;
  32. import java.util.stream.Collectors;
  33. import javax.annotation.CheckForNull;
  34. import javax.annotation.Nullable;
  35. import org.sonar.api.issue.DefaultTransitions;
  36. import org.sonar.api.rule.RuleKey;
  37. import org.sonar.api.rule.Severity;
  38. import org.sonar.api.rules.RuleType;
  39. import org.sonar.api.server.ws.Change;
  40. import org.sonar.api.server.ws.Request;
  41. import org.sonar.api.server.ws.Response;
  42. import org.sonar.api.server.ws.WebService;
  43. import org.sonar.api.utils.System2;
  44. import org.sonar.api.utils.log.Logger;
  45. import org.sonar.api.utils.log.Loggers;
  46. import org.sonar.api.web.UserRole;
  47. import org.sonar.core.issue.DefaultIssue;
  48. import org.sonar.core.issue.IssueChangeContext;
  49. import org.sonar.core.util.stream.MoreCollectors;
  50. import org.sonar.db.DbClient;
  51. import org.sonar.db.DbSession;
  52. import org.sonar.db.component.BranchDto;
  53. import org.sonar.db.component.BranchType;
  54. import org.sonar.db.component.ComponentDto;
  55. import org.sonar.db.issue.IssueDto;
  56. import org.sonar.db.rule.RuleDefinitionDto;
  57. import org.sonar.db.user.UserDto;
  58. import org.sonar.server.issue.Action;
  59. import org.sonar.server.issue.ActionContext;
  60. import org.sonar.server.issue.AddTagsAction;
  61. import org.sonar.server.issue.AssignAction;
  62. import org.sonar.server.issue.IssueChangePostProcessor;
  63. import org.sonar.server.issue.RemoveTagsAction;
  64. import org.sonar.server.issue.WebIssueStorage;
  65. import org.sonar.server.issue.notification.IssuesChangesNotificationBuilder;
  66. import org.sonar.server.issue.notification.IssuesChangesNotificationBuilder.ChangedIssue;
  67. import org.sonar.server.issue.notification.IssuesChangesNotificationBuilder.Project;
  68. import org.sonar.server.issue.notification.IssuesChangesNotificationBuilder.User;
  69. import org.sonar.server.issue.notification.IssuesChangesNotificationBuilder.UserChange;
  70. import org.sonar.server.issue.notification.IssuesChangesNotificationSerializer;
  71. import org.sonar.server.notification.NotificationManager;
  72. import org.sonar.server.user.UserSession;
  73. import org.sonarqube.ws.Issues;
  74. import static com.google.common.base.Preconditions.checkArgument;
  75. import static com.google.common.base.Preconditions.checkState;
  76. import static com.google.common.collect.ImmutableMap.of;
  77. import static java.lang.String.format;
  78. import static java.util.Objects.requireNonNull;
  79. import static java.util.function.Function.identity;
  80. import static java.util.stream.Collectors.toMap;
  81. import static org.sonar.api.issue.DefaultTransitions.OPEN_AS_VULNERABILITY;
  82. import static org.sonar.api.issue.DefaultTransitions.REOPEN;
  83. import static org.sonar.api.issue.DefaultTransitions.RESOLVE_AS_REVIEWED;
  84. import static org.sonar.api.issue.DefaultTransitions.SET_AS_IN_REVIEW;
  85. import static org.sonar.api.rule.Severity.BLOCKER;
  86. import static org.sonar.api.rules.RuleType.BUG;
  87. import static org.sonar.core.util.Uuids.UUID_EXAMPLE_01;
  88. import static org.sonar.core.util.Uuids.UUID_EXAMPLE_02;
  89. import static org.sonar.core.util.stream.MoreCollectors.toSet;
  90. import static org.sonar.core.util.stream.MoreCollectors.uniqueIndex;
  91. import static org.sonar.server.es.SearchOptions.MAX_LIMIT;
  92. import static org.sonar.server.issue.AbstractChangeTagsAction.TAGS_PARAMETER;
  93. import static org.sonar.server.issue.AssignAction.ASSIGNEE_PARAMETER;
  94. import static org.sonar.server.issue.CommentAction.COMMENT_KEY;
  95. import static org.sonar.server.issue.CommentAction.COMMENT_PROPERTY;
  96. import static org.sonar.server.issue.SetSeverityAction.SET_SEVERITY_KEY;
  97. import static org.sonar.server.issue.SetSeverityAction.SEVERITY_PARAMETER;
  98. import static org.sonar.server.issue.SetTypeAction.SET_TYPE_KEY;
  99. import static org.sonar.server.issue.SetTypeAction.TYPE_PARAMETER;
  100. import static org.sonar.server.issue.TransitionAction.DO_TRANSITION_KEY;
  101. import static org.sonar.server.issue.TransitionAction.TRANSITION_PARAMETER;
  102. import static org.sonar.server.ws.WsUtils.writeProtobuf;
  103. import static org.sonarqube.ws.client.issue.IssuesWsParameters.ACTION_BULK_CHANGE;
  104. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_ADD_TAGS;
  105. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_ASSIGN;
  106. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_COMMENT;
  107. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_DO_TRANSITION;
  108. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_ISSUES;
  109. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_REMOVE_TAGS;
  110. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_SEND_NOTIFICATIONS;
  111. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_SET_SEVERITY;
  112. import static org.sonarqube.ws.client.issue.IssuesWsParameters.PARAM_SET_TYPE;
  113. public class BulkChangeAction implements IssuesWsAction {
  114. private static final Logger LOG = Loggers.get(BulkChangeAction.class);
  115. private final System2 system2;
  116. private final UserSession userSession;
  117. private final DbClient dbClient;
  118. private final WebIssueStorage issueStorage;
  119. private final NotificationManager notificationService;
  120. private final List<Action> actions;
  121. private final IssueChangePostProcessor issueChangePostProcessor;
  122. private final IssuesChangesNotificationSerializer notificationSerializer;
  123. public BulkChangeAction(System2 system2, UserSession userSession, DbClient dbClient, WebIssueStorage issueStorage,
  124. NotificationManager notificationService, List<Action> actions,
  125. IssueChangePostProcessor issueChangePostProcessor, IssuesChangesNotificationSerializer notificationSerializer) {
  126. this.system2 = system2;
  127. this.userSession = userSession;
  128. this.dbClient = dbClient;
  129. this.issueStorage = issueStorage;
  130. this.notificationService = notificationService;
  131. this.actions = actions;
  132. this.issueChangePostProcessor = issueChangePostProcessor;
  133. this.notificationSerializer = notificationSerializer;
  134. }
  135. @Override
  136. public void define(WebService.NewController context) {
  137. WebService.NewAction action = context.createAction(ACTION_BULK_CHANGE)
  138. .setDescription("Bulk change on issues.<br/>" +
  139. "Requires authentication.")
  140. .setSince("3.7")
  141. .setChangelog(
  142. new Change("8.1", format("transitions '%s', '%s' and '%s' are no more supported", SET_AS_IN_REVIEW, RESOLVE_AS_REVIEWED, OPEN_AS_VULNERABILITY)),
  143. new Change("6.3", "'actions' parameter is ignored"))
  144. .setHandler(this)
  145. .setResponseExample(getClass().getResource("bulk_change-example.json"))
  146. .setPost(true);
  147. action.createParam(PARAM_ISSUES)
  148. .setDescription("Comma-separated list of issue keys")
  149. .setRequired(true)
  150. .setExampleValue(UUID_EXAMPLE_01 + "," + UUID_EXAMPLE_02);
  151. action.createParam(PARAM_ASSIGN)
  152. .setDescription("To assign the list of issues to a specific user (login), or un-assign all the issues")
  153. .setExampleValue("john.smith")
  154. .setDeprecatedKey("assign.assignee", "6.2");
  155. action.createParam(PARAM_SET_SEVERITY)
  156. .setDescription("To change the severity of the list of issues")
  157. .setExampleValue(BLOCKER)
  158. .setPossibleValues(Severity.ALL)
  159. .setDeprecatedKey("set_severity.severity", "6.2");
  160. action.createParam(PARAM_SET_TYPE)
  161. .setDescription("To change the type of the list of issues")
  162. .setExampleValue(BUG)
  163. .setPossibleValues(RuleType.names())
  164. .setSince("5.5")
  165. .setDeprecatedKey("set_type.type", "6.2");
  166. action.createParam(PARAM_DO_TRANSITION)
  167. .setDescription("Transition")
  168. .setExampleValue(REOPEN)
  169. .setPossibleValues(DefaultTransitions.ALL)
  170. .setDeprecatedKey("do_transition.transition", "6.2");
  171. action.createParam(PARAM_ADD_TAGS)
  172. .setDescription("Add tags")
  173. .setExampleValue("security,java8")
  174. .setDeprecatedKey("add_tags.tags", "6.2");
  175. action.createParam(PARAM_REMOVE_TAGS)
  176. .setDescription("Remove tags")
  177. .setExampleValue("security,java8")
  178. .setDeprecatedKey("remove_tags.tags", "6.2");
  179. action.createParam(PARAM_COMMENT)
  180. .setDescription("To add a comment to a list of issues")
  181. .setExampleValue("Here is my comment");
  182. action.createParam(PARAM_SEND_NOTIFICATIONS)
  183. .setSince("4.0")
  184. .setBooleanPossibleValues()
  185. .setDefaultValue("false");
  186. }
  187. @Override
  188. public void handle(Request request, Response response) throws Exception {
  189. userSession.checkLoggedIn();
  190. try (DbSession dbSession = dbClient.openSession(false)) {
  191. BulkChangeResult result = executeBulkChange(dbSession, request);
  192. writeProtobuf(toWsResponse(result), request, response);
  193. }
  194. }
  195. private BulkChangeResult executeBulkChange(DbSession dbSession, Request request) {
  196. BulkChangeData bulkChangeData = new BulkChangeData(dbSession, request);
  197. BulkChangeResult result = new BulkChangeResult(bulkChangeData.issues.size());
  198. IssueChangeContext issueChangeContext = IssueChangeContext.createUser(new Date(system2.now()), userSession.getUuid());
  199. List<DefaultIssue> items = bulkChangeData.issues.stream()
  200. .filter(bulkChange(issueChangeContext, bulkChangeData, result))
  201. .collect(MoreCollectors.toList());
  202. issueStorage.save(dbSession, items);
  203. refreshLiveMeasures(dbSession, bulkChangeData, result);
  204. Set<String> assigneeUuids = items.stream().map(DefaultIssue::assignee).filter(Objects::nonNull).collect(Collectors.toSet());
  205. Map<String, UserDto> userDtoByUuid = dbClient.userDao().selectByUuids(dbSession, assigneeUuids).stream().collect(toMap(UserDto::getUuid, u -> u));
  206. String authorUuid = requireNonNull(userSession.getUuid(), "User uuid cannot be null");
  207. UserDto author = dbClient.userDao().selectByUuid(dbSession, authorUuid);
  208. checkState(author != null, "User with uuid '%s' does not exist");
  209. sendNotification(items, bulkChangeData, userDtoByUuid, author);
  210. return result;
  211. }
  212. private void refreshLiveMeasures(DbSession dbSession, BulkChangeData data, BulkChangeResult result) {
  213. if (!data.shouldRefreshMeasures()) {
  214. return;
  215. }
  216. Set<String> touchedComponentUuids = result.success.stream()
  217. .map(DefaultIssue::componentUuid)
  218. .collect(Collectors.toSet());
  219. List<ComponentDto> touchedComponents = touchedComponentUuids.stream()
  220. .map(data.componentsByUuid::get)
  221. .collect(MoreCollectors.toList(touchedComponentUuids.size()));
  222. List<DefaultIssue> changedIssues = data.issues.stream().filter(result.success::contains).collect(MoreCollectors.toList());
  223. issueChangePostProcessor.process(dbSession, changedIssues, touchedComponents);
  224. }
  225. private static Predicate<DefaultIssue> bulkChange(IssueChangeContext issueChangeContext, BulkChangeData bulkChangeData, BulkChangeResult result) {
  226. return issue -> {
  227. ActionContext actionContext = new ActionContext(issue, issueChangeContext, bulkChangeData.projectsByUuid.get(issue.projectUuid()));
  228. bulkChangeData.getActionsWithoutComment().forEach(applyAction(actionContext, bulkChangeData, result));
  229. addCommentIfNeeded(actionContext, bulkChangeData);
  230. return result.success.contains(issue);
  231. };
  232. }
  233. private static Consumer<Action> applyAction(ActionContext actionContext, BulkChangeData bulkChangeData, BulkChangeResult result) {
  234. return action -> {
  235. DefaultIssue issue = actionContext.issue();
  236. try {
  237. if (action.supports(issue) && action.execute(bulkChangeData.getProperties(action.key()), actionContext)) {
  238. result.increaseSuccess(issue);
  239. }
  240. } catch (Exception e) {
  241. result.increaseFailure();
  242. LOG.error(format("An error occur when trying to apply the action : %s on issue : %s. This issue has been ignored. Error is '%s'",
  243. action.key(), issue.key(), e.getMessage()), e);
  244. }
  245. };
  246. }
  247. private static void addCommentIfNeeded(ActionContext actionContext, BulkChangeData bulkChangeData) {
  248. bulkChangeData.getCommentAction().ifPresent(action -> action.execute(bulkChangeData.getProperties(action.key()), actionContext));
  249. }
  250. private void sendNotification(Collection<DefaultIssue> issues, BulkChangeData bulkChangeData, Map<String, UserDto> userDtoByUuid, UserDto author) {
  251. if (!bulkChangeData.sendNotification) {
  252. return;
  253. }
  254. Set<ChangedIssue> changedIssues = issues.stream()
  255. // should not happen but filter it out anyway to avoid NPE in oldestUpdateDate call below
  256. .filter(issue -> issue.updateDate() != null)
  257. .map(issue -> toNotification(bulkChangeData, userDtoByUuid, issue))
  258. .filter(Objects::nonNull)
  259. .collect(toSet(issues.size()));
  260. if (changedIssues.isEmpty()) {
  261. return;
  262. }
  263. IssuesChangesNotificationBuilder builder = new IssuesChangesNotificationBuilder(
  264. changedIssues,
  265. new UserChange(oldestUpdateDate(issues), new User(author.getUuid(), author.getLogin(), author.getName())));
  266. notificationService.scheduleForSending(notificationSerializer.serialize(builder));
  267. }
  268. @CheckForNull
  269. private ChangedIssue toNotification(BulkChangeData bulkChangeData, Map<String, UserDto> userDtoByUuid, DefaultIssue issue) {
  270. BranchDto branchDto = bulkChangeData.branchesByProjectUuid.get(issue.projectUuid());
  271. if (!hasNotificationSupport(branchDto)) {
  272. return null;
  273. }
  274. RuleDefinitionDto ruleDefinitionDto = bulkChangeData.rulesByKey.get(issue.ruleKey());
  275. ComponentDto projectDto = bulkChangeData.projectsByUuid.get(issue.projectUuid());
  276. if (ruleDefinitionDto == null || projectDto == null) {
  277. return null;
  278. }
  279. Optional<UserDto> assignee = Optional.ofNullable(issue.assignee()).map(userDtoByUuid::get);
  280. return new ChangedIssue.Builder(issue.key())
  281. .setNewStatus(issue.status())
  282. .setNewResolution(issue.resolution())
  283. .setAssignee(assignee.map(u -> new User(u.getUuid(), u.getLogin(), u.getName())).orElse(null))
  284. .setRule(new IssuesChangesNotificationBuilder.Rule(ruleDefinitionDto.getKey(), RuleType.valueOfNullable(ruleDefinitionDto.getType()), ruleDefinitionDto.getName()))
  285. .setProject(new Project.Builder(projectDto.uuid())
  286. .setKey(projectDto.getKey())
  287. .setProjectName(projectDto.name())
  288. .setBranchName(branchDto.isMain() ? null : branchDto.getKey())
  289. .build())
  290. .build();
  291. }
  292. private static boolean hasNotificationSupport(@Nullable BranchDto branch) {
  293. return branch != null && branch.getBranchType() != BranchType.PULL_REQUEST;
  294. }
  295. private static long oldestUpdateDate(Collection<DefaultIssue> issues) {
  296. long res = Long.MAX_VALUE;
  297. for (DefaultIssue issue : issues) {
  298. long issueUpdateDate = issue.updateDate().getTime();
  299. if (issueUpdateDate < res) {
  300. res = issueUpdateDate;
  301. }
  302. }
  303. return res;
  304. }
  305. private static Issues.BulkChangeWsResponse toWsResponse(BulkChangeResult result) {
  306. return Issues.BulkChangeWsResponse.newBuilder()
  307. .setTotal(result.countTotal())
  308. .setSuccess(result.countSuccess())
  309. .setIgnored((long) result.countTotal() - (result.countSuccess() + result.countFailures()))
  310. .setFailures(result.countFailures())
  311. .build();
  312. }
  313. private class BulkChangeData {
  314. private final Map<String, Map<String, Object>> propertiesByActions;
  315. private final boolean sendNotification;
  316. private final Collection<DefaultIssue> issues;
  317. private final Map<String, ComponentDto> projectsByUuid;
  318. private final Map<String, BranchDto> branchesByProjectUuid;
  319. private final Map<String, ComponentDto> componentsByUuid;
  320. private final Map<RuleKey, RuleDefinitionDto> rulesByKey;
  321. private final List<Action> availableActions;
  322. BulkChangeData(DbSession dbSession, Request request) {
  323. this.sendNotification = request.mandatoryParamAsBoolean(PARAM_SEND_NOTIFICATIONS);
  324. this.propertiesByActions = toPropertiesByActions(request);
  325. List<String> issueKeys = request.mandatoryParamAsStrings(PARAM_ISSUES);
  326. checkArgument(issueKeys.size() <= MAX_LIMIT, "Number of issues is limited to %s", MAX_LIMIT);
  327. List<IssueDto> allIssues = dbClient.issueDao().selectByKeys(dbSession, issueKeys);
  328. List<ComponentDto> allProjects = getComponents(dbSession, allIssues.stream().map(IssueDto::getProjectUuid).collect(MoreCollectors.toSet()));
  329. this.projectsByUuid = getAuthorizedProjects(allProjects).stream().collect(uniqueIndex(ComponentDto::uuid, identity()));
  330. this.branchesByProjectUuid = dbClient.branchDao().selectByUuids(dbSession, projectsByUuid.keySet()).stream()
  331. .collect(uniqueIndex(BranchDto::getUuid, identity()));
  332. this.issues = getAuthorizedIssues(allIssues);
  333. this.componentsByUuid = getComponents(dbSession,
  334. issues.stream().map(DefaultIssue::componentUuid).collect(MoreCollectors.toSet())).stream()
  335. .collect(uniqueIndex(ComponentDto::uuid, identity()));
  336. this.rulesByKey = dbClient.ruleDao().selectDefinitionByKeys(dbSession,
  337. issues.stream().map(DefaultIssue::ruleKey).collect(MoreCollectors.toSet())).stream()
  338. .collect(uniqueIndex(RuleDefinitionDto::getKey, identity()));
  339. this.availableActions = actions.stream()
  340. .filter(action -> propertiesByActions.containsKey(action.key()))
  341. .filter(action -> action.verify(getProperties(action.key()), issues, userSession))
  342. .collect(MoreCollectors.toList());
  343. }
  344. private List<ComponentDto> getComponents(DbSession dbSession, Collection<String> componentUuids) {
  345. return dbClient.componentDao().selectByUuids(dbSession, componentUuids);
  346. }
  347. private List<ComponentDto> getAuthorizedProjects(List<ComponentDto> projectDtos) {
  348. return userSession.keepAuthorizedComponents(UserRole.USER, projectDtos);
  349. }
  350. private List<DefaultIssue> getAuthorizedIssues(List<IssueDto> allIssues) {
  351. Set<String> projectUuids = projectsByUuid.values().stream().map(ComponentDto::uuid).collect(MoreCollectors.toSet());
  352. return allIssues.stream()
  353. .filter(issue -> projectUuids.contains(issue.getProjectUuid()))
  354. .map(IssueDto::toDefaultIssue)
  355. .collect(MoreCollectors.toList());
  356. }
  357. Map<String, Object> getProperties(String actionKey) {
  358. return propertiesByActions.get(actionKey);
  359. }
  360. List<Action> getActionsWithoutComment() {
  361. return availableActions.stream().filter(action -> !action.key().equals(COMMENT_KEY)).collect(MoreCollectors.toList());
  362. }
  363. Optional<Action> getCommentAction() {
  364. return availableActions.stream().filter(action -> action.key().equals(COMMENT_KEY)).findFirst();
  365. }
  366. private Map<String, Map<String, Object>> toPropertiesByActions(Request request) {
  367. Map<String, Map<String, Object>> properties = new HashMap<>();
  368. request.getParam(PARAM_ASSIGN, value -> properties.put(AssignAction.ASSIGN_KEY, new HashMap<>(of(ASSIGNEE_PARAMETER, value))));
  369. request.getParam(PARAM_SET_SEVERITY, value -> properties.put(SET_SEVERITY_KEY, new HashMap<>(of(SEVERITY_PARAMETER, value))));
  370. request.getParam(PARAM_SET_TYPE, value -> properties.put(SET_TYPE_KEY, new HashMap<>(of(TYPE_PARAMETER, value))));
  371. request.getParam(PARAM_DO_TRANSITION, value -> properties.put(DO_TRANSITION_KEY, new HashMap<>(of(TRANSITION_PARAMETER, value))));
  372. request.getParam(PARAM_ADD_TAGS, value -> properties.put(AddTagsAction.KEY, new HashMap<>(of(TAGS_PARAMETER, value))));
  373. request.getParam(PARAM_REMOVE_TAGS, value -> properties.put(RemoveTagsAction.KEY, new HashMap<>(of(TAGS_PARAMETER, value))));
  374. request.getParam(PARAM_COMMENT, value -> properties.put(COMMENT_KEY, new HashMap<>(of(COMMENT_PROPERTY, value))));
  375. checkAtLeastOneActionIsDefined(properties.keySet());
  376. return properties;
  377. }
  378. private void checkAtLeastOneActionIsDefined(Set<String> actions) {
  379. long actionsDefined = actions.stream().filter(action -> !action.equals(COMMENT_KEY)).count();
  380. checkArgument(actionsDefined > 0, "At least one action must be provided");
  381. }
  382. private boolean shouldRefreshMeasures() {
  383. return availableActions.stream().anyMatch(Action::shouldRefreshMeasures);
  384. }
  385. }
  386. private static class BulkChangeResult {
  387. private final int total;
  388. private final Set<DefaultIssue> success = new HashSet<>();
  389. private int failures = 0;
  390. BulkChangeResult(int total) {
  391. this.total = total;
  392. }
  393. void increaseSuccess(DefaultIssue issue) {
  394. this.success.add(issue);
  395. }
  396. void increaseFailure() {
  397. this.failures++;
  398. }
  399. int countTotal() {
  400. return total;
  401. }
  402. int countSuccess() {
  403. return success.size();
  404. }
  405. int countFailures() {
  406. return failures;
  407. }
  408. }
  409. }