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.

GenerateAction.java 5.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2019 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.usertoken.ws;
  21. import org.sonar.api.server.ws.Request;
  22. import org.sonar.api.server.ws.Response;
  23. import org.sonar.api.server.ws.WebService;
  24. import org.sonar.api.utils.System2;
  25. import org.sonar.db.DbClient;
  26. import org.sonar.db.DbSession;
  27. import org.sonar.db.user.UserDto;
  28. import org.sonar.db.user.UserTokenDto;
  29. import org.sonar.server.exceptions.ServerException;
  30. import org.sonar.server.usertoken.TokenGenerator;
  31. import org.sonarqube.ws.UserTokens;
  32. import org.sonarqube.ws.UserTokens.GenerateWsResponse;
  33. import static java.net.HttpURLConnection.HTTP_INTERNAL_ERROR;
  34. import static org.sonar.api.utils.DateUtils.formatDateTime;
  35. import static org.sonar.server.usertoken.ws.UserTokenSupport.ACTION_GENERATE;
  36. import static org.sonar.server.usertoken.ws.UserTokenSupport.PARAM_LOGIN;
  37. import static org.sonar.server.usertoken.ws.UserTokenSupport.PARAM_NAME;
  38. import static org.sonar.server.ws.WsUtils.checkRequest;
  39. import static org.sonar.server.ws.WsUtils.writeProtobuf;
  40. public class GenerateAction implements UserTokensWsAction {
  41. private static final int MAX_TOKEN_NAME_LENGTH = 100;
  42. private final DbClient dbClient;
  43. private final System2 system;
  44. private final TokenGenerator tokenGenerator;
  45. private final UserTokenSupport userTokenSupport;
  46. public GenerateAction(DbClient dbClient, System2 system, TokenGenerator tokenGenerator, UserTokenSupport userTokenSupport) {
  47. this.dbClient = dbClient;
  48. this.system = system;
  49. this.tokenGenerator = tokenGenerator;
  50. this.userTokenSupport = userTokenSupport;
  51. }
  52. @Override
  53. public void define(WebService.NewController context) {
  54. WebService.NewAction action = context.createAction(ACTION_GENERATE)
  55. .setSince("5.3")
  56. .setPost(true)
  57. .setDescription("Generate a user access token. <br />" +
  58. "Please keep your tokens secret. They enable to authenticate and analyze projects.<br />" +
  59. "It requires administration permissions to specify a 'login' and generate a token for another user. Otherwise, a token is generated for the current user.")
  60. .setResponseExample(getClass().getResource("generate-example.json"))
  61. .setHandler(this);
  62. action.createParam(PARAM_LOGIN)
  63. .setDescription("User login. If not set, the token is generated for the authenticated user.")
  64. .setExampleValue("g.hopper");
  65. action.createParam(PARAM_NAME)
  66. .setRequired(true)
  67. .setMaximumLength(MAX_TOKEN_NAME_LENGTH)
  68. .setDescription("Token name")
  69. .setExampleValue("Project scan on Travis");
  70. }
  71. @Override
  72. public void handle(Request request, Response response) throws Exception {
  73. UserTokens.GenerateWsResponse generateWsResponse = doHandle(request);
  74. writeProtobuf(generateWsResponse, request, response);
  75. }
  76. private UserTokens.GenerateWsResponse doHandle(Request request) {
  77. try (DbSession dbSession = dbClient.openSession(false)) {
  78. String name = request.mandatoryParam(PARAM_NAME).trim();
  79. UserDto user = userTokenSupport.getUser(dbSession, request);
  80. checkTokenDoesNotAlreadyExists(dbSession, user, name);
  81. String token = tokenGenerator.generate();
  82. String tokenHash = hashToken(dbSession, token);
  83. UserTokenDto userTokenDto = insertTokenInDb(dbSession, user, name, tokenHash);
  84. return buildResponse(userTokenDto, token, user);
  85. }
  86. }
  87. private String hashToken(DbSession dbSession, String token) {
  88. String tokenHash = tokenGenerator.hash(token);
  89. UserTokenDto userToken = dbClient.userTokenDao().selectByTokenHash(dbSession, tokenHash);
  90. if (userToken == null) {
  91. return tokenHash;
  92. }
  93. throw new ServerException(HTTP_INTERNAL_ERROR, "Error while generating token. Please try again.");
  94. }
  95. private void checkTokenDoesNotAlreadyExists(DbSession dbSession, UserDto user, String name) {
  96. UserTokenDto userTokenDto = dbClient.userTokenDao().selectByUserAndName(dbSession, user, name);
  97. checkRequest(userTokenDto == null, "A user token for login '%s' and name '%s' already exists", user.getLogin(), name);
  98. }
  99. private UserTokenDto insertTokenInDb(DbSession dbSession, UserDto user, String name, String tokenHash) {
  100. UserTokenDto userTokenDto = new UserTokenDto()
  101. .setUserUuid(user.getUuid())
  102. .setName(name)
  103. .setTokenHash(tokenHash)
  104. .setCreatedAt(system.now());
  105. dbClient.userTokenDao().insert(dbSession, userTokenDto);
  106. dbSession.commit();
  107. return userTokenDto;
  108. }
  109. private static GenerateWsResponse buildResponse(UserTokenDto userTokenDto, String token, UserDto user) {
  110. return UserTokens.GenerateWsResponse.newBuilder()
  111. .setLogin(user.getLogin())
  112. .setName(userTokenDto.getName())
  113. .setCreatedAt(formatDateTime(userTokenDto.getCreatedAt()))
  114. .setToken(token)
  115. .build();
  116. }
  117. }