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.

BulkApplyTemplateAction.java 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2024 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.permission.ws.template;
  21. import java.util.Collection;
  22. import java.util.HashSet;
  23. import java.util.List;
  24. import java.util.Set;
  25. import java.util.stream.Collectors;
  26. import javax.annotation.CheckForNull;
  27. import javax.annotation.Nullable;
  28. import org.sonar.api.resources.Qualifiers;
  29. import org.sonar.api.resources.ResourceTypes;
  30. import org.sonar.api.server.ws.Change;
  31. import org.sonar.api.server.ws.Request;
  32. import org.sonar.api.server.ws.Response;
  33. import org.sonar.api.server.ws.WebService;
  34. import org.sonar.api.server.ws.WebService.Param;
  35. import org.sonar.core.i18n.I18n;
  36. import org.sonar.db.DatabaseUtils;
  37. import org.sonar.db.DbClient;
  38. import org.sonar.db.DbSession;
  39. import org.sonar.db.ce.CeTaskQuery;
  40. import org.sonar.db.component.ComponentDto;
  41. import org.sonar.db.component.ComponentQuery;
  42. import org.sonar.db.entity.EntityDto;
  43. import org.sonar.db.permission.template.PermissionTemplateDto;
  44. import org.sonar.server.management.ManagedProjectService;
  45. import org.sonar.server.common.permission.PermissionTemplateService;
  46. import org.sonar.server.permission.ws.PermissionWsSupport;
  47. import org.sonar.server.permission.ws.PermissionsWsAction;
  48. import org.sonar.server.permission.ws.WsParameters;
  49. import org.sonar.server.project.Visibility;
  50. import org.sonar.server.user.UserSession;
  51. import static java.lang.String.format;
  52. import static java.util.Collections.singleton;
  53. import static java.util.Objects.requireNonNull;
  54. import static java.util.Optional.ofNullable;
  55. import static org.sonar.api.utils.DateUtils.parseDateOrDateTime;
  56. import static org.sonar.db.Pagination.forPage;
  57. import static org.sonar.server.permission.PermissionPrivilegeChecker.checkGlobalAdmin;
  58. import static org.sonar.server.permission.ws.template.WsTemplateRef.newTemplateRef;
  59. import static org.sonar.server.ws.KeyExamples.KEY_PROJECT_EXAMPLE_001;
  60. import static org.sonar.server.ws.KeyExamples.KEY_PROJECT_EXAMPLE_002;
  61. import static org.sonar.server.ws.WsParameterBuilder.QualifierParameterContext.newQualifierParameterContext;
  62. import static org.sonar.server.ws.WsParameterBuilder.createRootQualifiersParameter;
  63. import static org.sonarqube.ws.client.permission.PermissionsWsParameters.PARAM_TEMPLATE_ID;
  64. import static org.sonarqube.ws.client.permission.PermissionsWsParameters.PARAM_TEMPLATE_NAME;
  65. import static org.sonarqube.ws.client.project.ProjectsWsParameters.PARAM_ANALYZED_BEFORE;
  66. import static org.sonarqube.ws.client.project.ProjectsWsParameters.PARAM_ON_PROVISIONED_ONLY;
  67. import static org.sonarqube.ws.client.project.ProjectsWsParameters.PARAM_PROJECTS;
  68. import static org.sonarqube.ws.client.project.ProjectsWsParameters.PARAM_QUALIFIERS;
  69. import static org.sonarqube.ws.client.project.ProjectsWsParameters.PARAM_VISIBILITY;
  70. public class BulkApplyTemplateAction implements PermissionsWsAction {
  71. private final DbClient dbClient;
  72. private final UserSession userSession;
  73. private final PermissionTemplateService permissionTemplateService;
  74. private final PermissionWsSupport wsSupport;
  75. private final I18n i18n;
  76. private final ResourceTypes resourceTypes;
  77. private final ManagedProjectService managedProjectService;
  78. public BulkApplyTemplateAction(DbClient dbClient, UserSession userSession, PermissionTemplateService permissionTemplateService, PermissionWsSupport wsSupport, I18n i18n,
  79. ResourceTypes resourceTypes, ManagedProjectService managedProjectService) {
  80. this.dbClient = dbClient;
  81. this.userSession = userSession;
  82. this.permissionTemplateService = permissionTemplateService;
  83. this.wsSupport = wsSupport;
  84. this.i18n = i18n;
  85. this.resourceTypes = resourceTypes;
  86. this.managedProjectService = managedProjectService;
  87. }
  88. @Override
  89. public void define(WebService.NewController context) {
  90. WebService.NewAction action = context.createAction("bulk_apply_template")
  91. .setDescription("Apply a permission template to several components. Managed projects will be ignored.<br />" +
  92. "The template id or name must be provided.<br />" +
  93. "Requires the following permission: 'Administer System'.")
  94. .setPost(true)
  95. .setSince("5.5")
  96. .setChangelog(new Change("6.7.2", format("Parameter %s accepts maximum %d values", PARAM_PROJECTS, DatabaseUtils.PARTITION_SIZE_FOR_ORACLE)))
  97. .setHandler(this);
  98. action.createParam(Param.TEXT_QUERY)
  99. .setDescription("Limit search to: <ul>" +
  100. "<li>project names that contain the supplied string</li>" +
  101. "<li>project keys that are exactly the same as the supplied string</li>" +
  102. "</ul>")
  103. .setExampleValue("apac");
  104. createRootQualifiersParameter(action, newQualifierParameterContext(i18n, resourceTypes))
  105. .setDefaultValue(Qualifiers.PROJECT);
  106. WsParameters.createTemplateParameters(action);
  107. action
  108. .createParam(PARAM_PROJECTS)
  109. .setDescription("Comma-separated list of project keys")
  110. .setSince("6.6")
  111. // Limitation of ComponentDao#selectByQuery(), max 1000 values are accepted.
  112. // Restricting size of HTTP parameter allows to not fail with SQL error
  113. .setMaxValuesAllowed(DatabaseUtils.PARTITION_SIZE_FOR_ORACLE)
  114. .setExampleValue(String.join(",", KEY_PROJECT_EXAMPLE_001, KEY_PROJECT_EXAMPLE_002));
  115. action.createParam(PARAM_VISIBILITY)
  116. .setDescription("Filter the projects that should be visible to everyone (%s), or only specific user/groups (%s).<br/>" +
  117. "If no visibility is specified, the default project visibility will be used.",
  118. Visibility.PUBLIC.getLabel(), Visibility.PRIVATE.getLabel())
  119. .setRequired(false)
  120. .setInternal(true)
  121. .setSince("6.6")
  122. .setPossibleValues(Visibility.getLabels());
  123. action.createParam(PARAM_ANALYZED_BEFORE)
  124. .setDescription("Filter the projects for which last analysis is older than the given date (exclusive).<br> " +
  125. "Either a date (server timezone) or datetime can be provided.")
  126. .setSince("6.6")
  127. .setExampleValue("2017-10-19 or 2017-10-19T13:00:00+0200");
  128. action.createParam(PARAM_ON_PROVISIONED_ONLY)
  129. .setDescription("Filter the projects that are provisioned")
  130. .setBooleanPossibleValues()
  131. .setDefaultValue("false")
  132. .setSince("6.6");
  133. }
  134. @Override
  135. public void handle(Request request, Response response) throws Exception {
  136. doHandle(toBulkApplyTemplateWsRequest(request));
  137. response.noContent();
  138. }
  139. private void doHandle(BulkApplyTemplateRequest request) {
  140. try (DbSession dbSession = dbClient.openSession(false)) {
  141. PermissionTemplateDto template = wsSupport.findTemplate(dbSession, newTemplateRef(
  142. request.getTemplateId(), request.getTemplateName()));
  143. checkGlobalAdmin(userSession);
  144. ComponentQuery componentQuery = buildDbQuery(request);
  145. List<ComponentDto> components = dbClient.componentDao().selectByQuery(dbSession, componentQuery, forPage(1).andSize(CeTaskQuery.MAX_COMPONENT_UUIDS));
  146. Set<String> entityUuids = components.stream()
  147. .map(ComponentDto::getKey)
  148. .collect(Collectors.toSet());
  149. List<EntityDto> entities = dbClient.entityDao().selectByKeys(dbSession, entityUuids).stream()
  150. .filter(entity -> !managedProjectService.isProjectManaged(dbSession, entity.getUuid()))
  151. .toList();
  152. permissionTemplateService.applyAndCommit(dbSession, template, entities);
  153. }
  154. }
  155. private static BulkApplyTemplateRequest toBulkApplyTemplateWsRequest(Request request) {
  156. return new BulkApplyTemplateRequest()
  157. .setTemplateId(request.param(PARAM_TEMPLATE_ID))
  158. .setTemplateName(request.param(PARAM_TEMPLATE_NAME))
  159. .setQualifiers(request.mandatoryParamAsStrings(PARAM_QUALIFIERS))
  160. .setQuery(request.param(Param.TEXT_QUERY))
  161. .setVisibility(request.param(PARAM_VISIBILITY))
  162. .setOnProvisionedOnly(request.mandatoryParamAsBoolean(PARAM_ON_PROVISIONED_ONLY))
  163. .setAnalyzedBefore(request.param(PARAM_ANALYZED_BEFORE))
  164. .setProjects(request.paramAsStrings(PARAM_PROJECTS));
  165. }
  166. private static ComponentQuery buildDbQuery(BulkApplyTemplateRequest request) {
  167. Collection<String> qualifiers = request.getQualifiers();
  168. ComponentQuery.Builder query = ComponentQuery.builder()
  169. .setQualifiers(qualifiers.toArray(new String[qualifiers.size()]));
  170. ofNullable(request.getQuery()).ifPresent(q -> {
  171. query.setNameOrKeyQuery(q);
  172. query.setPartialMatchOnKey(true);
  173. });
  174. ofNullable(request.getVisibility()).ifPresent(v -> query.setPrivate(Visibility.isPrivate(v)));
  175. ofNullable(request.getAnalyzedBefore()).ifPresent(d -> query.setAnalyzedBefore(parseDateOrDateTime(d).getTime()));
  176. query.setOnProvisionedOnly(request.isOnProvisionedOnly());
  177. ofNullable(request.getProjects()).ifPresent(keys -> query.setComponentKeys(new HashSet<>(keys)));
  178. return query.build();
  179. }
  180. private static class BulkApplyTemplateRequest {
  181. private String templateId;
  182. private String templateName;
  183. private String query;
  184. private Collection<String> qualifiers = singleton(Qualifiers.PROJECT);
  185. private String visibility;
  186. private String analyzedBefore;
  187. private boolean onProvisionedOnly = false;
  188. private Collection<String> projects;
  189. @CheckForNull
  190. public String getTemplateId() {
  191. return templateId;
  192. }
  193. public BulkApplyTemplateRequest setTemplateId(@Nullable String templateId) {
  194. this.templateId = templateId;
  195. return this;
  196. }
  197. @CheckForNull
  198. public String getTemplateName() {
  199. return templateName;
  200. }
  201. public BulkApplyTemplateRequest setTemplateName(@Nullable String templateName) {
  202. this.templateName = templateName;
  203. return this;
  204. }
  205. @CheckForNull
  206. public String getQuery() {
  207. return query;
  208. }
  209. public BulkApplyTemplateRequest setQuery(@Nullable String query) {
  210. this.query = query;
  211. return this;
  212. }
  213. public Collection<String> getQualifiers() {
  214. return qualifiers;
  215. }
  216. public BulkApplyTemplateRequest setQualifiers(Collection<String> qualifiers) {
  217. this.qualifiers = requireNonNull(qualifiers);
  218. return this;
  219. }
  220. @CheckForNull
  221. public String getVisibility() {
  222. return visibility;
  223. }
  224. public BulkApplyTemplateRequest setVisibility(@Nullable String visibility) {
  225. this.visibility = visibility;
  226. return this;
  227. }
  228. @CheckForNull
  229. public String getAnalyzedBefore() {
  230. return analyzedBefore;
  231. }
  232. public BulkApplyTemplateRequest setAnalyzedBefore(@Nullable String analyzedBefore) {
  233. this.analyzedBefore = analyzedBefore;
  234. return this;
  235. }
  236. public boolean isOnProvisionedOnly() {
  237. return onProvisionedOnly;
  238. }
  239. public BulkApplyTemplateRequest setOnProvisionedOnly(boolean onProvisionedOnly) {
  240. this.onProvisionedOnly = onProvisionedOnly;
  241. return this;
  242. }
  243. @CheckForNull
  244. public Collection<String> getProjects() {
  245. return projects;
  246. }
  247. public BulkApplyTemplateRequest setProjects(@Nullable Collection<String> projects) {
  248. this.projects = projects;
  249. return this;
  250. }
  251. }
  252. }