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.

RpcServlet.java 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  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.IOException;
  18. import java.text.MessageFormat;
  19. import java.util.ArrayList;
  20. import java.util.Collection;
  21. import java.util.HashMap;
  22. import java.util.List;
  23. import java.util.Map;
  24. import com.google.inject.Inject;
  25. import com.google.inject.Singleton;
  26. import javax.servlet.ServletException;
  27. import javax.servlet.http.HttpServletRequest;
  28. import javax.servlet.http.HttpServletResponse;
  29. import org.eclipse.jgit.lib.Repository;
  30. import com.gitblit.Constants;
  31. import com.gitblit.Constants.RpcRequest;
  32. import com.gitblit.GitBlitException;
  33. import com.gitblit.IStoredSettings;
  34. import com.gitblit.Keys;
  35. import com.gitblit.manager.IGitblit;
  36. import com.gitblit.models.RefModel;
  37. import com.gitblit.models.RegistrantAccessPermission;
  38. import com.gitblit.models.RepositoryModel;
  39. import com.gitblit.models.ServerSettings;
  40. import com.gitblit.models.TeamModel;
  41. import com.gitblit.models.UserModel;
  42. import com.gitblit.utils.DeepCopier;
  43. import com.gitblit.utils.HttpUtils;
  44. import com.gitblit.utils.JGitUtils;
  45. import com.gitblit.utils.RpcUtils;
  46. import com.gitblit.utils.StringUtils;
  47. /**
  48. * Handles remote procedure calls.
  49. *
  50. * @author James Moger
  51. */
  52. @Singleton
  53. public class RpcServlet extends JsonServlet {
  54. private static final long serialVersionUID = 1L;
  55. public static final int PROTOCOL_VERSION = 8;
  56. private IStoredSettings settings;
  57. private IGitblit gitblit;
  58. @Inject
  59. public RpcServlet(IStoredSettings settings, IGitblit gitblit) {
  60. this.settings = settings;
  61. this.gitblit = gitblit;
  62. }
  63. /**
  64. * Processes an rpc request.
  65. *
  66. * @param request
  67. * @param response
  68. * @throws javax.servlet.ServletException
  69. * @throws java.io.IOException
  70. */
  71. @Override
  72. protected void processRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException,
  73. IOException {
  74. RpcRequest reqType = RpcRequest.fromName(request.getParameter("req"));
  75. String objectName = request.getParameter("name");
  76. logger.info(MessageFormat.format("Rpc {0} request from {1}", reqType, request.getRemoteAddr()));
  77. UserModel user = (UserModel) request.getUserPrincipal();
  78. boolean allowManagement = user != null && user.canAdmin()
  79. && settings.getBoolean(Keys.web.enableRpcManagement, false);
  80. boolean allowAdmin = user != null && user.canAdmin()
  81. && settings.getBoolean(Keys.web.enableRpcAdministration, false);
  82. Object result = null;
  83. if (RpcRequest.GET_PROTOCOL.equals(reqType)) {
  84. // Return the protocol version
  85. result = PROTOCOL_VERSION;
  86. } else if (RpcRequest.LIST_REPOSITORIES.equals(reqType)) {
  87. // Determine the Gitblit clone url
  88. String gitblitUrl = settings.getString(Keys.web.canonicalUrl, null);
  89. if (StringUtils.isEmpty(gitblitUrl)) {
  90. gitblitUrl = HttpUtils.getGitblitURL(request);
  91. }
  92. StringBuilder sb = new StringBuilder();
  93. sb.append(gitblitUrl);
  94. sb.append(Constants.R_PATH);
  95. sb.append("{0}");
  96. String cloneUrl = sb.toString();
  97. // list repositories
  98. List<RepositoryModel> list = gitblit.getRepositoryModels(user);
  99. Map<String, RepositoryModel> repositories = new HashMap<String, RepositoryModel>();
  100. for (RepositoryModel model : list) {
  101. String url = MessageFormat.format(cloneUrl, model.name);
  102. repositories.put(url, model);
  103. }
  104. result = repositories;
  105. } else if (RpcRequest.LIST_BRANCHES.equals(reqType)) {
  106. // list all local branches in all repositories accessible to user
  107. Map<String, List<String>> localBranches = new HashMap<String, List<String>>();
  108. List<RepositoryModel> models = gitblit.getRepositoryModels(user);
  109. for (RepositoryModel model : models) {
  110. if (!model.hasCommits) {
  111. // skip empty repository
  112. continue;
  113. }
  114. if (model.isCollectingGarbage) {
  115. // skip garbage collecting repository
  116. logger.warn(MessageFormat.format("Temporarily excluding {0} from RPC, busy collecting garbage",
  117. model.name));
  118. continue;
  119. }
  120. // get local branches
  121. Repository repository = gitblit.getRepository(model.name);
  122. List<RefModel> refs = JGitUtils.getLocalBranches(repository, false, -1);
  123. if (model.showRemoteBranches) {
  124. // add remote branches if repository displays them
  125. refs.addAll(JGitUtils.getRemoteBranches(repository, false, -1));
  126. }
  127. if (refs.size() > 0) {
  128. List<String> branches = new ArrayList<String>();
  129. for (RefModel ref : refs) {
  130. branches.add(ref.getName());
  131. }
  132. localBranches.put(model.name, branches);
  133. }
  134. repository.close();
  135. }
  136. result = localBranches;
  137. } else if (RpcRequest.GET_USER.equals(reqType)) {
  138. if (StringUtils.isEmpty(objectName)) {
  139. if (UserModel.ANONYMOUS.equals(user)) {
  140. response.sendError(forbiddenCode);
  141. } else {
  142. // return the current user, reset credentials
  143. UserModel requestedUser = DeepCopier.copy(user);
  144. result = requestedUser;
  145. }
  146. } else {
  147. if (user.canAdmin() || objectName.equals(user.username)) {
  148. // return the specified user
  149. UserModel requestedUser = gitblit.getUserModel(objectName);
  150. if (requestedUser == null) {
  151. response.setStatus(failureCode);
  152. } else {
  153. result = requestedUser;
  154. }
  155. } else {
  156. response.sendError(forbiddenCode);
  157. }
  158. }
  159. } else if (RpcRequest.LIST_USERS.equals(reqType)) {
  160. // list users
  161. List<String> names = gitblit.getAllUsernames();
  162. List<UserModel> users = new ArrayList<UserModel>();
  163. for (String name : names) {
  164. users.add(gitblit.getUserModel(name));
  165. }
  166. result = users;
  167. } else if (RpcRequest.LIST_TEAMS.equals(reqType)) {
  168. // list teams
  169. List<String> names = gitblit.getAllTeamNames();
  170. List<TeamModel> teams = new ArrayList<TeamModel>();
  171. for (String name : names) {
  172. teams.add(gitblit.getTeamModel(name));
  173. }
  174. result = teams;
  175. } else if (RpcRequest.CREATE_REPOSITORY.equals(reqType)) {
  176. // create repository
  177. RepositoryModel model = deserialize(request, response, RepositoryModel.class);
  178. try {
  179. gitblit.updateRepositoryModel(model.name, model, true);
  180. } catch (GitBlitException e) {
  181. response.setStatus(failureCode);
  182. }
  183. } else if (RpcRequest.FORK_REPOSITORY.equals(reqType)) {
  184. // fork repository
  185. RepositoryModel origin = gitblit.getRepositoryModel(objectName);
  186. if (origin == null) {
  187. // failed to find repository, error is logged by the repository
  188. // manager
  189. response.setStatus(failureCode);
  190. } else {
  191. if (user == null || !user.canFork(origin)) {
  192. logger.error("User {} is not permitted to fork '{}'!", user == null ? "anonymous" : user.username,
  193. objectName);
  194. response.setStatus(failureCode);
  195. } else {
  196. try {
  197. // fork the origin
  198. RepositoryModel fork = gitblit.fork(origin, user);
  199. if (fork == null) {
  200. logger.error("Failed to fork repository '{}'!", objectName);
  201. response.setStatus(failureCode);
  202. } else {
  203. logger.info("User {} has forked '{}'!", user.username, objectName);
  204. }
  205. } catch (GitBlitException e) {
  206. response.setStatus(failureCode);
  207. }
  208. }
  209. }
  210. } else if (RpcRequest.EDIT_REPOSITORY.equals(reqType)) {
  211. // edit repository
  212. RepositoryModel model = deserialize(request, response, RepositoryModel.class);
  213. // name specifies original repository name in event of rename
  214. String repoName = objectName;
  215. if (repoName == null) {
  216. repoName = model.name;
  217. }
  218. try {
  219. gitblit.updateRepositoryModel(repoName, model, false);
  220. } catch (GitBlitException e) {
  221. response.setStatus(failureCode);
  222. }
  223. } else if (RpcRequest.DELETE_REPOSITORY.equals(reqType)) {
  224. // delete repository
  225. RepositoryModel model = deserialize(request, response, RepositoryModel.class);
  226. gitblit.deleteRepositoryModel(model);
  227. } else if (RpcRequest.CREATE_USER.equals(reqType)) {
  228. // create user
  229. UserModel model = deserialize(request, response, UserModel.class);
  230. try {
  231. gitblit.addUser(model);
  232. } catch (GitBlitException e) {
  233. response.setStatus(failureCode);
  234. }
  235. } else if (RpcRequest.EDIT_USER.equals(reqType)) {
  236. // edit user
  237. UserModel model = deserialize(request, response, UserModel.class);
  238. // name parameter specifies original user name in event of rename
  239. String username = objectName;
  240. if (username == null) {
  241. username = model.username;
  242. }
  243. try {
  244. gitblit.reviseUser(username, model);
  245. } catch (GitBlitException e) {
  246. response.setStatus(failureCode);
  247. }
  248. } else if (RpcRequest.DELETE_USER.equals(reqType)) {
  249. // delete user
  250. UserModel model = deserialize(request, response, UserModel.class);
  251. if (!gitblit.deleteUser(model.username)) {
  252. response.setStatus(failureCode);
  253. }
  254. } else if (RpcRequest.CREATE_TEAM.equals(reqType)) {
  255. // create team
  256. TeamModel model = deserialize(request, response, TeamModel.class);
  257. try {
  258. gitblit.addTeam(model);
  259. } catch (GitBlitException e) {
  260. response.setStatus(failureCode);
  261. }
  262. } else if (RpcRequest.EDIT_TEAM.equals(reqType)) {
  263. // edit team
  264. TeamModel model = deserialize(request, response, TeamModel.class);
  265. // name parameter specifies original team name in event of rename
  266. String teamname = objectName;
  267. if (teamname == null) {
  268. teamname = model.name;
  269. }
  270. try {
  271. gitblit.reviseTeam(teamname, model);
  272. } catch (GitBlitException e) {
  273. response.setStatus(failureCode);
  274. }
  275. } else if (RpcRequest.DELETE_TEAM.equals(reqType)) {
  276. // delete team
  277. TeamModel model = deserialize(request, response, TeamModel.class);
  278. if (!gitblit.deleteTeam(model.name)) {
  279. response.setStatus(failureCode);
  280. }
  281. } else if (RpcRequest.LIST_REPOSITORY_MEMBERS.equals(reqType)) {
  282. // get repository members
  283. RepositoryModel model = gitblit.getRepositoryModel(objectName);
  284. result = gitblit.getRepositoryUsers(model);
  285. } else if (RpcRequest.SET_REPOSITORY_MEMBERS.equals(reqType)) {
  286. // rejected since 1.2.0
  287. response.setStatus(failureCode);
  288. } else if (RpcRequest.LIST_REPOSITORY_MEMBER_PERMISSIONS.equals(reqType)) {
  289. // get repository member permissions
  290. RepositoryModel model = gitblit.getRepositoryModel(objectName);
  291. result = gitblit.getUserAccessPermissions(model);
  292. } else if (RpcRequest.SET_REPOSITORY_MEMBER_PERMISSIONS.equals(reqType)) {
  293. // set the repository permissions for the specified users
  294. RepositoryModel model = gitblit.getRepositoryModel(objectName);
  295. Collection<RegistrantAccessPermission> permissions = deserialize(request, response,
  296. RpcUtils.REGISTRANT_PERMISSIONS_TYPE);
  297. result = gitblit.setUserAccessPermissions(model, permissions);
  298. } else if (RpcRequest.LIST_REPOSITORY_TEAMS.equals(reqType)) {
  299. // get repository teams
  300. RepositoryModel model = gitblit.getRepositoryModel(objectName);
  301. result = gitblit.getRepositoryTeams(model);
  302. } else if (RpcRequest.SET_REPOSITORY_TEAMS.equals(reqType)) {
  303. // rejected since 1.2.0
  304. response.setStatus(failureCode);
  305. } else if (RpcRequest.LIST_REPOSITORY_TEAM_PERMISSIONS.equals(reqType)) {
  306. // get repository team permissions
  307. RepositoryModel model = gitblit.getRepositoryModel(objectName);
  308. result = gitblit.getTeamAccessPermissions(model);
  309. } else if (RpcRequest.SET_REPOSITORY_TEAM_PERMISSIONS.equals(reqType)) {
  310. // set the repository permissions for the specified teams
  311. RepositoryModel model = gitblit.getRepositoryModel(objectName);
  312. Collection<RegistrantAccessPermission> permissions = deserialize(request, response,
  313. RpcUtils.REGISTRANT_PERMISSIONS_TYPE);
  314. result = gitblit.setTeamAccessPermissions(model, permissions);
  315. } else if (RpcRequest.LIST_FEDERATION_REGISTRATIONS.equals(reqType)) {
  316. // return the list of federation registrations
  317. if (allowAdmin) {
  318. result = gitblit.getFederationRegistrations();
  319. } else {
  320. response.sendError(notAllowedCode);
  321. }
  322. } else if (RpcRequest.LIST_FEDERATION_RESULTS.equals(reqType)) {
  323. // return the list of federation result registrations
  324. if (allowAdmin && gitblit.canFederate()) {
  325. result = gitblit.getFederationResultRegistrations();
  326. } else {
  327. response.sendError(notAllowedCode);
  328. }
  329. } else if (RpcRequest.LIST_FEDERATION_PROPOSALS.equals(reqType)) {
  330. // return the list of federation proposals
  331. if (allowAdmin && gitblit.canFederate()) {
  332. result = gitblit.getPendingFederationProposals();
  333. } else {
  334. response.sendError(notAllowedCode);
  335. }
  336. } else if (RpcRequest.LIST_FEDERATION_SETS.equals(reqType)) {
  337. // return the list of federation sets
  338. if (allowAdmin && gitblit.canFederate()) {
  339. String gitblitUrl = settings.getString(Keys.web.canonicalUrl, null);
  340. if (StringUtils.isEmpty(gitblitUrl)) {
  341. gitblitUrl = HttpUtils.getGitblitURL(request);
  342. }
  343. result = gitblit.getFederationSets(gitblitUrl);
  344. } else {
  345. response.sendError(notAllowedCode);
  346. }
  347. } else if (RpcRequest.LIST_SETTINGS.equals(reqType)) {
  348. // return the server's settings
  349. ServerSettings serverSettings = gitblit.getSettingsModel();
  350. if (allowAdmin) {
  351. // return all settings
  352. result = serverSettings;
  353. } else {
  354. // anonymous users get a few settings to allow browser launching
  355. List<String> keys = new ArrayList<String>();
  356. keys.add(Keys.web.siteName);
  357. keys.add(Keys.web.mountParameters);
  358. keys.add(Keys.web.syndicationEntries);
  359. if (allowManagement) {
  360. // keys necessary for repository and/or user management
  361. keys.add(Keys.realm.minPasswordLength);
  362. keys.add(Keys.realm.passwordStorage);
  363. keys.add(Keys.federation.sets);
  364. }
  365. // build the settings
  366. ServerSettings managementSettings = new ServerSettings();
  367. for (String key : keys) {
  368. managementSettings.add(serverSettings.get(key));
  369. }
  370. if (allowManagement) {
  371. managementSettings.pushScripts = serverSettings.pushScripts;
  372. }
  373. result = managementSettings;
  374. }
  375. } else if (RpcRequest.EDIT_SETTINGS.equals(reqType)) {
  376. // update settings on the server
  377. if (allowAdmin) {
  378. Map<String, String> map = deserialize(request, response, RpcUtils.SETTINGS_TYPE);
  379. gitblit.updateSettings(map);
  380. } else {
  381. response.sendError(notAllowedCode);
  382. }
  383. } else if (RpcRequest.LIST_STATUS.equals(reqType)) {
  384. // return the server's status information
  385. if (allowAdmin) {
  386. result = gitblit.getStatus();
  387. } else {
  388. response.sendError(notAllowedCode);
  389. }
  390. } else if (RpcRequest.CLEAR_REPOSITORY_CACHE.equals(reqType)) {
  391. // clear the repository list cache
  392. if (allowManagement) {
  393. gitblit.resetRepositoryListCache();
  394. } else {
  395. response.sendError(notAllowedCode);
  396. }
  397. } else if (RpcRequest.REINDEX_TICKETS.equals(reqType)) {
  398. if (allowManagement) {
  399. if (StringUtils.isEmpty(objectName)) {
  400. // reindex all tickets
  401. gitblit.getTicketService().reindex();
  402. } else {
  403. // reindex tickets in a specific repository
  404. RepositoryModel model = gitblit.getRepositoryModel(objectName);
  405. gitblit.getTicketService().reindex(model);
  406. }
  407. } else {
  408. response.sendError(notAllowedCode);
  409. }
  410. }
  411. // send the result of the request
  412. serialize(response, result);
  413. }
  414. }