From 313c75ea72b0bdf59b91b1bb1712c01d9ce06b4f Mon Sep 17 00:00:00 2001 From: Simon Brandhof Date: Tue, 22 Jul 2014 00:02:51 +0200 Subject: Experimental fork --- server/pom.xml | 1 + .../java/org/sonar/process/ProcessWrapper.java | 19 +-- .../src/main/java/org/sonar/process/Props.java | 4 + .../main/java/org/sonar/search/ElasticSearch.java | 26 +--- .../java/org/sonar/search/ElasticSearchTest.java | 34 +---- server/sonar-server-app/pom.xml | 61 ++++++++ .../main/java/org/sonar/server/app/Connectors.java | 167 +++++++++++++++++++++ .../java/org/sonar/server/app/EmbeddedTomcat.java | 130 ++++++++++++++++ .../src/main/java/org/sonar/server/app/Env.java | 69 +++++++++ .../main/java/org/sonar/server/app/Logging.java | 81 ++++++++++ .../java/org/sonar/server/app/NullJarScanner.java | 36 +++++ .../java/org/sonar/server/app/ServerProcess.java | 52 +++++++ .../src/main/java/org/sonar/server/app/Webapp.java | 77 ++++++++++ server/sonar-server/pom.xml | 27 +--- .../java/org/sonar/server/db/EmbeddedDatabase.java | 2 +- .../server/platform/DefaultServerFileSystem.java | 2 +- .../platform/PlatformServletContextListener.java | 31 ++-- .../java/org/sonar/server/platform/ServerImpl.java | 2 +- .../main/java/org/sonar/server/ui/JRubyFacade.java | 2 +- .../java/org/sonar/server/tester/ServerTester.java | 2 +- server/sonar-web/pom.xml | 13 +- 21 files changed, 729 insertions(+), 109 deletions(-) create mode 100644 server/sonar-server-app/pom.xml create mode 100644 server/sonar-server-app/src/main/java/org/sonar/server/app/Connectors.java create mode 100644 server/sonar-server-app/src/main/java/org/sonar/server/app/EmbeddedTomcat.java create mode 100644 server/sonar-server-app/src/main/java/org/sonar/server/app/Env.java create mode 100644 server/sonar-server-app/src/main/java/org/sonar/server/app/Logging.java create mode 100644 server/sonar-server-app/src/main/java/org/sonar/server/app/NullJarScanner.java create mode 100644 server/sonar-server-app/src/main/java/org/sonar/server/app/ServerProcess.java create mode 100644 server/sonar-server-app/src/main/java/org/sonar/server/app/Webapp.java (limited to 'server') diff --git a/server/pom.xml b/server/pom.xml index 131d3efcbcd..b26febac729 100644 --- a/server/pom.xml +++ b/server/pom.xml @@ -14,6 +14,7 @@ sonar-process sonar-search sonar-server + sonar-server-app sonar-web sonar-ws-client diff --git a/server/sonar-process/src/main/java/org/sonar/process/ProcessWrapper.java b/server/sonar-process/src/main/java/org/sonar/process/ProcessWrapper.java index 2ec28711b70..0f0f8ff2fd0 100644 --- a/server/sonar-process/src/main/java/org/sonar/process/ProcessWrapper.java +++ b/server/sonar-process/src/main/java/org/sonar/process/ProcessWrapper.java @@ -103,8 +103,8 @@ public class ProcessWrapper extends Thread { LOGGER.info("Try #{} to connect to JMX server for process '{}'", i, name); try { String protocol = "rmi"; - String path = "/jndi/rmi://" + InetAddress.getLocalHost().getHostName() + ":" + port + "/jmxrmi"; - JMXServiceURL jmxUrl = new JMXServiceURL(protocol, InetAddress.getLocalHost().getHostName(), port, path); + String path = "/jndi/rmi://" + InetAddress.getLocalHost().getHostAddress() + ":" + port + "/jmxrmi"; + JMXServiceURL jmxUrl = new JMXServiceURL(protocol, InetAddress.getLocalHost().getHostAddress(), port, path); JMXConnector jmxConnector = JMXConnectorFactory.connect(jmxUrl, null); MBeanServerConnection mBeanServer = jmxConnector.getMBeanServerConnection(); ProcessMXBean bean = JMX.newMBeanProxy(mBeanServer, Process.objectNameFor(name), ProcessMXBean.class); @@ -147,7 +147,7 @@ public class ProcessWrapper extends Thread { } private List getJMXOptions() { - return ImmutableList.of( + return ImmutableList.of( "-Dcom.sun.management.jmxremote", "-Dcom.sun.management.jmxremote.port=" + this.port, "-Dcom.sun.management.jmxremote.authenticate=false", @@ -155,18 +155,13 @@ public class ProcessWrapper extends Thread { } private List getClassPath() { - String separator = System.getProperty("file.separator"); - return ImmutableList.of( - "-cp", - StringUtils.join(classPath, separator)); + // java specification : "multiple path entries are separated by semi-colons", not by + // system file separator, + return ImmutableList.of("-cp", StringUtils.join(classPath, ";")); } private String getPropertyFile() { File propertyFile = new File(FileUtils.getTempDirectory(), UUID.randomUUID().toString()); -// if (!propertyFile.canWrite()) { -// throw new IllegalStateException("Cannot write temp propertyFile to '" + -// propertyFile.getAbsolutePath() + "'"); -// } try { Properties props = new Properties(); for (Map.Entry property : properties.entrySet()) { @@ -189,6 +184,7 @@ public class ProcessWrapper extends Thread { LOGGER.info("ProcessWrapper::executeProcess() START"); ProcessBuilder processBuilder = new ProcessBuilder(); + processBuilder.environment().put("SONAR_HOME", workDir); processBuilder.command().add(getJavaCommand()); if (!StringUtils.isEmpty(javaOpts)) { @@ -199,6 +195,7 @@ public class ProcessWrapper extends Thread { } processBuilder.command().addAll(getJMXOptions()); processBuilder.command().addAll(getClassPath()); + processBuilder.command().add(className); processBuilder.command().add(getPropertyFile()); diff --git a/server/sonar-process/src/main/java/org/sonar/process/Props.java b/server/sonar-process/src/main/java/org/sonar/process/Props.java index 0eb24a37ba5..cd752403a86 100644 --- a/server/sonar-process/src/main/java/org/sonar/process/Props.java +++ b/server/sonar-process/src/main/java/org/sonar/process/Props.java @@ -67,6 +67,10 @@ public class Props { return i == null ? defaultValue : i; } + public Properties properties() { + return props; + } + public static Props create(Properties properties) { Properties p = new Properties(); diff --git a/server/sonar-search/src/main/java/org/sonar/search/ElasticSearch.java b/server/sonar-search/src/main/java/org/sonar/search/ElasticSearch.java index 5c44678b51c..a2e3bf42680 100644 --- a/server/sonar-search/src/main/java/org/sonar/search/ElasticSearch.java +++ b/server/sonar-search/src/main/java/org/sonar/search/ElasticSearch.java @@ -25,23 +25,18 @@ import org.elasticsearch.common.settings.ImmutableSettings; import org.elasticsearch.common.unit.TimeValue; import org.elasticsearch.node.Node; import org.elasticsearch.node.NodeBuilder; -import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.sonar.process.Process; import org.sonar.process.Props; import org.sonar.search.script.ListUpdate; -public class ElasticSearch extends Process { +import java.io.File; - private static final Logger LOGGER = LoggerFactory.getLogger(ElasticSearch.class); +public class ElasticSearch extends Process { public static final String ES_DEBUG_PROPERTY = "esDebug"; - public static final String ES_PORT_PROPERTY = "esPort"; + public static final String ES_PORT_PROPERTY = "sonar.es.node.port"; public static final String ES_CLUSTER_PROPERTY = "esCluster"; - public static final String ES_HOME_PROPERTY = "esHome"; - - public static final String MISSING_ES_PORT = "Missing ES port Argument"; - public static final String MISSING_ES_HOME = "Missing ES home directory Argument"; public static final String DEFAULT_CLUSTER_NAME = "sonarqube"; @@ -72,22 +67,13 @@ public class ElasticSearch extends Process { @Override public void onStart() { - String home = props.of(ES_HOME_PROPERTY); - if (home == null) { - throw new IllegalStateException(MISSING_ES_HOME); - } - + String dataDir = props.of("sonar.path.data"); Integer port = props.intOf(ES_PORT_PROPERTY); - if (port == null) { - throw new IllegalStateException(MISSING_ES_PORT); - } - String clusterName = props.of(ES_CLUSTER_PROPERTY, DEFAULT_CLUSTER_NAME); - LOGGER.info("Starting ES[{}] on port: {}", clusterName, port); + LoggerFactory.getLogger(ElasticSearch.class).info("Starting ES[{}] on port: {}", clusterName, port); ImmutableSettings.Builder esSettings = ImmutableSettings.settingsBuilder() - .put("es.foreground", "yes") .put("discovery.zen.ping.multicast.enabled", "false") @@ -108,7 +94,7 @@ public class ElasticSearch extends Process { .put("node.data", true) .put("node.local", false) .put("transport.tcp.port", port) - .put("path.home", home); + .put("path.data", new File(dataDir, "es").getAbsolutePath()); if (props.booleanOf(ES_DEBUG_PROPERTY, false)) { esSettings diff --git a/server/sonar-search/src/test/java/org/sonar/search/ElasticSearchTest.java b/server/sonar-search/src/test/java/org/sonar/search/ElasticSearchTest.java index 743b2bc427b..d06de0fac40 100644 --- a/server/sonar-search/src/test/java/org/sonar/search/ElasticSearchTest.java +++ b/server/sonar-search/src/test/java/org/sonar/search/ElasticSearchTest.java @@ -79,42 +79,12 @@ public class ElasticSearchTest { } } - - @Test - public void missing_properties() throws IOException, MBeanRegistrationException, InstanceNotFoundException { - - Properties properties = new Properties(); - properties.setProperty(Process.SONAR_HOME, FileUtils.getTempDirectoryPath()); - properties.setProperty(Process.NAME_PROPERTY, "ES"); - properties.setProperty(Process.PORT_PROPERTY, Integer.toString(freePort)); - - try { - elasticSearch = new ElasticSearch(Props.create(properties)); - } catch (Exception e) { - assertThat(e.getMessage()).isEqualTo(ElasticSearch.MISSING_ES_HOME); - } - - properties.setProperty(ElasticSearch.ES_HOME_PROPERTY, tempDirectory.getAbsolutePath()); - try { - resetMBeanServer(); - elasticSearch = new ElasticSearch(Props.create(properties)); - } catch (Exception e) { - assertThat(e.getMessage()).isEqualTo(ElasticSearch.MISSING_ES_PORT); - } - resetMBeanServer(); - - properties.setProperty(ElasticSearch.ES_PORT_PROPERTY, Integer.toString(freeESPort)); - elasticSearch = new ElasticSearch(Props.create(properties)); - assertThat(elasticSearch).isNotNull(); - } - @Test public void can_connect() throws SocketException { - Properties properties = new Properties(); properties.setProperty(Process.SONAR_HOME, FileUtils.getTempDirectoryPath()); properties.setProperty(Process.NAME_PROPERTY, "ES"); - properties.setProperty(ElasticSearch.ES_HOME_PROPERTY, tempDirectory.getAbsolutePath()); + properties.setProperty("sonar.path.data", tempDirectory.getAbsolutePath()); properties.setProperty(ElasticSearch.ES_PORT_PROPERTY, Integer.toString(freeESPort)); elasticSearch = new ElasticSearch(Props.create(properties)); @@ -157,4 +127,4 @@ public class ElasticSearchTest { assertThat(e.getMessage()).isEqualTo("No node available"); } } -} \ No newline at end of file +} diff --git a/server/sonar-server-app/pom.xml b/server/sonar-server-app/pom.xml new file mode 100644 index 00000000000..d21903df63f --- /dev/null +++ b/server/sonar-server-app/pom.xml @@ -0,0 +1,61 @@ + + + 4.0.0 + + org.codehaus.sonar + server + 4.5-SNAPSHOT + .. + + sonar-server-app + jar + SonarQube :: Server :: App + Temporary bootstrapper of SonarQube web server + + + + org.codehaus.sonar + sonar-process + ${project.version} + + + org.slf4j + slf4j-api + + + org.slf4j + jul-to-slf4j + + + ch.qos.logback + logback-access + + + ch.qos.logback + logback-classic + + + ch.qos.logback + logback-core + + + org.apache.tomcat.embed + tomcat-embed-core + + + org.apache.tomcat.embed + tomcat-embed-jasper + + + org.apache.tomcat.embed + tomcat-embed-logging-juli + + + + com.google.code.findbugs + jsr305 + provided + + + diff --git a/server/sonar-server-app/src/main/java/org/sonar/server/app/Connectors.java b/server/sonar-server-app/src/main/java/org/sonar/server/app/Connectors.java new file mode 100644 index 00000000000..fa14a25ceaf --- /dev/null +++ b/server/sonar-server-app/src/main/java/org/sonar/server/app/Connectors.java @@ -0,0 +1,167 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2014 SonarSource + * mailto:contact AT sonarsource DOT com + * + * SonarQube is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * SonarQube is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package org.sonar.server.app; + +import org.apache.catalina.connector.Connector; +import org.apache.catalina.startup.Tomcat; +import org.slf4j.LoggerFactory; +import org.sonar.process.Props; + +import javax.annotation.Nullable; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +class Connectors { + + private static final int DISABLED_PORT = -1; + static final String HTTP_PROTOCOL = "HTTP/1.1"; + static final String AJP_PROTOCOL = "AJP/1.3"; + + static void configure(Tomcat tomcat, Props props) { + configureShutdown(tomcat, props); + configureConnectors(tomcat, props); + } + + private static void configureConnectors(Tomcat tomcat, Props props) { + List connectors = new ArrayList(); + connectors.addAll(Arrays.asList(newHttpConnector(props), newAjpConnector(props), newHttpsConnector(props))); + connectors.removeAll(Collections.singleton(null)); + + verify(connectors); + + tomcat.setConnector(connectors.get(0)); + for (Connector connector : connectors) { + tomcat.getService().addConnector(connector); + } + } + + private static void verify(List connectors) { + if (connectors.isEmpty()) { + throw new IllegalStateException("HTTP connectors are disabled"); + } + Set ports = new HashSet(); + for (Connector connector : connectors) { + int port = connector.getPort(); + if (ports.contains(port)) { + throw new IllegalStateException(String.format("HTTP, AJP and HTTPS must not use the same port %d", port)); + } + ports.add(port); + } + } + + private static void configureShutdown(Tomcat tomcat, Props props) { + String shutdownToken = props.of("sonar.web.shutdown.token"); + Integer shutdownPort = props.intOf("sonar.web.shutdown.port"); + if (shutdownToken != null && !"".equals(shutdownToken) && shutdownPort != null) { + tomcat.getServer().setPort(shutdownPort); + tomcat.getServer().setShutdown(shutdownToken); + info("Shutdown command is enabled on port " + shutdownPort); + } + } + + @Nullable + private static Connector newHttpConnector(Props props) { + Connector connector = null; + // Not named "sonar.web.http.port" to keep backward-compatibility + int port = props.intOf("sonar.web.port", 9000); + if (port > DISABLED_PORT) { + connector = newConnector(props, HTTP_PROTOCOL, "http"); + connector.setPort(port); + info("HTTP connector is enabled on port " + port); + } + return connector; + } + + @Nullable + private static Connector newAjpConnector(Props props) { + Connector connector = null; + int port = props.intOf("sonar.ajp.port", DISABLED_PORT); + if (port > DISABLED_PORT) { + connector = newConnector(props, AJP_PROTOCOL, "http"); + connector.setPort(port); + info("AJP connector is enabled on port " + port); + } + return connector; + } + + @Nullable + private static Connector newHttpsConnector(Props props) { + Connector connector = null; + int port = props.intOf("sonar.web.https.port", DISABLED_PORT); + if (port > DISABLED_PORT) { + connector = newConnector(props, HTTP_PROTOCOL, "https"); + connector.setPort(port); + connector.setSecure(true); + connector.setScheme("https"); + setConnectorAttribute(connector, "keyAlias", props.of("sonar.web.https.keyAlias")); + String keyPassword = props.of("sonar.web.https.keyPass", "changeit"); + setConnectorAttribute(connector, "keyPass", keyPassword); + setConnectorAttribute(connector, "keystorePass", props.of("sonar.web.https.keystorePass", keyPassword)); + setConnectorAttribute(connector, "keystoreFile", props.of("sonar.web.https.keystoreFile")); + setConnectorAttribute(connector, "keystoreType", props.of("sonar.web.https.keystoreType", "JKS")); + setConnectorAttribute(connector, "keystoreProvider", props.of("sonar.web.https.keystoreProvider")); + setConnectorAttribute(connector, "truststorePass", props.of("sonar.web.https.truststorePass", "changeit")); + setConnectorAttribute(connector, "truststoreFile", props.of("sonar.web.https.truststoreFile")); + setConnectorAttribute(connector, "truststoreType", props.of("sonar.web.https.truststoreType", "JKS")); + setConnectorAttribute(connector, "truststoreProvider", props.of("sonar.web.https.truststoreProvider")); + setConnectorAttribute(connector, "clientAuth", props.of("sonar.web.https.clientAuth", "false")); + setConnectorAttribute(connector, "sslProtocol", "TLS"); + setConnectorAttribute(connector, "SSLEnabled", true); + info("HTTPS connector is enabled on port " + port); + } + return connector; + } + + private static Connector newConnector(Props props, String protocol, String scheme) { + Connector connector = new Connector(protocol); + connector.setURIEncoding("UTF-8"); + connector.setProperty("address", props.of("sonar.web.host", "0.0.0.0")); + configurePool(props, connector, scheme); + configureCompression(connector); + return connector; + } + + private static void configurePool(Props props, Connector connector, String scheme) { + connector.setProperty("acceptorThreadCount", String.valueOf(2)); + connector.setProperty("minSpareThreads", String.valueOf(props.intOf("sonar.web." + scheme + ".minThreads", 5))); + connector.setProperty("maxThreads", String.valueOf(props.intOf("sonar.web." + scheme + ".maxThreads", 50))); + connector.setProperty("acceptCount", String.valueOf(props.intOf("sonar.web." + scheme + ".acceptCount", 25))); + } + + private static void configureCompression(Connector connector) { + connector.setProperty("compression", "on"); + connector.setProperty("compressionMinSize", "1024"); + connector.setProperty("compressableMimeType", "text/html,text/xml,text/plain,text/css,application/json,application/javascript"); + } + + private static void setConnectorAttribute(Connector c, String key, @Nullable Object value) { + if (value != null) { + c.setAttribute(key, value); + } + } + + private static void info(String message) { + LoggerFactory.getLogger(Connectors.class).info(message); + } +} diff --git a/server/sonar-server-app/src/main/java/org/sonar/server/app/EmbeddedTomcat.java b/server/sonar-server-app/src/main/java/org/sonar/server/app/EmbeddedTomcat.java new file mode 100644 index 00000000000..971414b10dd --- /dev/null +++ b/server/sonar-server-app/src/main/java/org/sonar/server/app/EmbeddedTomcat.java @@ -0,0 +1,130 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2014 SonarSource + * mailto:contact AT sonarsource DOT com + * + * SonarQube is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * SonarQube is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package org.sonar.server.app; + +import org.apache.catalina.LifecycleException; +import org.apache.catalina.connector.Connector; +import org.apache.catalina.startup.Tomcat; +import org.apache.commons.io.FileUtils; + +import java.io.File; + +class EmbeddedTomcat { + + public static final String TEMP_RELATIVE_PATH = "temp/tomcat"; + + private final Env env; + private Tomcat tomcat = null; + private Thread hook = null; + private boolean stopping = false, ready = false; + + EmbeddedTomcat(Env env) { + this.env = env; + } + + void start() { + if (tomcat != null || hook != null) { + throw new IllegalStateException("Server is already started"); + } + + try { + // '%2F' (slash /) and '%5C' (backslash \) are permitted as path delimiters in URLs + // See Ruby on Rails url_for + System.setProperty("org.apache.tomcat.util.buf.UDecoder.ALLOW_ENCODED_SLASH", "true"); + + System.setProperty("org.apache.catalina.startup.EXIT_ON_INIT_FAILURE", "true"); + + tomcat = new Tomcat(); + + // Initialize directories + String basedir = env.freshDir(TEMP_RELATIVE_PATH).getCanonicalPath(); + tomcat.setBaseDir(basedir); + tomcat.getHost().setAppBase(basedir); + tomcat.getHost().setAutoDeploy(false); + tomcat.getHost().setCreateDirs(false); + tomcat.getHost().setDeployOnStartup(true); + + Logging.configure(tomcat, env, env.props()); + Connectors.configure(tomcat, env.props()); + Webapp.configure(tomcat, env, env.props()); + tomcat.start(); + addShutdownHook(); + ready = true; + tomcat.getServer().await(); + } catch (Exception e) { + throw new IllegalStateException("Fail to start web server", e); + } + // Shutdown command received + stop(); + } + + private void addShutdownHook() { + hook = new Thread() { + @Override + public void run() { + EmbeddedTomcat.this.doStop(); + } + }; + Runtime.getRuntime().addShutdownHook(hook); + } + + + void stop() { + removeShutdownHook(); + doStop(); + } + + private synchronized void doStop() { + try { + if (tomcat != null && !stopping) { + stopping = true; + tomcat.stop(); + tomcat.destroy(); + } + tomcat = null; + stopping = false; + ready = false; + File tempDir = env.file(TEMP_RELATIVE_PATH); + FileUtils.deleteQuietly(tempDir); + + } catch (LifecycleException e) { + throw new IllegalStateException("Fail to stop web server", e); + } + } + + private void removeShutdownHook() { + if (hook != null && !hook.isAlive()) { + Runtime.getRuntime().removeShutdownHook(hook); + hook = null; + } + } + + boolean isReady( ){ + return ready; + } + + int port() { + Connector[] connectors = tomcat.getService().findConnectors(); + if (connectors.length > 0) { + return connectors[0].getLocalPort(); + } + return -1; + } +} diff --git a/server/sonar-server-app/src/main/java/org/sonar/server/app/Env.java b/server/sonar-server-app/src/main/java/org/sonar/server/app/Env.java new file mode 100644 index 00000000000..885fbbad7b1 --- /dev/null +++ b/server/sonar-server-app/src/main/java/org/sonar/server/app/Env.java @@ -0,0 +1,69 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2014 SonarSource + * mailto:contact AT sonarsource DOT com + * + * SonarQube is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * SonarQube is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package org.sonar.server.app; + +import org.apache.commons.io.FileUtils; +import org.sonar.process.Props; + +import java.io.File; +import java.io.IOException; + +class Env { + + private final Props props; + + Env(Props props) { + this.props = props; + } + + Props props() { + return props; + } + + File rootDir() { + return new File(props.of("sonar.path.home")); + } + + File file(String relativePath) { + return new File(rootDir(), relativePath); + } + + File freshDir(String relativePath) { + File dir = new File(rootDir(), relativePath); + FileUtils.deleteQuietly(dir); + dir.mkdirs(); + return dir; + } + + /** + * This check is required in order to provide more meaningful message than JRuby - see SONAR-2715 + */ + void verifyWritableTempDir() { + File file = null; + try { + file = File.createTempFile("sonarqube-check", "tmp"); + } catch (IOException e) { + throw new IllegalStateException("Unable to create file in temporary directory, please check existence " + + "and permissions of: " + FileUtils.getTempDirectory(), e); + } finally { + FileUtils.deleteQuietly(file); + } + } +} diff --git a/server/sonar-server-app/src/main/java/org/sonar/server/app/Logging.java b/server/sonar-server-app/src/main/java/org/sonar/server/app/Logging.java new file mode 100644 index 00000000000..45c4afefeb7 --- /dev/null +++ b/server/sonar-server-app/src/main/java/org/sonar/server/app/Logging.java @@ -0,0 +1,81 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2014 SonarSource + * mailto:contact AT sonarsource DOT com + * + * SonarQube is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * SonarQube is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package org.sonar.server.app; + +import ch.qos.logback.access.tomcat.LogbackValve; +import org.apache.catalina.LifecycleEvent; +import org.apache.catalina.LifecycleListener; +import org.apache.catalina.startup.Tomcat; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.slf4j.bridge.SLF4JBridgeHandler; +import org.sonar.process.Props; + +import java.util.logging.LogManager; + +class Logging { + + static final String ACCESS_RELATIVE_PATH = "web/WEB-INF/config/logback-access.xml"; + static final String PROPERTY_ENABLE_ACCESS_LOGS = "sonar.web.accessLogs.enable"; + + static void init() { + // Configure java.util.logging, used by Tomcat, in order to forward to slf4j + LogManager.getLogManager().reset(); + SLF4JBridgeHandler.install(); + } + + static void configure(Tomcat tomcat, Env env, Props props) { + tomcat.setSilent(false); + tomcat.getService().addLifecycleListener(new LifecycleLogger(console())); + configureLogbackAccess(tomcat, env, props); + } + + static Logger console() { + return LoggerFactory.getLogger("console"); + } + + private static void configureLogbackAccess(Tomcat tomcat, Env env, Props props) { + if (props.booleanOf(PROPERTY_ENABLE_ACCESS_LOGS, true)) { + LogbackValve valve = new LogbackValve(); + valve.setQuiet(true); + valve.setFilename(env.file(ACCESS_RELATIVE_PATH).getAbsolutePath()); + tomcat.getHost().getPipeline().addValve(valve); + } + } + + static class LifecycleLogger implements LifecycleListener { + private Logger logger; + + LifecycleLogger(Logger logger) { + this.logger = logger; + } + + @Override + public void lifecycleEvent(LifecycleEvent event) { + if ("after_start".equals(event.getType())) { + logger.info("Web server is started"); + + } else if ("after_destroy".equals(event.getType())) { + logger.info("Web server is stopped"); + } + } + } + +} diff --git a/server/sonar-server-app/src/main/java/org/sonar/server/app/NullJarScanner.java b/server/sonar-server-app/src/main/java/org/sonar/server/app/NullJarScanner.java new file mode 100644 index 00000000000..4f8ac9e5312 --- /dev/null +++ b/server/sonar-server-app/src/main/java/org/sonar/server/app/NullJarScanner.java @@ -0,0 +1,36 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2014 SonarSource + * mailto:contact AT sonarsource DOT com + * + * SonarQube is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * SonarQube is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package org.sonar.server.app; + +import org.apache.tomcat.JarScanner; +import org.apache.tomcat.JarScannerCallback; + +import javax.servlet.ServletContext; +import java.util.Set; + +/** + * Disable taglib and web-fragment.xml scanning of Tomcat. Should speed up startup. + */ +class NullJarScanner implements JarScanner { + @Override + public void scan(ServletContext context, ClassLoader classloader, JarScannerCallback callback, Set jarsToSkip) { + // doing nothing is fast! + } +} diff --git a/server/sonar-server-app/src/main/java/org/sonar/server/app/ServerProcess.java b/server/sonar-server-app/src/main/java/org/sonar/server/app/ServerProcess.java new file mode 100644 index 00000000000..287176a828d --- /dev/null +++ b/server/sonar-server-app/src/main/java/org/sonar/server/app/ServerProcess.java @@ -0,0 +1,52 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2014 SonarSource + * mailto:contact AT sonarsource DOT com + * + * SonarQube is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * SonarQube is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package org.sonar.server.app; + +public class ServerProcess extends org.sonar.process.Process { + + private final EmbeddedTomcat tomcat; + + public ServerProcess(String[] args) { + super(args); + Logging.init(); + Env env = new Env(props); + env.verifyWritableTempDir(); + this.tomcat = new EmbeddedTomcat(env); + } + + @Override + public void onStart() { + tomcat.start(); + } + + @Override + public void onTerminate() { + tomcat.stop(); + } + + @Override + public boolean isReady() { + return tomcat.isReady(); + } + + public static void main(String[] args) { + new ServerProcess(args).start(); + } +} diff --git a/server/sonar-server-app/src/main/java/org/sonar/server/app/Webapp.java b/server/sonar-server-app/src/main/java/org/sonar/server/app/Webapp.java new file mode 100644 index 00000000000..e20500f15bd --- /dev/null +++ b/server/sonar-server-app/src/main/java/org/sonar/server/app/Webapp.java @@ -0,0 +1,77 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2014 SonarSource + * mailto:contact AT sonarsource DOT com + * + * SonarQube is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * SonarQube is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package org.sonar.server.app; + +import org.apache.catalina.Context; +import org.apache.catalina.startup.Tomcat; +import org.slf4j.LoggerFactory; +import org.sonar.process.Props; + +import java.util.Map; + +class Webapp { + + private static final String JRUBY_MAX_RUNTIMES = "jruby.max.runtimes"; + private static final String RAILS_ENV = "rails.env"; + private static final String PROPERTY_CONTEXT = "sonar.web.context"; + private static final String PROPERTY_LOG_PROFILING_LEVEL = "sonar.log.profilingLevel"; + private static final String PROPERTY_LOG_CONSOLE = "sonar.log.console"; + + static void configure(Tomcat tomcat, Env env, Props props) { + try { + Context context = tomcat.addWebapp(getContextPath(props), env.file("web").getAbsolutePath()); + context.setConfigFile(env.file("web/META-INF/context.xml").toURI().toURL()); + context.addParameter(PROPERTY_LOG_PROFILING_LEVEL, props.of(PROPERTY_LOG_PROFILING_LEVEL, "NONE")); + context.addParameter(PROPERTY_LOG_CONSOLE, props.of(PROPERTY_LOG_CONSOLE, "false")); + for (Map.Entry entry : props.properties().entrySet()) { + String key = entry.getKey().toString(); + if (key.startsWith("sonar.")) { + context.addParameter(key, entry.getValue().toString()); + } + } + configureRailsMode(props, context); + context.setJarScanner(new NullJarScanner()); + + } catch (Exception e) { + throw new IllegalStateException("Fail to configure webapp", e); + } + } + + static String getContextPath(Props props) { + String context = props.of(PROPERTY_CONTEXT, ""); + if ("/".equals(context)) { + context = ""; + } else if (!"".equals(context) && !context.startsWith("/")) { + throw new IllegalStateException(String.format("Value of '%s' must start with a forward slash: '%s'", PROPERTY_CONTEXT, context)); + } + return context; + } + + static void configureRailsMode(Props props, Context context) { + if (props.booleanOf("sonar.rails.dev")) { + context.addParameter(RAILS_ENV, "development"); + context.addParameter(JRUBY_MAX_RUNTIMES, "3"); + LoggerFactory.getLogger(Webapp.class).warn("\n\n\n------ RAILS DEVELOPMENT MODE IS ENABLED ------\n\n\n"); + } else { + context.addParameter(RAILS_ENV, "production"); + context.addParameter(JRUBY_MAX_RUNTIMES, "1"); + } + } +} diff --git a/server/sonar-server/pom.xml b/server/sonar-server/pom.xml index e462e45623d..1665e581eb8 100644 --- a/server/sonar-server/pom.xml +++ b/server/sonar-server/pom.xml @@ -151,32 +151,9 @@ org.apache.tomcat.embed tomcat-embed-core + 7.0.42 + provided - - org.apache.tomcat.embed - tomcat-embed-jasper - - - org.apache.tomcat.embed - tomcat-embed-logging-juli - - - ch.qos.logback - logback-access - - - ch.qos.logback - logback-classic - - - ch.qos.logback - logback-core - - - org.slf4j - jul-to-slf4j - - com.google.code.findbugs jsr305 diff --git a/server/sonar-server/src/main/java/org/sonar/server/db/EmbeddedDatabase.java b/server/sonar-server/src/main/java/org/sonar/server/db/EmbeddedDatabase.java index 2ed48e023e9..389258369af 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/db/EmbeddedDatabase.java +++ b/server/sonar-server/src/main/java/org/sonar/server/db/EmbeddedDatabase.java @@ -100,7 +100,7 @@ public class EmbeddedDatabase implements Startable { } private File getSonarHomeDataDirectory(Settings settings) { - File sonarHome = new File(settings.getString(CoreProperties.SONAR_HOME)); + File sonarHome = new File(settings.getString("sonar.path.home")); if (!sonarHome.isDirectory()) { throw new IllegalStateException("SonarQube home directory is not valid"); } diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/DefaultServerFileSystem.java b/server/sonar-server/src/main/java/org/sonar/server/platform/DefaultServerFileSystem.java index 8095d1911b6..cceb7c4cb43 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/platform/DefaultServerFileSystem.java +++ b/server/sonar-server/src/main/java/org/sonar/server/platform/DefaultServerFileSystem.java @@ -56,7 +56,7 @@ public class DefaultServerFileSystem implements ServerFileSystem, Startable { public DefaultServerFileSystem(Database database, Settings settings, Server server) { this.database = database; this.server = server; - this.homeDir = new File(settings.getString(CoreProperties.SONAR_HOME)); + this.homeDir = new File(settings.getString("sonar.path.home")); } /** diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/PlatformServletContextListener.java b/server/sonar-server/src/main/java/org/sonar/server/platform/PlatformServletContextListener.java index 82e2e659596..ab87e7db4b8 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/platform/PlatformServletContextListener.java +++ b/server/sonar-server/src/main/java/org/sonar/server/platform/PlatformServletContextListener.java @@ -24,10 +24,12 @@ import org.slf4j.LoggerFactory; import org.sonar.core.config.Logback; import org.sonar.core.profiling.Profiling; +import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; - +import java.util.Enumeration; import java.util.Map; +import java.util.Properties; import static org.apache.commons.lang.StringUtils.defaultIfEmpty; @@ -50,7 +52,16 @@ public final class PlatformServletContextListener implements ServletContextListe public void contextInitialized(ServletContextEvent event) { try { configureLogback(event); - Platform.getInstance().init(System.getProperties()); + Properties props = new Properties(); + ServletContext context = event.getServletContext(); + Enumeration paramKeys = context.getInitParameterNames(); + while (paramKeys.hasMoreElements()) { + String key = paramKeys.nextElement(); + if (key.startsWith("sonar.")) { + props.put(key, context.getInitParameter(key)); + } + } + Platform.getInstance().init(props); Platform.getInstance().doStart(); } catch (Throwable t) { // Tomcat 7 "limitations": @@ -80,19 +91,19 @@ public final class PlatformServletContextListener implements ServletContextListe */ private void configureLogback(ServletContextEvent event) { String configProfilingLevel = defaultIfEmpty( - event.getServletContext().getInitParameter(Profiling.CONFIG_PROFILING_LEVEL), - System.getProperty(Profiling.CONFIG_PROFILING_LEVEL)); + event.getServletContext().getInitParameter(Profiling.CONFIG_PROFILING_LEVEL), + System.getProperty(Profiling.CONFIG_PROFILING_LEVEL)); Profiling.Level profilingLevel = Profiling.Level.fromConfigString(configProfilingLevel); String consoleEnabled = defaultIfEmpty(defaultIfEmpty( event.getServletContext().getInitParameter(CONFIG_LOG_CONSOLE), System.getProperty(CONFIG_LOG_CONSOLE)), - // Line below used in last resort - "false"); + // Line below used in last resort + "false"); Map variables = ImmutableMap.of( - "RAILS_LOGGER_LEVEL", profilingLevel == Profiling.Level.FULL ? "DEBUG" : "WARN", - "LOGFILE_LOGGING_FORMAT", profilingLevel == Profiling.Level.FULL ? LOGFILE_FULL_LOGGING_FORMAT : LOGFILE_STANDARD_LOGGING_FORMAT, - "CONSOLE_LOGGING_FORMAT", profilingLevel == Profiling.Level.FULL ? CONSOLE_FULL_LOGGING_FORMAT : CONSOLE_STANDARD_LOGGING_FORMAT, - "CONSOLE_ENABLED", consoleEnabled); + "RAILS_LOGGER_LEVEL", profilingLevel == Profiling.Level.FULL ? "DEBUG" : "WARN", + "LOGFILE_LOGGING_FORMAT", profilingLevel == Profiling.Level.FULL ? LOGFILE_FULL_LOGGING_FORMAT : LOGFILE_STANDARD_LOGGING_FORMAT, + "CONSOLE_LOGGING_FORMAT", profilingLevel == Profiling.Level.FULL ? CONSOLE_FULL_LOGGING_FORMAT : CONSOLE_STANDARD_LOGGING_FORMAT, + "CONSOLE_ENABLED", consoleEnabled); Logback.configure("/org/sonar/server/platform/logback.xml", variables); } } diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/ServerImpl.java b/server/sonar-server/src/main/java/org/sonar/server/platform/ServerImpl.java index bcff83ded1f..f1f4a4000f2 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/platform/ServerImpl.java +++ b/server/sonar-server/src/main/java/org/sonar/server/platform/ServerImpl.java @@ -77,7 +77,7 @@ public final class ServerImpl extends Server implements Startable { // Remove trailing slashes .replaceFirst("(\\/+)$", ""); - sonarHome = new File(settings.getString(CoreProperties.SONAR_HOME)); + sonarHome = new File(settings.getString("sonar.path.home")); if (!sonarHome.isDirectory()) { throw new IllegalStateException("SonarQube home directory is not valid"); } diff --git a/server/sonar-server/src/main/java/org/sonar/server/ui/JRubyFacade.java b/server/sonar-server/src/main/java/org/sonar/server/ui/JRubyFacade.java index 4c93d8a66cb..5e7613339d1 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/ui/JRubyFacade.java +++ b/server/sonar-server/src/main/java/org/sonar/server/ui/JRubyFacade.java @@ -369,7 +369,7 @@ public final class JRubyFacade { } public String getServerHome() { - return get(Settings.class).getString(CoreProperties.SONAR_HOME); + return get(Settings.class).getString("sonar.path.home"); } public ComponentContainer getContainer() { diff --git a/server/sonar-server/src/test/java/org/sonar/server/tester/ServerTester.java b/server/sonar-server/src/test/java/org/sonar/server/tester/ServerTester.java index 8c7b6812118..b1bef3250f1 100644 --- a/server/sonar-server/src/test/java/org/sonar/server/tester/ServerTester.java +++ b/server/sonar-server/src/test/java/org/sonar/server/tester/ServerTester.java @@ -76,7 +76,7 @@ public class ServerTester extends ExternalResource { Properties properties = new Properties(); properties.putAll(initialProps); properties.setProperty(IndexProperties.TYPE, IndexProperties.ES_TYPE.MEMORY.name()); - properties.setProperty(CoreProperties.SONAR_HOME, homeDir.getAbsolutePath()); + properties.setProperty("sonar.path.home", homeDir.getAbsolutePath()); properties.setProperty(DatabaseProperties.PROP_URL, "jdbc:h2:" + homeDir.getAbsolutePath() + "/h2"); for (Map.Entry entry : System.getProperties().entrySet()) { String key = entry.getKey().toString(); diff --git a/server/sonar-web/pom.xml b/server/sonar-web/pom.xml index cc6e2043a1b..6e3ed52fcf7 100644 --- a/server/sonar-web/pom.xml +++ b/server/sonar-web/pom.xml @@ -19,6 +19,13 @@ src/main/js/third-party/**/*,src/main/js/require.js,src/main/js/tests/**/* + + + org.codehaus.sonar + sonar-server + ${project.version} + + @@ -361,12 +368,6 @@ h2 provided - - org.codehaus.sonar - sonar-server - ${project.version} - provided - org.codehaus.sonar.plugins -- cgit v1.2.3