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.

GitBlitServer.java 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  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;
  17. import java.io.BufferedReader;
  18. import java.io.File;
  19. import java.io.IOException;
  20. import java.io.InputStreamReader;
  21. import java.io.OutputStream;
  22. import java.net.InetAddress;
  23. import java.net.ServerSocket;
  24. import java.net.Socket;
  25. import java.net.URL;
  26. import java.net.UnknownHostException;
  27. import java.security.ProtectionDomain;
  28. import java.text.MessageFormat;
  29. import java.util.ArrayList;
  30. import java.util.List;
  31. import org.eclipse.jetty.server.Connector;
  32. import org.eclipse.jetty.server.Server;
  33. import org.eclipse.jetty.server.bio.SocketConnector;
  34. import org.eclipse.jetty.server.nio.SelectChannelConnector;
  35. import org.eclipse.jetty.server.session.HashSessionManager;
  36. import org.eclipse.jetty.server.ssl.SslConnector;
  37. import org.eclipse.jetty.server.ssl.SslSelectChannelConnector;
  38. import org.eclipse.jetty.server.ssl.SslSocketConnector;
  39. import org.eclipse.jetty.servlet.FilterMapping;
  40. import org.eclipse.jetty.servlet.ServletHolder;
  41. import org.eclipse.jetty.util.thread.QueuedThreadPool;
  42. import org.eclipse.jetty.webapp.WebAppContext;
  43. import org.eclipse.jgit.http.server.GitServlet;
  44. import org.slf4j.Logger;
  45. import org.slf4j.LoggerFactory;
  46. import com.beust.jcommander.JCommander;
  47. import com.beust.jcommander.Parameter;
  48. import com.beust.jcommander.ParameterException;
  49. import com.beust.jcommander.Parameters;
  50. import com.gitblit.utils.StringUtils;
  51. public class GitBlitServer {
  52. private static Logger logger;
  53. public static void main(String[] args) {
  54. Params params = new Params();
  55. JCommander jc = new JCommander(params);
  56. try {
  57. jc.parse(args);
  58. if (params.help) {
  59. usage(jc, null);
  60. }
  61. } catch (ParameterException t) {
  62. usage(jc, t);
  63. }
  64. if (params.stop) {
  65. stop(params);
  66. } else {
  67. start(params);
  68. }
  69. }
  70. private static void usage(JCommander jc, ParameterException t) {
  71. System.out.println(Constants.BORDER);
  72. System.out.println(Constants.getGitBlitVersion());
  73. System.out.println(Constants.BORDER);
  74. System.out.println();
  75. if (t != null) {
  76. System.out.println(t.getMessage());
  77. System.out.println();
  78. }
  79. if (jc != null) {
  80. jc.usage();
  81. System.out
  82. .println("\nExample:\n java -server -Xmx1024M -jar gitblit.jar --repos c:\\git --port 80 --securePort 443");
  83. }
  84. System.exit(0);
  85. }
  86. /**
  87. * Stop Server.
  88. */
  89. public static void stop(Params params) {
  90. try {
  91. Socket s = new Socket(InetAddress.getByName("127.0.0.1"), params.shutdownPort);
  92. OutputStream out = s.getOutputStream();
  93. System.out.println("Sending Shutdown Request to " + Constants.NAME);
  94. out.write("\r\n".getBytes());
  95. out.flush();
  96. s.close();
  97. } catch (UnknownHostException e) {
  98. e.printStackTrace();
  99. } catch (IOException e) {
  100. e.printStackTrace();
  101. }
  102. }
  103. /**
  104. * Start Server.
  105. */
  106. private static void start(Params params) {
  107. FileSettings settings = params.FILESETTINGS;
  108. logger = LoggerFactory.getLogger(GitBlitServer.class);
  109. logger.info(Constants.BORDER);
  110. logger.info(Constants.getGitBlitVersion());
  111. logger.info(Constants.BORDER);
  112. String osname = System.getProperty("os.name");
  113. String osversion = System.getProperty("os.version");
  114. logger.info("Running on " + osname + " (" + osversion + ")");
  115. // Determine port connectors
  116. List<Connector> connectors = new ArrayList<Connector>();
  117. if (params.port > 0) {
  118. Connector httpConnector = createConnector(params.useNIO, params.port);
  119. String bindInterface = settings.getString(Keys.server.httpBindInterface, null);
  120. if (!StringUtils.isEmpty(bindInterface)) {
  121. logger.warn(MessageFormat.format("Binding connector on port {0} to {1}",
  122. params.port, bindInterface));
  123. httpConnector.setHost(bindInterface);
  124. }
  125. connectors.add(httpConnector);
  126. }
  127. if (params.securePort > 0) {
  128. File keystore = new File("keystore");
  129. if (!keystore.exists()) {
  130. logger.info("Generating self-signed SSL certificate");
  131. MakeCertificate.generateSelfSignedCertificate("localhost", keystore,
  132. params.storePassword);
  133. }
  134. if (keystore.exists()) {
  135. Connector secureConnector = createSSLConnector(keystore, params.storePassword,
  136. params.useNIO, params.securePort);
  137. String bindInterface = settings.getString(Keys.server.httpsBindInterface, null);
  138. if (!StringUtils.isEmpty(bindInterface)) {
  139. logger.warn(MessageFormat.format("Binding ssl connector on port {0} to {1}",
  140. params.securePort, bindInterface));
  141. secureConnector.setHost(bindInterface);
  142. }
  143. connectors.add(secureConnector);
  144. } else {
  145. logger.warn("Failed to find or load Keystore?");
  146. logger.warn("SSL connector DISABLED.");
  147. }
  148. }
  149. // tempDir = Directory where...
  150. // * WebApp is expanded
  151. //
  152. File tempDir = new File(params.temp);
  153. if (tempDir.exists()) {
  154. if (!deleteRecursively(tempDir)) {
  155. logger.warn("Failed to delete temp dir " + tempDir.getAbsolutePath());
  156. }
  157. }
  158. if (!tempDir.mkdirs()) {
  159. logger.warn("Failed to create temp dir " + tempDir.getAbsolutePath());
  160. }
  161. Server server = new Server();
  162. server.setStopAtShutdown(true);
  163. server.setConnectors(connectors.toArray(new Connector[connectors.size()]));
  164. // Get the execution path of this class
  165. // We use this to set the WAR path.
  166. ProtectionDomain protectionDomain = GitBlitServer.class.getProtectionDomain();
  167. URL location = protectionDomain.getCodeSource().getLocation();
  168. // Root WebApp Context
  169. WebAppContext rootContext = new WebAppContext();
  170. rootContext.setContextPath("/");
  171. rootContext.setServer(server);
  172. rootContext.setWar(location.toExternalForm());
  173. rootContext.setTempDirectory(tempDir);
  174. // Set cookies HttpOnly so they are not accessible to JavaScript engines
  175. HashSessionManager sessionManager = new HashSessionManager();
  176. sessionManager.setHttpOnly(true);
  177. // Use secure cookies if only serving https
  178. sessionManager.setSecureCookies(params.port <= 0 && params.securePort > 0);
  179. rootContext.getSessionHandler().setSessionManager(sessionManager);
  180. // JGit Filter and Servlet
  181. String jgitPathSpec = Constants.GIT_PATH + "*";
  182. rootContext.addFilter(GitFilter.class, jgitPathSpec, FilterMapping.DEFAULT);
  183. ServletHolder jGitServlet = rootContext.addServlet(GitServlet.class, jgitPathSpec);
  184. jGitServlet.setInitParameter("base-path", params.repositoriesFolder);
  185. jGitServlet.setInitParameter("export-all", "1");
  186. // Ensure there is a defined Login Service
  187. String realmUsers = params.realmFile;
  188. if (StringUtils.isEmpty(realmUsers)) {
  189. logger.error(MessageFormat.format("PLEASE SPECIFY {0}!!", Keys.realm.realmFile));
  190. return;
  191. }
  192. // Update settings
  193. // settings.put(Keys.realm.realmFile, params.realmFile);
  194. // settings.put(Keys.git.repositoriesFolder, params.repositoriesFolder);
  195. // Set the server's contexts
  196. server.setHandler(rootContext);
  197. // Setup the GitBlit context
  198. GitBlit gitblit = GitBlit.self();
  199. gitblit.configureContext(settings);
  200. rootContext.addEventListener(gitblit);
  201. // Start the Server
  202. try {
  203. if (params.shutdownPort > 0) {
  204. Thread shutdownMonitor = new ShutdownMonitorThread(server, params);
  205. shutdownMonitor.start();
  206. }
  207. server.start();
  208. server.join();
  209. } catch (Exception e) {
  210. e.printStackTrace();
  211. System.exit(100);
  212. }
  213. }
  214. private static Connector createConnector(boolean useNIO, int port) {
  215. Connector connector;
  216. if (useNIO) {
  217. logger.info("Setting up NIO SelectChannelConnector on port " + port);
  218. SelectChannelConnector nioconn = new SelectChannelConnector();
  219. nioconn.setSoLingerTime(-1);
  220. nioconn.setThreadPool(new QueuedThreadPool(20));
  221. connector = nioconn;
  222. } else {
  223. logger.info("Setting up SocketConnector on port " + port);
  224. SocketConnector sockconn = new SocketConnector();
  225. connector = sockconn;
  226. }
  227. connector.setPort(port);
  228. connector.setMaxIdleTime(30000);
  229. return connector;
  230. }
  231. private static Connector createSSLConnector(File keystore, String password, boolean useNIO,
  232. int port) {
  233. SslConnector connector;
  234. if (useNIO) {
  235. logger.info("Setting up NIO SslSelectChannelConnector on port " + port);
  236. SslSelectChannelConnector ssl = new SslSelectChannelConnector();
  237. ssl.setSoLingerTime(-1);
  238. ssl.setThreadPool(new QueuedThreadPool(20));
  239. connector = ssl;
  240. } else {
  241. logger.info("Setting up NIO SslSocketConnector on port " + port);
  242. SslSocketConnector ssl = new SslSocketConnector();
  243. connector = ssl;
  244. }
  245. connector.setAllowRenegotiate(false);
  246. connector.setKeystore(keystore.getAbsolutePath());
  247. connector.setPassword(password);
  248. connector.setPort(port);
  249. connector.setMaxIdleTime(30000);
  250. return connector;
  251. }
  252. /**
  253. * Recursively delete a folder and its contents.
  254. *
  255. * @param folder
  256. */
  257. private static boolean deleteRecursively(File folder) {
  258. boolean deleted = true;
  259. for (File file : folder.listFiles()) {
  260. if (file.isDirectory()) {
  261. deleted &= deleteRecursively(file);
  262. } else {
  263. deleted &= file.delete();
  264. }
  265. }
  266. return deleted && folder.delete();
  267. }
  268. private static class ShutdownMonitorThread extends Thread {
  269. private final ServerSocket socket;
  270. private final Server server;
  271. private final Logger logger = LoggerFactory.getLogger(ShutdownMonitorThread.class);
  272. public ShutdownMonitorThread(Server server, Params params) {
  273. this.server = server;
  274. setDaemon(true);
  275. setName(Constants.NAME + " Shutdown Monitor");
  276. ServerSocket skt = null;
  277. try {
  278. skt = new ServerSocket(params.shutdownPort, 1, InetAddress.getByName("127.0.0.1"));
  279. } catch (Exception e) {
  280. logger.warn("Could not open shutdown monitor on port " + params.shutdownPort, e);
  281. }
  282. socket = skt;
  283. }
  284. @Override
  285. public void run() {
  286. logger.info("Shutdown Monitor listening on port " + socket.getLocalPort());
  287. Socket accept;
  288. try {
  289. accept = socket.accept();
  290. BufferedReader reader = new BufferedReader(new InputStreamReader(
  291. accept.getInputStream()));
  292. reader.readLine();
  293. logger.info(Constants.BORDER);
  294. logger.info("Stopping " + Constants.NAME);
  295. logger.info(Constants.BORDER);
  296. server.stop();
  297. server.setStopAtShutdown(false);
  298. accept.close();
  299. socket.close();
  300. } catch (Exception e) {
  301. logger.warn("Failed to shutdown Jetty", e);
  302. }
  303. }
  304. }
  305. @Parameters(separators = " ")
  306. private static class Params {
  307. private static final FileSettings FILESETTINGS = new FileSettings(Constants.PROPERTIES_FILE);
  308. /*
  309. * Server parameters
  310. */
  311. @Parameter(names = { "-h", "--help" }, description = "Show this help")
  312. public Boolean help = false;
  313. @Parameter(names = { "--stop" }, description = "Stop Server")
  314. public Boolean stop = false;
  315. @Parameter(names = { "--tempFolder" }, description = "Server temp folder")
  316. public String temp = FILESETTINGS.getString(Keys.server.tempFolder, "temp");
  317. /*
  318. * GIT Servlet Parameters
  319. */
  320. @Parameter(names = { "--repositoriesFolder" }, description = "Git Repositories Folder")
  321. public String repositoriesFolder = FILESETTINGS.getString(Keys.git.repositoriesFolder,
  322. "repos");
  323. /*
  324. * Authentication Parameters
  325. */
  326. @Parameter(names = { "--realmFile" }, description = "Users Realm Hash File")
  327. public String realmFile = FILESETTINGS.getString(Keys.realm.realmFile, "users.properties");
  328. /*
  329. * JETTY Parameters
  330. */
  331. @Parameter(names = { "--useNio" }, description = "Use NIO Connector else use Socket Connector.")
  332. public Boolean useNIO = FILESETTINGS.getBoolean(Keys.server.useNio, true);
  333. @Parameter(names = "--httpPort", description = "HTTP port for to serve. (port <= 0 will disable this connector)")
  334. public Integer port = FILESETTINGS.getInteger(Keys.server.httpPort, 80);
  335. @Parameter(names = "--httpsPort", description = "HTTPS port to serve. (port <= 0 will disable this connector)")
  336. public Integer securePort = FILESETTINGS.getInteger(Keys.server.httpsPort, 443);
  337. @Parameter(names = "--storePassword", description = "Password for SSL (https) keystore.")
  338. public String storePassword = FILESETTINGS.getString(Keys.server.storePassword, "");
  339. @Parameter(names = "--shutdownPort", description = "Port for Shutdown Monitor to listen on. (port <= 0 will disable this monitor)")
  340. public Integer shutdownPort = FILESETTINGS.getInteger(Keys.server.shutdownPort, 8081);
  341. }
  342. }