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.

UsersAction.java 8.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2023 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;
  21. import com.google.common.collect.Multimap;
  22. import com.google.common.collect.Ordering;
  23. import com.google.common.collect.TreeMultimap;
  24. import java.util.List;
  25. import java.util.Map;
  26. import java.util.Set;
  27. import javax.annotation.Nullable;
  28. import org.sonar.api.server.ws.Change;
  29. import org.sonar.api.server.ws.Request;
  30. import org.sonar.api.server.ws.Response;
  31. import org.sonar.api.server.ws.WebService;
  32. import org.sonar.api.server.ws.WebService.Param;
  33. import org.sonar.api.utils.Paging;
  34. import org.sonar.db.DbClient;
  35. import org.sonar.db.DbSession;
  36. import org.sonar.db.entity.EntityDto;
  37. import org.sonar.db.permission.PermissionQuery;
  38. import org.sonar.db.permission.UserPermissionDto;
  39. import org.sonar.db.user.UserDto;
  40. import org.sonar.server.common.avatar.AvatarResolver;
  41. import org.sonar.server.management.ManagedInstanceService;
  42. import org.sonar.server.permission.RequestValidator;
  43. import org.sonar.server.user.UserSession;
  44. import org.sonarqube.ws.Permissions;
  45. import org.sonarqube.ws.Permissions.UsersWsResponse;
  46. import static com.google.common.base.Strings.emptyToNull;
  47. import static java.util.Collections.emptyList;
  48. import static java.util.Optional.ofNullable;
  49. import static java.util.stream.Collectors.toSet;
  50. import static org.sonar.db.permission.PermissionQuery.DEFAULT_PAGE_SIZE;
  51. import static org.sonar.db.permission.PermissionQuery.RESULTS_MAX_SIZE;
  52. import static org.sonar.db.permission.PermissionQuery.SEARCH_QUERY_MIN_LENGTH;
  53. import static org.sonar.server.permission.RequestValidator.validateGlobalPermission;
  54. import static org.sonar.server.permission.ws.WsParameters.createProjectParameters;
  55. import static org.sonar.server.ws.WsUtils.writeProtobuf;
  56. import static org.sonarqube.ws.client.permission.PermissionsWsParameters.PARAM_PERMISSION;
  57. public class UsersAction implements PermissionsWsAction {
  58. private final DbClient dbClient;
  59. private final UserSession userSession;
  60. private final PermissionWsSupport wsSupport;
  61. private final AvatarResolver avatarResolver;
  62. private final WsParameters wsParameters;
  63. private final RequestValidator requestValidator;
  64. private final ManagedInstanceService managedInstanceService;
  65. public UsersAction(DbClient dbClient, UserSession userSession, PermissionWsSupport wsSupport, AvatarResolver avatarResolver, WsParameters wsParameters,
  66. RequestValidator requestValidator, ManagedInstanceService managedInstanceService) {
  67. this.dbClient = dbClient;
  68. this.userSession = userSession;
  69. this.wsSupport = wsSupport;
  70. this.avatarResolver = avatarResolver;
  71. this.wsParameters = wsParameters;
  72. this.requestValidator = requestValidator;
  73. this.managedInstanceService = managedInstanceService;
  74. }
  75. @Override
  76. public void define(WebService.NewController context) {
  77. WebService.NewAction action = context.createAction("users")
  78. .setSince("5.2")
  79. .setDescription("Lists the users with their permissions as individual users rather than through group affiliation.<br>" +
  80. "This service defaults to global permissions, but can be limited to project permissions by providing project id or project key.<br> " +
  81. "This service defaults to all users, but can be limited to users with a specific permission by providing the desired permission.<br>" +
  82. "Requires one of the following permissions:" +
  83. "<ul>" +
  84. "<li>'Administer System'</li>" +
  85. "<li>'Administer' rights on the specified project</li>" +
  86. "</ul>")
  87. .addPagingParams(DEFAULT_PAGE_SIZE, RESULTS_MAX_SIZE)
  88. .setChangelog(
  89. new Change("10.0", "Response includes 'managed' field."),
  90. new Change("7.4", "The response list is returning all users even those without permissions, the users with permission are at the top of the list."))
  91. .setInternal(true)
  92. .setResponseExample(getClass().getResource("users-example.json"))
  93. .setHandler(this);
  94. action.createParam(Param.TEXT_QUERY)
  95. .setMinimumLength(SEARCH_QUERY_MIN_LENGTH)
  96. .setDescription("Limit search to user names that contain the supplied string. <br/>")
  97. .setExampleValue("eri");
  98. wsParameters.createPermissionParameter(action).setRequired(false);
  99. createProjectParameters(action);
  100. }
  101. @Override
  102. public void handle(Request request, Response response) throws Exception {
  103. try (DbSession dbSession = dbClient.openSession(false)) {
  104. EntityDto entity = wsSupport.findEntity(dbSession, request);
  105. wsSupport.checkPermissionManagementAccess(userSession, entity);
  106. PermissionQuery query = buildPermissionQuery(request, entity);
  107. List<UserDto> users = findUsers(dbSession, query);
  108. int total = dbClient.userPermissionDao().countUsersByQuery(dbSession, query);
  109. List<UserPermissionDto> userPermissions = findUserPermissions(dbSession, users, entity);
  110. Paging paging = Paging.forPageIndex(request.mandatoryParamAsInt(Param.PAGE)).withPageSize(query.getPageSize()).andTotal(total);
  111. Map<String, Boolean> userUuidToIsManaged = managedInstanceService.getUserUuidToManaged(dbSession, getUserUuids(users));
  112. UsersWsResponse usersWsResponse = buildResponse(users, userPermissions, userUuidToIsManaged, paging);
  113. writeProtobuf(usersWsResponse, request, response);
  114. }
  115. }
  116. private static Set<String> getUserUuids(List<UserDto> users) {
  117. return users.stream().map(UserDto::getUuid).collect(toSet());
  118. }
  119. private PermissionQuery buildPermissionQuery(Request request, @Nullable EntityDto entity) {
  120. String textQuery = request.param(Param.TEXT_QUERY);
  121. String permission = request.param(PARAM_PERMISSION);
  122. PermissionQuery.Builder permissionQuery = PermissionQuery.builder()
  123. .setPermission(permission)
  124. .setPageIndex(request.mandatoryParamAsInt(Param.PAGE))
  125. .setPageSize(request.mandatoryParamAsInt(Param.PAGE_SIZE))
  126. .setSearchQuery(textQuery);
  127. if (entity != null) {
  128. permissionQuery.setEntityUuid(entity.getUuid());
  129. }
  130. if (permission != null) {
  131. if (entity != null) {
  132. requestValidator.validateProjectPermission(permission);
  133. } else {
  134. validateGlobalPermission(permission);
  135. }
  136. }
  137. return permissionQuery.build();
  138. }
  139. private UsersWsResponse buildResponse(List<UserDto> users, List<UserPermissionDto> userPermissions, Map<String, Boolean> userUuidToIsManaged,
  140. Paging paging) {
  141. Multimap<String, String> permissionsByUserUuid = TreeMultimap.create();
  142. userPermissions.forEach(userPermission -> permissionsByUserUuid.put(userPermission.getUserUuid(), userPermission.getPermission()));
  143. UsersWsResponse.Builder response = UsersWsResponse.newBuilder();
  144. users.forEach(user -> {
  145. Permissions.User.Builder userResponse = response.addUsersBuilder()
  146. .setLogin(user.getLogin())
  147. .addAllPermissions(permissionsByUserUuid.get(user.getUuid()));
  148. ofNullable(user.getEmail()).ifPresent(userResponse::setEmail);
  149. ofNullable(emptyToNull(user.getEmail())).ifPresent(u -> userResponse.setAvatar(avatarResolver.create(user)));
  150. ofNullable(user.getName()).ifPresent(userResponse::setName);
  151. ofNullable(userUuidToIsManaged.get(user.getUuid())).ifPresent(userResponse::setManaged);
  152. });
  153. response.getPagingBuilder()
  154. .setPageIndex(paging.pageIndex())
  155. .setPageSize(paging.pageSize())
  156. .setTotal(paging.total())
  157. .build();
  158. return response.build();
  159. }
  160. private List<UserDto> findUsers(DbSession dbSession, PermissionQuery query) {
  161. List<String> orderedUuids = dbClient.userPermissionDao().selectUserUuidsByQueryAndScope(dbSession, query);
  162. return Ordering.explicit(orderedUuids).onResultOf(UserDto::getUuid).immutableSortedCopy(dbClient.userDao().selectByUuids(dbSession, orderedUuids));
  163. }
  164. private List<UserPermissionDto> findUserPermissions(DbSession dbSession, List<UserDto> users, @Nullable EntityDto entity) {
  165. if (users.isEmpty()) {
  166. return emptyList();
  167. }
  168. List<String> userUuids = users.stream().map(UserDto::getUuid).toList();
  169. PermissionQuery.Builder queryBuilder = PermissionQuery.builder()
  170. .withAtLeastOnePermission();
  171. if (entity != null) {
  172. queryBuilder.setEntityUuid(entity.getUuid());
  173. }
  174. return dbClient.userPermissionDao().selectUserPermissionsByQuery(dbSession, queryBuilder.build(), userUuids);
  175. }
  176. }