Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

FederationServlet.java 10KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. /*
  2. * Copyright 2011 gitblit.com.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.gitblit.servlet;
  17. import java.io.File;
  18. import java.text.MessageFormat;
  19. import java.util.ArrayList;
  20. import java.util.Date;
  21. import java.util.HashMap;
  22. import java.util.HashSet;
  23. import java.util.List;
  24. import java.util.Map;
  25. import java.util.Set;
  26. import javax.inject.Inject;
  27. import javax.inject.Singleton;
  28. import javax.servlet.http.HttpServletResponse;
  29. import com.gitblit.Constants.FederationRequest;
  30. import com.gitblit.IStoredSettings;
  31. import com.gitblit.Keys;
  32. import com.gitblit.manager.IFederationManager;
  33. import com.gitblit.manager.IRepositoryManager;
  34. import com.gitblit.manager.IRuntimeManager;
  35. import com.gitblit.manager.IUserManager;
  36. import com.gitblit.models.FederationModel;
  37. import com.gitblit.models.FederationProposal;
  38. import com.gitblit.models.TeamModel;
  39. import com.gitblit.models.UserModel;
  40. import com.gitblit.utils.FederationUtils;
  41. import com.gitblit.utils.FileUtils;
  42. import com.gitblit.utils.HttpUtils;
  43. import com.gitblit.utils.StringUtils;
  44. import com.gitblit.utils.TimeUtils;
  45. /**
  46. * Handles federation requests.
  47. *
  48. * @author James Moger
  49. *
  50. */
  51. @Singleton
  52. public class FederationServlet extends JsonServlet {
  53. private static final long serialVersionUID = 1L;
  54. private final IStoredSettings settings;
  55. private final IUserManager userManager;
  56. private final IRepositoryManager repositoryManager;
  57. private final IFederationManager federationManager;
  58. @Inject
  59. public FederationServlet(
  60. IRuntimeManager runtimeManager,
  61. IUserManager userManager,
  62. IRepositoryManager repositoryManager,
  63. IFederationManager federationManager) {
  64. super();
  65. this.settings = runtimeManager.getSettings();
  66. this.userManager = userManager;
  67. this.repositoryManager = repositoryManager;
  68. this.federationManager = federationManager;
  69. }
  70. /**
  71. * Processes a federation request.
  72. *
  73. * @param request
  74. * @param response
  75. * @throws javax.servlet.ServletException
  76. * @throws java.io.IOException
  77. */
  78. @Override
  79. protected void processRequest(javax.servlet.http.HttpServletRequest request,
  80. javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException,
  81. java.io.IOException {
  82. FederationRequest reqType = FederationRequest.fromName(request.getParameter("req"));
  83. logger.info(MessageFormat.format("Federation {0} request from {1}", reqType,
  84. request.getRemoteAddr()));
  85. if (FederationRequest.POKE.equals(reqType)) {
  86. // Gitblit always responds to POKE requests to verify a connection
  87. logger.info("Received federation POKE from " + request.getRemoteAddr());
  88. return;
  89. }
  90. if (!settings.getBoolean(Keys.git.enableGitServlet, true)) {
  91. logger.warn(Keys.git.enableGitServlet + " must be set TRUE for federation requests.");
  92. response.sendError(HttpServletResponse.SC_FORBIDDEN);
  93. return;
  94. }
  95. String uuid = settings.getString(Keys.federation.passphrase, "");
  96. if (StringUtils.isEmpty(uuid)) {
  97. logger.warn(Keys.federation.passphrase
  98. + " is not properly set! Federation request denied.");
  99. response.sendError(HttpServletResponse.SC_FORBIDDEN);
  100. return;
  101. }
  102. if (FederationRequest.PROPOSAL.equals(reqType)) {
  103. // Receive a gitblit federation proposal
  104. FederationProposal proposal = deserialize(request, response, FederationProposal.class);
  105. if (proposal == null) {
  106. return;
  107. }
  108. // reject proposal, if not receipt prohibited
  109. if (!settings.getBoolean(Keys.federation.allowProposals, false)) {
  110. logger.error(MessageFormat.format("Rejected {0} federation proposal from {1}",
  111. proposal.tokenType.name(), proposal.url));
  112. response.setStatus(HttpServletResponse.SC_METHOD_NOT_ALLOWED);
  113. return;
  114. }
  115. // poke the origin Gitblit instance that is proposing federation
  116. boolean poked = false;
  117. try {
  118. poked = FederationUtils.poke(proposal.url);
  119. } catch (Exception e) {
  120. logger.error("Failed to poke origin", e);
  121. }
  122. if (!poked) {
  123. logger.error(MessageFormat.format("Failed to send federation poke to {0}",
  124. proposal.url));
  125. response.setStatus(HttpServletResponse.SC_NOT_ACCEPTABLE);
  126. return;
  127. }
  128. String url = HttpUtils.getGitblitURL(request);
  129. federationManager.submitFederationProposal(proposal, url);
  130. logger.info(MessageFormat.format(
  131. "Submitted {0} federation proposal to pull {1} repositories from {2}",
  132. proposal.tokenType.name(), proposal.repositories.size(), proposal.url));
  133. response.setStatus(HttpServletResponse.SC_OK);
  134. return;
  135. }
  136. if (FederationRequest.STATUS.equals(reqType)) {
  137. // Receive a gitblit federation status acknowledgment
  138. String remoteId = StringUtils.decodeFromHtml(request.getParameter("url"));
  139. String identification = MessageFormat.format("{0} ({1})", remoteId,
  140. request.getRemoteAddr());
  141. // deserialize the status data
  142. FederationModel results = deserialize(request, response, FederationModel.class);
  143. if (results == null) {
  144. return;
  145. }
  146. // setup the last and netx pull dates
  147. results.lastPull = new Date();
  148. int mins = TimeUtils.convertFrequencyToMinutes(results.frequency);
  149. results.nextPull = new Date(System.currentTimeMillis() + (mins * 60 * 1000L));
  150. // acknowledge the receipt of status
  151. federationManager.acknowledgeFederationStatus(identification, results);
  152. logger.info(MessageFormat.format(
  153. "Received status of {0} federated repositories from {1}", results
  154. .getStatusList().size(), identification));
  155. response.setStatus(HttpServletResponse.SC_OK);
  156. return;
  157. }
  158. // Determine the federation tokens for this gitblit instance
  159. String token = request.getParameter("token");
  160. List<String> tokens = federationManager.getFederationTokens();
  161. if (!tokens.contains(token)) {
  162. logger.warn(MessageFormat.format(
  163. "Received Federation token ''{0}'' does not match the server tokens", token));
  164. response.sendError(HttpServletResponse.SC_FORBIDDEN);
  165. return;
  166. }
  167. Object result = null;
  168. if (FederationRequest.PULL_REPOSITORIES.equals(reqType)) {
  169. String gitblitUrl = HttpUtils.getGitblitURL(request);
  170. result = federationManager.getRepositories(gitblitUrl, token);
  171. } else {
  172. if (FederationRequest.PULL_SETTINGS.equals(reqType)) {
  173. // pull settings
  174. if (!federationManager.validateFederationRequest(reqType, token)) {
  175. // invalid token to pull users or settings
  176. logger.warn(MessageFormat.format(
  177. "Federation token from {0} not authorized to pull SETTINGS",
  178. request.getRemoteAddr()));
  179. response.sendError(HttpServletResponse.SC_FORBIDDEN);
  180. return;
  181. }
  182. Map<String, String> map = new HashMap<String, String>();
  183. List<String> keys = settings.getAllKeys(null);
  184. for (String key : keys) {
  185. map.put(key, settings.getString(key, ""));
  186. }
  187. result = map;
  188. } else if (FederationRequest.PULL_USERS.equals(reqType)) {
  189. // pull users
  190. if (!federationManager.validateFederationRequest(reqType, token)) {
  191. // invalid token to pull users or settings
  192. logger.warn(MessageFormat.format(
  193. "Federation token from {0} not authorized to pull USERS",
  194. request.getRemoteAddr()));
  195. response.sendError(HttpServletResponse.SC_FORBIDDEN);
  196. return;
  197. }
  198. List<String> usernames = userManager.getAllUsernames();
  199. List<UserModel> users = new ArrayList<UserModel>();
  200. for (String username : usernames) {
  201. UserModel user = userManager.getUserModel(username);
  202. if (!user.excludeFromFederation) {
  203. users.add(user);
  204. }
  205. }
  206. result = users;
  207. } else if (FederationRequest.PULL_TEAMS.equals(reqType)) {
  208. // pull teams
  209. if (!federationManager.validateFederationRequest(reqType, token)) {
  210. // invalid token to pull teams
  211. logger.warn(MessageFormat.format(
  212. "Federation token from {0} not authorized to pull TEAMS",
  213. request.getRemoteAddr()));
  214. response.sendError(HttpServletResponse.SC_FORBIDDEN);
  215. return;
  216. }
  217. List<String> teamnames = userManager.getAllTeamNames();
  218. List<TeamModel> teams = new ArrayList<TeamModel>();
  219. for (String teamname : teamnames) {
  220. TeamModel user = userManager.getTeamModel(teamname);
  221. teams.add(user);
  222. }
  223. result = teams;
  224. } else if (FederationRequest.PULL_SCRIPTS.equals(reqType)) {
  225. // pull scripts
  226. if (!federationManager.validateFederationRequest(reqType, token)) {
  227. // invalid token to pull script
  228. logger.warn(MessageFormat.format(
  229. "Federation token from {0} not authorized to pull SCRIPTS",
  230. request.getRemoteAddr()));
  231. response.sendError(HttpServletResponse.SC_FORBIDDEN);
  232. return;
  233. }
  234. Map<String, String> scripts = new HashMap<String, String>();
  235. Set<String> names = new HashSet<String>();
  236. names.addAll(settings.getStrings(Keys.groovy.preReceiveScripts));
  237. names.addAll(settings.getStrings(Keys.groovy.postReceiveScripts));
  238. for (TeamModel team : userManager.getAllTeams()) {
  239. names.addAll(team.preReceiveScripts);
  240. names.addAll(team.postReceiveScripts);
  241. }
  242. File scriptsFolder = repositoryManager.getHooksFolder();
  243. for (String name : names) {
  244. File file = new File(scriptsFolder, name);
  245. if (!file.exists() && !file.getName().endsWith(".groovy")) {
  246. file = new File(scriptsFolder, name + ".groovy");
  247. }
  248. if (file.exists()) {
  249. // read the script
  250. String content = FileUtils.readContent(file, "\n");
  251. scripts.put(name, content);
  252. } else {
  253. // missing script?!
  254. logger.warn(MessageFormat.format("Failed to find push script \"{0}\"", name));
  255. }
  256. }
  257. result = scripts;
  258. }
  259. }
  260. // send the result of the request
  261. serialize(response, result);
  262. }
  263. }