diff options
author | Simon Brandhof <simon.brandhof@gmail.com> | 2013-10-01 16:51:37 +0200 |
---|---|---|
committer | Simon Brandhof <simon.brandhof@gmail.com> | 2013-10-01 16:51:57 +0200 |
commit | af19a257160eae52f2f5a08cfb979625a9cf5cff (patch) | |
tree | 3aa8ecd06fd0f57619c98b4bbc133dab8b96aeef /sonar-application | |
parent | 3a9dfb1fd7e4c854da5eab3d041e70b27e9c709c (diff) | |
download | sonarqube-af19a257160eae52f2f5a08cfb979625a9cf5cff.tar.gz sonarqube-af19a257160eae52f2f5a08cfb979625a9cf5cff.zip |
SONAR-4675 Replace Jetty web server by Tomcat 7
Diffstat (limited to 'sonar-application')
33 files changed, 1212 insertions, 710 deletions
diff --git a/sonar-application/pom.xml b/sonar-application/pom.xml index 5a9e17e4cd8..66290c24389 100644 --- a/sonar-application/pom.xml +++ b/sonar-application/pom.xml @@ -1,5 +1,6 @@ <?xml version="1.0" encoding="UTF-8"?> -<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> +<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> @@ -16,37 +17,62 @@ <dependencies> <dependency> - <groupId>org.codehaus.sonar</groupId> - <artifactId>sonar-server</artifactId> - <version>${project.version}</version> - <type>war</type> + <groupId>commons-io</groupId> + <artifactId>commons-io</artifactId> </dependency> <dependency> - <groupId>org.eclipse.jetty</groupId> - <artifactId>jetty-server</artifactId> + <groupId>com.google.code.findbugs</groupId> + <artifactId>jsr305</artifactId> + <scope>provided</scope> </dependency> <dependency> - <groupId>org.eclipse.jetty</groupId> - <artifactId>jetty-webapp</artifactId> + <groupId>org.slf4j</groupId> + <artifactId>slf4j-api</artifactId> </dependency> <dependency> - <groupId>commons-io</groupId> - <artifactId>commons-io</artifactId> + <groupId>org.slf4j</groupId> + <artifactId>jul-to-slf4j</artifactId> </dependency> <dependency> - <groupId>commons-lang</groupId> - <artifactId>commons-lang</artifactId> - <scope>test</scope> + <groupId>ch.qos.logback</groupId> + <artifactId>logback-access</artifactId> </dependency> <dependency> - <groupId>org.codehaus.sonar</groupId> - <artifactId>sonar-testing-harness</artifactId> - <scope>test</scope> + <groupId>ch.qos.logback</groupId> + <artifactId>logback-classic</artifactId> </dependency> <dependency> - <groupId>org.mortbay.jetty</groupId> - <artifactId>servlet-api</artifactId> - <scope>test</scope> + <groupId>ch.qos.logback</groupId> + <artifactId>logback-core</artifactId> + </dependency> + <dependency> + <groupId>org.apache.tomcat.embed</groupId> + <artifactId>tomcat-embed-core</artifactId> + </dependency> + <dependency> + <!-- TODO remove this unused dependency, but how to disable Jasper in embedded tomcat ? --> + <groupId>org.apache.tomcat.embed</groupId> + <artifactId>tomcat-embed-jasper</artifactId> + </dependency> + <dependency> + <groupId>org.apache.tomcat.embed</groupId> + <artifactId>tomcat-embed-logging-juli</artifactId> + </dependency> + <!-- + TODO + <dependency> + <groupId>org.apache.tomcat</groupId> + <artifactId>tomcat-dbcp</artifactId> + <version>${tomcat.version}</version> + </dependency> + --> + + + <dependency> + <groupId>org.codehaus.sonar</groupId> + <artifactId>sonar-server</artifactId> + <version>${project.version}</version> + <type>war</type> </dependency> <dependency> <groupId>mysql</groupId> @@ -161,7 +187,6 @@ <type>sonar-plugin</type> <scope>runtime</scope> </dependency> - <dependency> <groupId>org.sonatype.jsw-binaries</groupId> <artifactId>jsw-binaries</artifactId> @@ -170,11 +195,34 @@ <scope>provided</scope> </dependency> <dependency> + <!-- do not upgrade because of licensing change --> <groupId>tanukisoft</groupId> <artifactId>wrapper</artifactId> <version>3.2.3</version> <scope>runtime</scope> </dependency> + + <!-- unit tests --> + <dependency> + <groupId>junit</groupId> + <artifactId>junit</artifactId> + <scope>test</scope> + </dependency> + <dependency> + <groupId>org.easytesting</groupId> + <artifactId>fest-assert</artifactId> + <scope>test</scope> + </dependency> + <dependency> + <groupId>org.mockito</groupId> + <artifactId>mockito-all</artifactId> + <scope>test</scope> + </dependency> + <dependency> + <groupId>com.github.kevinsawicki</groupId> + <artifactId>http-request</artifactId> + <scope>test</scope> + </dependency> </dependencies> <build> @@ -204,8 +252,8 @@ <phase>package</phase> <configuration> <target> - <checksum file="${project.build.directory}/sonar-${project.version}.zip" algorithm="md5" /> - <checksum file="${project.build.directory}/sonar-${project.version}.zip" algorithm="sha" /> + <checksum file="${project.build.directory}/sonar-${project.version}.zip" algorithm="md5"/> + <checksum file="${project.build.directory}/sonar-${project.version}.zip" algorithm="sha"/> </target> </configuration> <goals> diff --git a/sonar-application/src/main/assembly/conf/logback-access.xml b/sonar-application/src/main/assembly/conf/logback-access.xml new file mode 100644 index 00000000000..8e2f63b3cd9 --- /dev/null +++ b/sonar-application/src/main/assembly/conf/logback-access.xml @@ -0,0 +1,25 @@ +<!-- + + See http://logback.qos.ch/access.html#configuration + +--> +<configuration debug="false"> + <contextListener class="ch.qos.logback.classic.jul.LevelChangePropagator"/> + + <appender name="FILE" class="ch.qos.logback.core.rolling.RollingFileAppender"> + <file>${SONAR_HOME}/logs/access.log</file> + <rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy"> + <param name="FileNamePattern" value="${SONAR_HOME}/logs/access.%i.log"/> + <param name="MinIndex" value="1"/> + <param name="MaxIndex" value="3"/> + </rollingPolicy> + <triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy"> + <param name="MaxFileSize" value="5MB"/> + </triggeringPolicy> + <encoder> + <pattern>combined</pattern> + </encoder> + </appender> + + <appender-ref ref="FILE" /> +</configuration> diff --git a/sonar-application/src/main/assembly/conf/logback.xml b/sonar-application/src/main/assembly/conf/logback.xml index 5859163242d..570c7169247 100644 --- a/sonar-application/src/main/assembly/conf/logback.xml +++ b/sonar-application/src/main/assembly/conf/logback.xml @@ -1,6 +1,7 @@ <?xml version="1.0" encoding="UTF-8" ?> <configuration debug="false"> + <contextListener class="ch.qos.logback.classic.jul.LevelChangePropagator"/> <appender name="SONAR_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender"> <File>${SONAR_HOME}/logs/sonar.log</File> @@ -13,24 +14,7 @@ <param name="MaxFileSize" value="5MB"/> </triggeringPolicy> <encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder"> - <pattern> - %d{yyyy.MM.dd HH:mm:ss} %-5level %logger{20} %X %msg%n - </pattern> - </encoder> - </appender> - - <!-- appender used to profile Sonar Server --> - <appender name="PROFILING_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender"> - <File>${SONAR_HOME}/logs/profiling.log</File> - <rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy"> - <param name="FileNamePattern" value="${SONAR_HOME}/logs/profiling.%i.log"/> - <param name="MinIndex" value="1"/> - <param name="MaxIndex" value="3"/> - </rollingPolicy> - <triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy"> - <param name="MaxFileSize" value="5MB"/> - </triggeringPolicy> - <encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder"> + <!-- Use %d{yyyy.MM.dd HH:mm:ss.SSS} to display milliseconds --> <pattern> %d{yyyy.MM.dd HH:mm:ss} %-5level %logger{20} %X %msg%n </pattern> @@ -47,11 +31,10 @@ <logger name="rails" additivity="false"> <level value="DEBUG"/> - <appender-ref ref="PROFILING_FILE"/> </logger> Example of command-line to get the HTTP requests with execution time greater than 10s : - grep 'rails Completed in [0-9]\{5,\}ms' < profiling.log + grep 'rails Completed in [0-9]\{5,\}ms' < sonar.log --> <logger name="org.hibernate.cache.ReadWriteCache"> @@ -96,4 +79,4 @@ <appender-ref ref="SONAR_FILE"/> </root> -</configuration>
\ No newline at end of file +</configuration> diff --git a/sonar-application/src/main/assembly/conf/sonar.properties b/sonar-application/src/main/assembly/conf/sonar.properties index 28ce4528bbb..3b4fc79cd0e 100644 --- a/sonar-application/src/main/assembly/conf/sonar.properties +++ b/sonar-application/src/main/assembly/conf/sonar.properties @@ -1,4 +1,3 @@ -#-------------------------------------------------------- # This file must contain only ISO 8859-1 characters # see http://docs.oracle.com/javase/1.5.0/docs/api/java/util/Properties.html#load(java.io.InputStream) # @@ -8,34 +7,45 @@ # # # See also the file conf/wrapper.conf for JVM advanced settings -#--------------------------------------------------------- -#--------------------------------------------------------- -# WEB SETTINGS - STANDALONE MODE ONLY -# These settings are ignored when the war file is deployed to a JEE server. -#--------------------------------------------------------- -# Listen host/port and context path (for example / or /sonar). Default values are 0.0.0.0:9000/. + +#-------------------------------------------------------------------------------------------------- +# WEB SERVER + +# Binding address #sonar.web.host=0.0.0.0 + +# TCP port for incoming HTTP connections #sonar.web.port=9000 + +# Web context must start with slash (/) #sonar.web.context=/ -# Log HTTP requests. Deactivated by default. -#sonar.web.jettyRequestLogs= ../../logs/jetty-yyyy_mm_dd.request.log -#sonar.web.jetty.threads.min=5 -#sonar.web.jetty.threads.max=50 +# The maximum number of connections that the server will accept and process at any given time. +# When this number has been reached, the server will not accept any more connections until +# the number of connections falls below this value. The operating system may still accept connections +# based on the sonar.web.connections.acceptCount property. The default value is 50. +#sonar.web.connections.maxThreads=50 + +# The minimum number of threads always kept running. If not specified, the default of 5 is used. +#sonar.web.connections.minThreads=5 + +# The maximum queue length for incoming connection requests when all possible request processing +# threads are in use. Any requests received when the queue is full will be refused. +# The default value is 25. +#sonar.web.connections.acceptCount=25 -#----------------------------------------------------------------------- + + +#-------------------------------------------------------------------------------------------------- # DATABASE # -# IMPORTANT: the embedded database H2 is used by default. -# It is recommended for tests only. Please use an external database -# for production environment (MySQL, Oracle, Postgresql, SQLServer) -# -#----------------------------------------------------------------------- +# IMPORTANT: the embedded H2 database is used by default. It is recommended for tests only. +# Please use a production-ready database. Supported databases are MySQL, Oracle, PostgreSQL +# and Microsoft SQLServer. -#----- Credentials -# Permissions to create tables and indexes must be granted to JDBC user. +# Permissions to create tables, indices and triggers must be granted to JDBC user. # The schema must be created first. sonar.jdbc.username=sonar sonar.jdbc.password=sonar @@ -100,12 +110,12 @@ sonar.jdbc.minEvictableIdleTimeMillis=600000 sonar.jdbc.timeBetweenEvictionRunsMillis=30000 -#--------------------------------------------------------- + +#-------------------------------------------------------------------------------------------------- # UPDATE CENTER -#--------------------------------------------------------- # The Update Center requires an internet connection to request http://update.sonarsource.org -# It is activated by default: +# It is enabled by default. #sonar.updatecenter.activate=true # HTTP proxy (default none) @@ -123,9 +133,10 @@ sonar.jdbc.timeBetweenEvictionRunsMillis=30000 #http.proxyUser= #http.proxyPassword= -#--------------------------------------------------------- + + +#-------------------------------------------------------------------------------------------------- # NOTIFICATIONS -#--------------------------------------------------------- # Delay (in seconds) between processing of notification queue sonar.notifications.delay=60 diff --git a/sonar-application/src/main/assembly/conf/wrapper.conf b/sonar-application/src/main/assembly/conf/wrapper.conf index 30f68945ede..b12d3701bc7 100644 --- a/sonar-application/src/main/assembly/conf/wrapper.conf +++ b/sonar-application/src/main/assembly/conf/wrapper.conf @@ -32,7 +32,7 @@ wrapper.java.mainclass=org.tanukisoftware.wrapper.WrapperSimpleApp # Java Classpath (include wrapper.jar) Add class path elements as # needed starting from 1 wrapper.java.classpath.1=../../lib/*.jar -wrapper.java.classpath.2=../../ +wrapper.java.classpath.2=../../conf wrapper.java.classpath.3=../../extensions/jdbc-driver/mysql/*.jar wrapper.java.classpath.4=../../extensions/jdbc-driver/oracle/*.jar wrapper.java.classpath.5=../../extensions/jdbc-driver/postgresql/*.jar @@ -70,7 +70,7 @@ wrapper.console.loglevel=INFO wrapper.logfile=../../logs/sonar.log # Format of output for the log file. (See docs for formats) -wrapper.logfile.format=LPTM +wrapper.logfile.format=TLM # Log Level for log file output. (See docs for log levels) wrapper.logfile.loglevel=INFO diff --git a/sonar-application/src/main/java/org/sonar/application/Connectors.java b/sonar-application/src/main/java/org/sonar/application/Connectors.java new file mode 100644 index 00000000000..f6c8965d166 --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/Connectors.java @@ -0,0 +1,69 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.catalina.connector.Connector; +import org.apache.catalina.startup.Tomcat; + +class Connectors { + + static final String PROPERTY_SHUTDOWN_TOKEN = "sonar.web.shutdown.token"; + static final String PROPERTY_SHUTDOWN_PORT = "sonar.web.shutdown.port"; + static final String PROPERTY_MIN_THREADS = "sonar.web.connections.minThreads"; + static final String PROPERTY_MAX_THREADS = "sonar.web.connections.maxThreads"; + static final String PROPERTY_ACCEPT_COUNT = "sonar.web.connections.acceptCount"; + + static void configure(Tomcat tomcat, Props props) { + tomcat.getServer().setAddress(props.of("sonar.web.host", "0.0.0.0")); + configureShutdown(tomcat, props); + + Connector connector = new Connector("HTTP/1.1"); + + // TODO manage redirects from other ports ? + + connector.setPort(props.intOf("sonar.web.port", 9000)); + configurePool(props, connector); + configureCompression(connector); + tomcat.setConnector(connector); + tomcat.getService().addConnector(connector); + } + + private static void configureShutdown(Tomcat tomcat, Props props) { + String shutdownToken = props.of(PROPERTY_SHUTDOWN_TOKEN); + Integer shutdownPort = props.intOf(PROPERTY_SHUTDOWN_PORT); + if (shutdownToken != null && !"".equals(shutdownToken) && shutdownPort != null) { + tomcat.getServer().setPort(shutdownPort); + tomcat.getServer().setShutdown(shutdownToken); + } + } + + private static void configurePool(Props props, Connector connector) { + connector.setProperty("acceptorThreadCount", String.valueOf(2)); + connector.setProperty("minSpareThreads", String.valueOf(props.intOf(PROPERTY_MIN_THREADS, 5))); + connector.setProperty("maxThreads", String.valueOf(props.intOf(PROPERTY_MAX_THREADS, 50))); + connector.setProperty("acceptCount", String.valueOf(props.intOf(PROPERTY_ACCEPT_COUNT, 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"); + } +} diff --git a/sonar-application/src/main/java/org/sonar/application/EmbeddedTomcat.java b/sonar-application/src/main/java/org/sonar/application/EmbeddedTomcat.java new file mode 100644 index 00000000000..79d9cef2a56 --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/EmbeddedTomcat.java @@ -0,0 +1,112 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.catalina.LifecycleException; +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; + + EmbeddedTomcat(Env env) { + this.env = env; + } + + void start() throws Exception { + if (tomcat != null || hook != null) { + throw new IllegalStateException("Tomcat is already started"); + } + + 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); + + Props props = Props.create(env); + Logging.configure(tomcat, env); + Connectors.configure(tomcat, props); + Webapp.configure(tomcat, env, props); + + tomcat.start(); + addShutdownHook(); + tomcat.getServer().await(); + + // 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; + 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; + } + } + + int port() { + return tomcat.getService().findConnectors()[0].getLocalPort(); + } +} diff --git a/sonar-application/src/main/java/org/sonar/application/Env.java b/sonar-application/src/main/java/org/sonar/application/Env.java new file mode 100644 index 00000000000..53f25a9b8d7 --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/Env.java @@ -0,0 +1,69 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.commons.io.FileUtils; + +import java.io.File; +import java.io.IOException; +import java.net.URISyntaxException; + +class Env { + + private final File confFile; + + Env(File confFile) { + this.confFile = confFile; + } + + Env() throws URISyntaxException { + this(new File(Env.class.getResource("/sonar.properties").toURI())); + } + + File rootDir() { + return confFile.getParentFile().getParentFile(); + } + + 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/sonar-application/src/main/java/org/sonar/application/JettyEmbedder.java b/sonar-application/src/main/java/org/sonar/application/JettyEmbedder.java deleted file mode 100644 index 975db243f04..00000000000 --- a/sonar-application/src/main/java/org/sonar/application/JettyEmbedder.java +++ /dev/null @@ -1,196 +0,0 @@ -/* - * SonarQube, open source software quality management tool. - * Copyright (C) 2008-2013 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.application; - -import org.apache.commons.io.FileUtils; -import org.eclipse.jetty.server.Handler; -import org.eclipse.jetty.server.NCSARequestLog; -import org.eclipse.jetty.server.Server; -import org.eclipse.jetty.server.handler.HandlerCollection; -import org.eclipse.jetty.server.handler.RequestLogHandler; -import org.eclipse.jetty.server.handler.ShutdownHandler; -import org.eclipse.jetty.server.nio.SelectChannelConnector; -import org.eclipse.jetty.util.thread.QueuedThreadPool; -import org.eclipse.jetty.webapp.WebAppContext; - -import java.io.File; -import java.io.IOException; -import java.net.URISyntaxException; -import java.net.URL; -import java.util.ArrayList; -import java.util.Collection; -import java.util.List; -import java.util.Properties; - -public class JettyEmbedder { - - private final Server server; - private final String host; - private final int port; - private final String contextPath; - private final Properties configuration; - - public JettyEmbedder(String host, int port, String contextPath, Properties configuration) throws Exception { - this.host = host.trim(); - this.port = port; - this.contextPath = contextPath; - this.configuration = configuration; - server = new Server(); - configureProgrammatically(); - } - - /** - * for tests - */ - JettyEmbedder(String host, int port) throws Exception { - this(host, port, null, new Properties()); - } - - public void start() throws Exception { - server.start(); - - Runtime.getRuntime().addShutdownHook(new Thread() { - @Override - public void run() { - try { - server.stop(); - } catch (Exception e) { - System.err.println("Can not stop the Jetty server"); - e.printStackTrace(); - } - } - }); - } - - private Server configureProgrammatically() throws URISyntaxException { - configureServer(); - - List<Handler> handlers = new ArrayList<Handler>(); - - String shutdownCookie = System.getProperty("sonar.shutdownToken"); - if (shutdownCookie != null && !"".equals(shutdownCookie)) { - System.out.println("Registering shutdown handler"); - ShutdownHandler shutdownHandler = new ShutdownHandler(server, shutdownCookie); - shutdownHandler.setExitJvm(true); - handlers.add(shutdownHandler); - } - - WebAppContext context = new WebAppContext(getPath("/web"), contextPath); - // Set up the path to the custom webdefault.xml (SONAR-3962). - context.setDefaultsDescriptor("/org/sonar/application/webdefault.xml"); - handlers.add(context); - - String filenamePattern = configuration.getProperty("sonar.web.jettyRequestLogs"); - if (filenamePattern != null) { - handlers.add(configureRequestLogHandler(filenamePattern)); - } - - HandlerCollection handler = new HandlerCollection(); - handler.setHandlers(handlers.toArray(new Handler[handlers.size()])); - server.setHandler(handler); - - return server; - } - - private RequestLogHandler configureRequestLogHandler(String filenamePattern) { - RequestLogHandler requestLogHandler = new RequestLogHandler(); - NCSARequestLog requestLog = new NCSARequestLog(filenamePattern); - requestLog.setRetainDays(7); - requestLog.setAppend(true); - requestLog.setExtended(true); - requestLog.setLogTimeZone("GMT"); - requestLogHandler.setRequestLog(requestLog); - return requestLogHandler; - } - - private void configureServer() { - QueuedThreadPool threadPool = new QueuedThreadPool(); - threadPool.setMinThreads(getIntProperty("sonar.web.jetty.threads.min", 5)); - threadPool.setMaxThreads(getIntProperty("sonar.web.jetty.threads.max", 50)); - server.setThreadPool(threadPool); - SelectChannelConnector connector = new SelectChannelConnector(); - connector.setHost(host); - connector.setPort(port); - connector.setStatsOn(false); - connector.setAcceptors(2); - connector.setConfidentialPort(8443); - server.addConnector(connector); - server.setStopAtShutdown(true); - server.setSendServerVersion(false); - server.setSendDateHeader(true); - server.setGracefulShutdown(1000); - } - - private int getIntProperty(String name, int defaultValue) { - String value = configuration.getProperty(name); - if (null == value) { - return defaultValue; - } - - return Integer.parseInt(value); - } - - final String getPluginsClasspath(String pluginsPathFromClassloader) throws URISyntaxException, IOException { - URL resource = getClass().getResource(pluginsPathFromClassloader); - if (resource == null) { - return null; - } - - List<String> paths = new ArrayList<String>(); - - File pluginsDir = new File(resource.toURI()); - paths.add(pluginsDir.getCanonicalPath() + System.getProperty("file.separator")); - - Collection<File> files = FileUtils.listFiles(pluginsDir, new String[] {"jar"}, false); - for (File file : files) { - paths.add(file.getCanonicalPath()); - } - - return join(paths, ","); - } - - private String join(List<String> paths, String separator) { - StringBuilder sb = new StringBuilder(); - for (String path : paths) { - if (sb.length() > 0) { - sb.append(separator); - } - sb.append(path); - } - return sb.toString(); - } - - private String getPath(String resourcePath) throws URISyntaxException { - URL resource = getClass().getResource(resourcePath); - if (resource != null) { - return resource.toURI().toString(); - } - return null; - } - - Server getServer() { - return server; - } - - @Override - public String toString() { - return new StringBuilder().append("http://").append(host).append(":").append(port).append(contextPath).toString(); - } -} diff --git a/sonar-application/src/main/java/org/sonar/application/Logging.java b/sonar-application/src/main/java/org/sonar/application/Logging.java new file mode 100644 index 00000000000..29d8d097698 --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/Logging.java @@ -0,0 +1,50 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import ch.qos.logback.access.tomcat.LogbackValve; +import org.apache.catalina.startup.Tomcat; +import org.slf4j.bridge.SLF4JBridgeHandler; + +import java.io.File; +import java.util.logging.LogManager; + +class Logging { + + static final String CONF_PATH = "conf/logback-access.xml"; + + 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) { + tomcat.setSilent(false); + LogbackValve valve = new LogbackValve(); + valve.setQuiet(true); + File confFile = env.file(CONF_PATH); + if (!confFile.exists()) { + throw new IllegalStateException("File is missing: " + confFile.getAbsolutePath()); + } + valve.setFilename(confFile.getAbsolutePath()); + tomcat.getHost().getPipeline().addValve(valve); + } +} diff --git a/sonar-application/src/main/java/org/sonar/application/NullJarScanner.java b/sonar-application/src/main/java/org/sonar/application/NullJarScanner.java new file mode 100644 index 00000000000..c6c82e2104d --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/NullJarScanner.java @@ -0,0 +1,35 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +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<String> jarsToSkip) { + } +} diff --git a/sonar-application/src/main/java/org/sonar/application/Props.java b/sonar-application/src/main/java/org/sonar/application/Props.java new file mode 100644 index 00000000000..a3d1a8b27e2 --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/Props.java @@ -0,0 +1,84 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.commons.io.IOUtils; + +import javax.annotation.Nullable; +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileReader; +import java.io.IOException; +import java.util.Properties; + +/** + * TODO support env substitution + */ +class Props { + private final Properties props; + + Props(Properties props) { + this.props = props; + } + + String of(String key) { + return props.getProperty(key); + } + + String of(String key, @Nullable String defaultValue) { + String s = of(key); + return s == null ? defaultValue : s; + } + + Integer intOf(String key) { + String s = of(key); + if (s != null && !"".equals(s)) { + try { + return Integer.parseInt(s); + } catch (NumberFormatException e) { + throw new IllegalStateException("Value of property " + key + " is not an integer: " + s, e); + } + } + return null; + } + + int intOf(String key, int defaultValue) { + Integer i = intOf(key); + return i == null ? defaultValue : i; + } + + static Props create(Env env) { + File propsFile = env.file("conf/sonar.properties"); + Properties p = new Properties(); + FileReader reader = null; + try { + reader = new FileReader(propsFile); + p.load(reader); + p.putAll(System.getProperties()); + return new Props(p); + + } catch (Exception e) { + throw new IllegalStateException("File does not exist or can't be open: " + propsFile, e); + + } finally { + IOUtils.closeQuietly(reader); + } + } +} diff --git a/sonar-application/src/main/java/org/sonar/application/StartServer.java b/sonar-application/src/main/java/org/sonar/application/StartServer.java index 923f6ddd153..3704265007c 100644 --- a/sonar-application/src/main/java/org/sonar/application/StartServer.java +++ b/sonar-application/src/main/java/org/sonar/application/StartServer.java @@ -19,61 +19,33 @@ */ package org.sonar.application; -import org.apache.commons.io.FileUtils; - -import java.io.File; -import java.io.IOException; -import java.net.URISyntaxException; -import java.util.Properties; +// TODO dev mode +// TODO sanitize jetty dependencies +// TODO remove logback/slf4j from sonar-server public final class StartServer { - private static final String DEFAULT_WEB_HOST = "0.0.0.0"; - private static final int DEFAULT_WEB_PORT = 9000; - private static final String DEFAULT_WEB_CONTEXT = "/"; - private static final String PROPERTIES_FILE_PATH = "/conf/sonar.properties"; - - private StartServer() { - } - public static void main(String[] args) throws Exception { - canCreateTemporaryFiles(); - configureHome(); + private final EmbeddedTomcat tomcat; - Properties configuration = getConfiguration(); - String host = configuration.getProperty("sonar.web.host", DEFAULT_WEB_HOST); - int port = Integer.parseInt(configuration.getProperty("sonar.web.port", "" + DEFAULT_WEB_PORT)); - String context = configuration.getProperty("sonar.web.context", DEFAULT_WEB_CONTEXT); - JettyEmbedder jetty = new JettyEmbedder(host, port, context, configuration); + public StartServer(Env env) { + Logging.init(); + env.verifyWritableTempDir(); + this.tomcat = new EmbeddedTomcat(env); + } - jetty.start(); - Thread.currentThread().join(); + void start() throws Exception { + tomcat.start(); } - /** - * This check is required in order to provide more meaningful message than JRuby - see SONAR-2715 - */ - private static void canCreateTemporaryFiles() { - File file = null; - try { - file = File.createTempFile("sonar-check", "tmp"); - } catch (IOException e) { - throw new IllegalStateException("Unable to create file in temporary directory, please check existence of it and permissions: " + FileUtils.getTempDirectoryPath(), e); - } finally { - FileUtils.deleteQuietly(file); - } + int port() { + return tomcat.port(); } - private static Properties getConfiguration() throws IOException { - Properties properties = new Properties(); - properties.load(StartServer.class.getResourceAsStream(PROPERTIES_FILE_PATH)); - return properties; + void stop() throws Exception { + tomcat.stop(); } - /** - * @see org.sonar.server.platform.SonarHome#SONAR_HOME - */ - private static void configureHome() throws URISyntaxException { - File confFile = new File(StartServer.class.getResource(PROPERTIES_FILE_PATH).toURI()); - System.setProperty("SONAR_HOME", confFile.getParentFile().getParentFile().getAbsolutePath()); + public static void main(String[] args) throws Exception { + new StartServer(new Env()).start(); } } diff --git a/sonar-application/src/main/java/org/sonar/application/Webapp.java b/sonar-application/src/main/java/org/sonar/application/Webapp.java new file mode 100644 index 00000000000..70d33d829b5 --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/Webapp.java @@ -0,0 +1,39 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.catalina.Context; +import org.apache.catalina.startup.Tomcat; + +class Webapp { + + static void configure(Tomcat tomcat, Env env, Props props) { + String ctx = props.of("sonar.web.context", "/"); + try { + System.setProperty("SONAR_HOME", env.rootDir().getAbsolutePath()); + Context context = tomcat.addWebapp(ctx, env.file("web").getAbsolutePath()); + context.setConfigFile(env.file("web/META-INF/context.xml").toURL()); + context.setJarScanner(new NullJarScanner()); + + } catch (Exception e) { + throw new IllegalStateException("Fail to configure webapp", e); + } + } +} diff --git a/sonar-application/src/main/java/org/sonar/application/package-info.java b/sonar-application/src/main/java/org/sonar/application/package-info.java new file mode 100644 index 00000000000..009b187eda5 --- /dev/null +++ b/sonar-application/src/main/java/org/sonar/application/package-info.java @@ -0,0 +1,23 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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. + */ +@ParametersAreNonnullByDefault +package org.sonar.application; + +import javax.annotation.ParametersAreNonnullByDefault; diff --git a/sonar-application/src/main/resources/org/sonar/application/webdefault.xml b/sonar-application/src/main/resources/org/sonar/application/webdefault.xml deleted file mode 100644 index 75599d24cb4..00000000000 --- a/sonar-application/src/main/resources/org/sonar/application/webdefault.xml +++ /dev/null @@ -1,347 +0,0 @@ -<?xml version="1.0" encoding="ISO-8859-1"?> - - <!-- Modified version of webdefault.xml shipped inside jetty-webapp in order to remove JSP servlet --> - - <!-- ===================================================================== --> - <!-- This file contains the default descriptor for web applications. --> - <!-- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - --> - <!-- The intent of this descriptor is to include jetty specific or common --> - <!-- configuration for all webapps. If a context has a webdefault.xml --> - <!-- descriptor, it is applied before the contexts own web.xml file --> - <!-- --> - <!-- A context may be assigned a default descriptor by: --> - <!-- + Calling WebApplicationContext.setDefaultsDescriptor --> - <!-- + Passed an arg to addWebApplications --> - <!-- --> - <!-- This file is used both as the resource within the jetty.jar (which is --> - <!-- used as the default if no explicit defaults descriptor is set) and it --> - <!-- is copied to the etc directory of the Jetty distro and explicitly --> - <!-- by the jetty.xml file. --> - <!-- --> - <!-- ===================================================================== --> -<web-app - xmlns="http://java.sun.com/xml/ns/javaee" - xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" - xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" - metadata-complete="true" - version="2.5" -> - - <description> - Default web.xml file. - This file is applied to a Web application before it's own WEB_INF/web.xml file - </description> - - <!-- ==================================================================== --> - <!-- Removes static references to beans from javax.el.BeanELResolver to --> - <!-- ensure webapp classloader can be released on undeploy --> - <!-- ==================================================================== --> - <listener> - <listener-class>org.eclipse.jetty.servlet.listener.ELContextCleaner</listener-class> - </listener> - - <!-- ==================================================================== --> - <!-- Removes static cache of Methods from java.beans.Introspector to --> - <!-- ensure webapp classloader can be released on undeploy --> - <!-- ==================================================================== --> - <listener> - <listener-class>org.eclipse.jetty.servlet.listener.IntrospectorCleaner</listener-class> - </listener> - - <!-- ==================================================================== --> - <!-- The default servlet. --> - <!-- This servlet, normally mapped to /, provides the handling for static --> - <!-- content, OPTIONS and TRACE methods for the context. --> - <!-- The following initParameters are supported: --> - <!-- - * acceptRanges If true, range requests and responses are - * supported - * - * dirAllowed If true, directory listings are returned if no - * welcome file is found. Else 403 Forbidden. - * - * welcomeServlets If true, attempt to dispatch to welcome files - * that are servlets, but only after no matching static - * resources could be found. If false, then a welcome - * file must exist on disk. If "exact", then exact - * servlet matches are supported without an existing file. - * Default is true. - * - * This must be false if you want directory listings, - * but have index.jsp in your welcome file list. - * - * redirectWelcome If true, welcome files are redirected rather than - * forwarded to. - * - * gzip If set to true, then static content will be served as - * gzip content encoded if a matching resource is - * found ending with ".gz" - * - * resourceBase Set to replace the context resource base - * - * resourceCache If set, this is a context attribute name, which the servlet - * will use to look for a shared ResourceCache instance. - * - * relativeResourceBase - * Set with a pathname relative to the base of the - * servlet context root. Useful for only serving static content out - * of only specific subdirectories. - * - * aliases If True, aliases of resources are allowed (eg. symbolic - * links and caps variations). May bypass security constraints. - * - * maxCacheSize The maximum total size of the cache or 0 for no cache. - * maxCachedFileSize The maximum size of a file to cache - * maxCachedFiles The maximum number of files to cache - * - * useFileMappedBuffer - * If set to true, it will use mapped file buffer to serve static content - * when using NIO connector. Setting this value to false means that - * a direct buffer will be used instead of a mapped file buffer. - * By default, this is set to true. - * - * cacheControl If set, all static content will have this value set as the cache-control - * header. - --> - - - <!-- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - --> - <servlet> - <servlet-name>default</servlet-name> - <servlet-class>org.eclipse.jetty.servlet.DefaultServlet</servlet-class> - <init-param> - <param-name>aliases</param-name> - <param-value>false</param-value> - </init-param> - <init-param> - <param-name>acceptRanges</param-name> - <param-value>true</param-value> - </init-param> - <init-param> - <param-name>dirAllowed</param-name> - <param-value>true</param-value> - </init-param> - <init-param> - <param-name>welcomeServlets</param-name> - <param-value>false</param-value> - </init-param> - <init-param> - <param-name>redirectWelcome</param-name> - <param-value>false</param-value> - </init-param> - <init-param> - <param-name>maxCacheSize</param-name> - <param-value>256000000</param-value> - </init-param> - <init-param> - <param-name>maxCachedFileSize</param-name> - <param-value>200000000</param-value> - </init-param> - <init-param> - <param-name>maxCachedFiles</param-name> - <param-value>2048</param-value> - </init-param> - <init-param> - <param-name>gzip</param-name> - <param-value>true</param-value> - </init-param> - <init-param> - <param-name>useFileMappedBuffer</param-name> - <param-value>true</param-value> - </init-param> - <!-- - <init-param> - <param-name>resourceCache</param-name> - <param-value>resourceCache</param-value> - </init-param> - --> - <!-- - <init-param> - <param-name>cacheControl</param-name> - <param-value>max-age=3600,public</param-value> - </init-param> - --> - <load-on-startup>0</load-on-startup> - </servlet> - - <servlet-mapping> - <servlet-name>default</servlet-name> - <url-pattern>/</url-pattern> - </servlet-mapping> - - - <!-- ==================================================================== --> - <session-config> - <session-timeout>30</session-timeout> - </session-config> - - <!-- ==================================================================== --> - <locale-encoding-mapping-list> - <locale-encoding-mapping> - <locale>ar</locale> - <encoding>ISO-8859-6</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>be</locale> - <encoding>ISO-8859-5</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>bg</locale> - <encoding>ISO-8859-5</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>ca</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>cs</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>da</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>de</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>el</locale> - <encoding>ISO-8859-7</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>en</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>es</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>et</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>fi</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>fr</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>hr</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>hu</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>is</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>it</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>iw</locale> - <encoding>ISO-8859-8</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>ja</locale> - <encoding>Shift_JIS</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>ko</locale> - <encoding>EUC-KR</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>lt</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>lv</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>mk</locale> - <encoding>ISO-8859-5</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>nl</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>no</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>pl</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>pt</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>ro</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>ru</locale> - <encoding>ISO-8859-5</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>sh</locale> - <encoding>ISO-8859-5</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>sk</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>sl</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>sq</locale> - <encoding>ISO-8859-2</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>sr</locale> - <encoding>ISO-8859-5</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>sv</locale> - <encoding>ISO-8859-1</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>tr</locale> - <encoding>ISO-8859-9</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>uk</locale> - <encoding>ISO-8859-5</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>zh</locale> - <encoding>GB2312</encoding> - </locale-encoding-mapping> - <locale-encoding-mapping> - <locale>zh_TW</locale> - <encoding>Big5</encoding> - </locale-encoding-mapping> - </locale-encoding-mapping-list> - - <security-constraint> - <web-resource-collection> - <web-resource-name>Disable TRACE</web-resource-name> - <url-pattern>/</url-pattern> - <http-method>TRACE</http-method> - </web-resource-collection> - <auth-constraint/> - </security-constraint> - -</web-app> diff --git a/sonar-application/src/test/fake-app/conf/logback-access.xml b/sonar-application/src/test/fake-app/conf/logback-access.xml new file mode 100644 index 00000000000..77cfc885d09 --- /dev/null +++ b/sonar-application/src/test/fake-app/conf/logback-access.xml @@ -0,0 +1,17 @@ +<!-- + + See http://logback.qos.ch/access.html#configuration + +--> +<configuration debug="false"> + <contextListener class="ch.qos.logback.classic.jul.LevelChangePropagator"/> + + <appender name="FILE" class="ch.qos.logback.core.FileAppender"> + <file>${SONAR_HOME}/logs/access.log</file> + <encoder> + <pattern>combined</pattern> + </encoder> + </appender> + + <appender-ref ref="FILE" /> +</configuration> diff --git a/sonar-application/src/test/fake-app/conf/sonar.properties b/sonar-application/src/test/fake-app/conf/sonar.properties new file mode 100644 index 00000000000..e8e111dcab8 --- /dev/null +++ b/sonar-application/src/test/fake-app/conf/sonar.properties @@ -0,0 +1,2 @@ +# random open port +sonar.web.port=0
\ No newline at end of file diff --git a/sonar-application/src/test/fake-app/web/META-INF/context.xml b/sonar-application/src/test/fake-app/web/META-INF/context.xml new file mode 100644 index 00000000000..5e31888a15e --- /dev/null +++ b/sonar-application/src/test/fake-app/web/META-INF/context.xml @@ -0,0 +1,2 @@ +<Context antiJARLocking="true" antiResourceLocking="true"> +</Context> diff --git a/sonar-application/src/test/fake-app/web/WEB-INF/web.xml b/sonar-application/src/test/fake-app/web/WEB-INF/web.xml new file mode 100644 index 00000000000..9f13dabf72f --- /dev/null +++ b/sonar-application/src/test/fake-app/web/WEB-INF/web.xml @@ -0,0 +1,11 @@ +<?xml version="1.0" encoding="UTF-8"?> +<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xmlns="http://java.sun.com/xml/ns/javaee" + xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" + id="Fake" + version="3.0" + metadata-complete="true"> + + <display-name>Fake</display-name> + +</web-app> diff --git a/sonar-application/src/test/fake-app/web/index.html b/sonar-application/src/test/fake-app/web/index.html new file mode 100644 index 00000000000..5e1c309dae7 --- /dev/null +++ b/sonar-application/src/test/fake-app/web/index.html @@ -0,0 +1 @@ +Hello World
\ No newline at end of file diff --git a/sonar-application/src/test/java/org/sonar/application/ConnectorsTest.java b/sonar-application/src/test/java/org/sonar/application/ConnectorsTest.java new file mode 100644 index 00000000000..b437227211f --- /dev/null +++ b/sonar-application/src/test/java/org/sonar/application/ConnectorsTest.java @@ -0,0 +1,102 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.catalina.connector.Connector; +import org.apache.catalina.startup.Tomcat; +import org.junit.Test; +import org.mockito.ArgumentMatcher; +import org.mockito.Mockito; + +import java.util.Properties; + +import static org.mockito.Matchers.anyInt; +import static org.mockito.Matchers.anyString; +import static org.mockito.Matchers.argThat; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.never; +import static org.mockito.Mockito.verify; + +public class ConnectorsTest { + @Test + public void enable_shutdown_port() throws Exception { + Properties p = new Properties(); + p.setProperty(Connectors.PROPERTY_SHUTDOWN_PORT, "9010"); + p.setProperty(Connectors.PROPERTY_SHUTDOWN_TOKEN, "SHUTDOWN"); + Props props = new Props(p); + + Tomcat tomcat = mock(Tomcat.class, Mockito.RETURNS_DEEP_STUBS); + Connectors.configure(tomcat, props); + + verify(tomcat.getServer()).setPort(9010); + verify(tomcat.getServer()).setShutdown("SHUTDOWN"); + } + + @Test + public void disable_shutdown_port_by_default() throws Exception { + Props props = new Props(new Properties()); + + Tomcat tomcat = mock(Tomcat.class, Mockito.RETURNS_DEEP_STUBS); + Connectors.configure(tomcat, props); + + verify(tomcat.getServer(), never()).setPort(anyInt()); + verify(tomcat.getServer(), never()).setShutdown(anyString()); + } + + @Test + public void configure_thread_pool() throws Exception { + Properties p = new Properties(); + p.setProperty(Connectors.PROPERTY_MIN_THREADS, "2"); + p.setProperty(Connectors.PROPERTY_MAX_THREADS, "30"); + p.setProperty(Connectors.PROPERTY_ACCEPT_COUNT, "20"); + Props props = new Props(p); + + Tomcat tomcat = mock(Tomcat.class, Mockito.RETURNS_DEEP_STUBS); + Connectors.configure(tomcat, props); + + verify(tomcat).setConnector(argThat(new ArgumentMatcher<Connector>() { + @Override + public boolean matches(Object o) { + Connector c = (Connector)o; + return (Integer)c.getProperty("minSpareThreads") == 2 && + (Integer) c.getProperty("maxThreads") == 30 && + (Integer) c.getProperty("acceptCount") == 20; + } + })); + } + + @Test + public void configure_default_thread_pool() throws Exception { + Props props = new Props(new Properties()); + + Tomcat tomcat = mock(Tomcat.class, Mockito.RETURNS_DEEP_STUBS); + Connectors.configure(tomcat, props); + + verify(tomcat).setConnector(argThat(new ArgumentMatcher<Connector>() { + @Override + public boolean matches(Object o) { + Connector c = (Connector)o; + return (Integer)c.getProperty("minSpareThreads") == 5 && + (Integer) c.getProperty("maxThreads") == 50 && + (Integer) c.getProperty("acceptCount") == 25; + } + })); + } +} diff --git a/sonar-application/src/test/java/org/sonar/application/EnvTest.java b/sonar-application/src/test/java/org/sonar/application/EnvTest.java new file mode 100644 index 00000000000..0aab59fa26e --- /dev/null +++ b/sonar-application/src/test/java/org/sonar/application/EnvTest.java @@ -0,0 +1,80 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.commons.io.FileUtils; +import org.junit.Rule; +import org.junit.Test; +import org.junit.rules.TemporaryFolder; + +import java.io.File; + +import static org.fest.assertions.Assertions.assertThat; + +public class EnvTest { + + @Rule + public TemporaryFolder temp = new TemporaryFolder(); + + @Test + public void files() throws Exception { + File home = temp.newFolder(); + File confFile = new File(home, "conf/sonar.properties"); + File logFile = new File(home, "logs/sonar.log"); + + FileUtils.touch(confFile); + FileUtils.touch(logFile); + + Env env = new Env(confFile); + + assertThat(env.rootDir()).isDirectory().exists().isEqualTo(home); + assertThat(env.file("conf/sonar.properties")).isFile().exists().isEqualTo(confFile); + assertThat(env.file("logs/sonar.log")).isFile().exists().isEqualTo(logFile); + assertThat(env.file("xxx/unknown.log")).doesNotExist(); + } + + @Test + public void fresh_dir() throws Exception { + File home = temp.newFolder(); + File confFile = new File(home, "conf/sonar.properties"); + File logFile = new File(home, "logs/sonar.log"); + + FileUtils.touch(confFile); + FileUtils.touch(logFile); + + Env env = new Env(confFile); + + File data = env.freshDir("data/h2"); + assertThat(data).isDirectory().exists(); + assertThat(data.getParentFile().getName()).isEqualTo("data"); + assertThat(data.getParentFile().getParentFile()).isEqualTo(home); + + // clean directory + File logs = env.freshDir("logs"); + assertThat(logs).isDirectory().exists(); + assertThat(logs.listFiles()).isEmpty(); + } + + @Test + public void temp_dir_should_be_writable() throws Exception { + new Env(temp.newFile()).verifyWritableTempDir(); + // do not fail + } +} diff --git a/sonar-application/src/test/java/org/sonar/application/JettyEmbedderTest.java b/sonar-application/src/test/java/org/sonar/application/JettyEmbedderTest.java deleted file mode 100644 index 5c7a35e163c..00000000000 --- a/sonar-application/src/test/java/org/sonar/application/JettyEmbedderTest.java +++ /dev/null @@ -1,52 +0,0 @@ -/* - * SonarQube, open source software quality management tool. - * Copyright (C) 2008-2013 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.application; - -import org.apache.commons.lang.StringUtils; -import org.junit.Test; - -import static org.fest.assertions.Assertions.assertThat; - -public class JettyEmbedderTest { - - @Test - public void shouldConfigureProgrammatically() throws Exception { - JettyEmbedder jetty = new JettyEmbedder("1.2.3.4", 9999); - - assertThat(jetty.getServer().getConnectors()).hasSize(1); - assertThat(jetty.getServer().getConnectors()[0].getPort()).isEqualTo(9999); - assertThat(jetty.getServer().getConnectors()[0].getHost()).isEqualTo("1.2.3.4"); - } - - @Test - public void shouldLoadPluginsClasspath() throws Exception { - JettyEmbedder jetty = new JettyEmbedder("127.0.0.1", 9999); - - String classpath = jetty.getPluginsClasspath("/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath"); - classpath = StringUtils.replaceChars(classpath, "\\", "/"); - - assertThat(classpath).contains("org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin1.jar"); - assertThat(classpath).contains("org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin1.jar"); - assertThat(classpath).contains("org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin2.jar"); - - // important : directories end with / - assertThat(classpath).contains("org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/,"); - } -} diff --git a/sonar-application/src/test/java/org/sonar/application/LoggingTest.java b/sonar-application/src/test/java/org/sonar/application/LoggingTest.java new file mode 100644 index 00000000000..ca5f6ef3c51 --- /dev/null +++ b/sonar-application/src/test/java/org/sonar/application/LoggingTest.java @@ -0,0 +1,68 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import ch.qos.logback.access.tomcat.LogbackValve; +import org.apache.catalina.Valve; +import org.apache.catalina.startup.Tomcat; +import org.junit.Test; +import org.mockito.ArgumentMatcher; +import org.mockito.Mockito; + +import java.io.File; + +import static org.fest.assertions.Assertions.assertThat; +import static org.fest.assertions.Fail.fail; +import static org.mockito.Matchers.argThat; +import static org.mockito.Mockito.*; + +public class LoggingTest { + @Test + public void configure_access_logs() throws Exception { + Tomcat tomcat = mock(Tomcat.class, Mockito.RETURNS_DEEP_STUBS); + Env env = mock(Env.class); + final File propsFile = new File(getClass().getResource("/org/sonar/application/LoggingTest/logback-access.xml").toURI()); + when(env.file(Logging.CONF_PATH)).thenReturn(propsFile); + Logging.configure(tomcat, env); + + verify(tomcat.getHost().getPipeline()).addValve(argThat(new ArgumentMatcher<Valve>() { + @Override + public boolean matches(Object o) { + LogbackValve v = (LogbackValve) o; + return v.getFilename().equals(propsFile.getAbsolutePath()); + } + })); + } + + @Test + public void fail_if_missing_conf_file() throws Exception { + Tomcat tomcat = mock(Tomcat.class, Mockito.RETURNS_DEEP_STUBS); + Env env = mock(Env.class); + final File confFile = new File("target/does_not_exist/logback-access.xml"); + when(env.file(Logging.CONF_PATH)).thenReturn(confFile); + + try { + Logging.configure(tomcat, env); + fail(); + } catch (IllegalStateException e) { + assertThat(e).hasMessage("File is missing: " + confFile.getAbsolutePath()); + } + } +} diff --git a/sonar-application/src/test/java/org/sonar/application/NullJarScannerTest.java b/sonar-application/src/test/java/org/sonar/application/NullJarScannerTest.java new file mode 100644 index 00000000000..50e3a9a60d0 --- /dev/null +++ b/sonar-application/src/test/java/org/sonar/application/NullJarScannerTest.java @@ -0,0 +1,43 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.apache.tomcat.JarScannerCallback; +import org.junit.Test; + +import javax.servlet.ServletContext; +import java.util.HashSet; + +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.verifyZeroInteractions; + +public class NullJarScannerTest { + + @Test + public void does_nothing() { + ServletContext context = mock(ServletContext.class); + ClassLoader classloader = mock(ClassLoader.class); + JarScannerCallback callback = mock(JarScannerCallback.class); + + new NullJarScanner().scan(context, classloader, callback, new HashSet<String>()); + + verifyZeroInteractions(context, classloader, callback); + } +} diff --git a/sonar-application/src/test/java/org/sonar/application/PropsTest.java b/sonar-application/src/test/java/org/sonar/application/PropsTest.java new file mode 100644 index 00000000000..b2bd5d0fd69 --- /dev/null +++ b/sonar-application/src/test/java/org/sonar/application/PropsTest.java @@ -0,0 +1,98 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import org.junit.Test; + +import java.io.File; +import java.util.Properties; + +import static org.fest.assertions.Assertions.assertThat; +import static org.fest.assertions.Fail.fail; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; + +public class PropsTest { + @Test + public void of() throws Exception { + Properties p = new Properties(); + p.setProperty("foo", "bar"); + Props props = new Props(p); + + assertThat(props.of("foo")).isEqualTo("bar"); + assertThat(props.of("foo", "default value")).isEqualTo("bar"); + assertThat(props.of("unknown")).isNull(); + assertThat(props.of("unknown", "default value")).isEqualTo("default value"); + } + + @Test + public void intOf() throws Exception { + Properties p = new Properties(); + p.setProperty("foo", "33"); + Props props = new Props(p); + + assertThat(props.intOf("foo")).isEqualTo(33); + assertThat(props.intOf("foo", 44)).isEqualTo(33); + assertThat(props.intOf("unknown")).isNull(); + assertThat(props.intOf("unknown", 44)).isEqualTo(44); + } + + @Test + public void intOf_not_integer() throws Exception { + Properties p = new Properties(); + p.setProperty("foo", "bar"); + Props props = new Props(p); + + try { + props.intOf("foo"); + fail(); + } catch (IllegalStateException e) { + assertThat(e).hasMessage("Value of property foo is not an integer: bar"); + } + } + + @Test + public void load_file_and_system_properties() throws Exception { + Env env = mock(Env.class); + File propsFile = new File(getClass().getResource("/org/sonar/application/PropsTest/sonar.properties").toURI()); + when(env.file("conf/sonar.properties")).thenReturn(propsFile); + + Props props = Props.create(env); + + assertThat(props.of("foo")).isEqualTo("bar"); + assertThat(props.of("java.version")).isNotNull(); + + // system properties override file properties + assertThat(props.of("java.io.tmpdir")).isNotEmpty().isNotEqualTo("/should/be/overridden"); + } + + @Test + public void fail_if_file_does_not_exist() throws Exception { + Env env = mock(Env.class); + when(env.file("conf/sonar.properties")).thenReturn(new File("target/not_exist/sonar.properties")); + + try { + Props.create(env); + fail(); + } catch (IllegalStateException e) { + assertThat(e).hasMessage("File does not exist or can't be open: target/not_exist/sonar.properties"); + } + } +} diff --git a/sonar-application/src/test/java/org/sonar/application/StartServerTest.java b/sonar-application/src/test/java/org/sonar/application/StartServerTest.java new file mode 100644 index 00000000000..868a51288d2 --- /dev/null +++ b/sonar-application/src/test/java/org/sonar/application/StartServerTest.java @@ -0,0 +1,150 @@ +/* + * SonarQube, open source software quality management tool. + * Copyright (C) 2008-2013 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.application; + +import com.github.kevinsawicki.http.HttpRequest; +import org.apache.commons.io.FileUtils; +import org.junit.After; +import org.junit.Before; +import org.junit.Test; + +import java.io.File; +import java.io.IOException; + +import static org.fest.assertions.Assertions.assertThat; +import static org.fest.assertions.Fail.fail; + +public class StartServerTest { + + Env env = new Env(locateFakeConfFile()); + StartServer starter = new StartServer(env); + + @Before + @After + public void clean_generated_dirs() throws IOException { + FileUtils.deleteDirectory(env.file("temp")); + FileUtils.deleteDirectory(env.file("logs")); + } + + @Test + public void start_server() throws Exception { + BackgroundThread background = new BackgroundThread(starter); + int port = 0; + try { + background.start(); + boolean started = false; + for (int i = 0; i < 100; i++) { + // Waiting for server to be started. + // A random and open port is used (see conf/sonar.properties) + Thread.sleep(500L); + if (verifyUp() && verifyLogs()) { + port = starter.port(); + started = true; + break; + } + } + assertThat(started).isTrue(); + } finally { + starter.stop(); + } + + // Server is down + try { + assertThat(HttpRequest.get("http://localhost:" + port).ok()).isFalse(); + } catch (HttpRequest.HttpRequestException e) { + // ok + } + } + + private boolean verifyUp() { + if (starter.port() > 0) { + String url = "http://localhost:" + starter.port() + "/index.html"; + HttpRequest request = HttpRequest.get(url); + if (request.ok() && "Hello World".equals(request.body(HttpRequest.CHARSET_UTF8))) { + return true; + } + } + return false; + } + + private boolean verifyLogs() { + File logFile = env.file("logs/access.log"); + return logFile.isFile() && logFile.exists() && logFile.length()>0; + } + + @Test + public void fail_if_started_twice() throws Exception { + BackgroundThread background = new BackgroundThread(starter); + try { + background.start(); + boolean started = false; + for (int i = 0; i < 100; i++) { + // Waiting for server to be started. + // A random and open port is used (see conf/sonar.properties) + Thread.sleep(500L); + if (starter.port() > 0) { + try { + starter.start(); + fail(); + } catch (IllegalStateException e) { + assertThat(e.getMessage()).isEqualTo("Tomcat is already started"); + started = true; + break; + } + } + } + assertThat(started).isTrue(); + + } finally { + starter.stop(); + } + } + + @Test + public void ignore_stop_if_not_running() throws Exception { + starter.stop(); + starter.stop(); + } + + private File locateFakeConfFile() { + File confFile = new File("src/test/fake-app/conf/sonar.properties"); + if (!confFile.exists()) { + confFile = new File("sonar-application/src/test/fake-app/conf/sonar.properties"); + } + return confFile; + } + + static class BackgroundThread extends Thread { + private StartServer server; + + BackgroundThread(StartServer server) { + this.server = server; + } + + @Override + public void run() { + try { + server.start(); + } catch (Exception e) { + throw new IllegalStateException(e); + } + } + } +} diff --git a/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/foo.xml b/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/foo.xml deleted file mode 100644 index e69de29bb2d..00000000000 --- a/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/foo.xml +++ /dev/null diff --git a/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin1.jar b/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin1.jar deleted file mode 100644 index e69de29bb2d..00000000000 --- a/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin1.jar +++ /dev/null diff --git a/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin2.jar b/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin2.jar deleted file mode 100644 index e69de29bb2d..00000000000 --- a/sonar-application/src/test/resources/org/sonar/application/JettyEmbedderTest/shouldLoadPluginsClasspath/plugin2.jar +++ /dev/null diff --git a/sonar-application/src/test/resources/org/sonar/application/LoggingTest/logback-access.xml b/sonar-application/src/test/resources/org/sonar/application/LoggingTest/logback-access.xml new file mode 100644 index 00000000000..298193e01fa --- /dev/null +++ b/sonar-application/src/test/resources/org/sonar/application/LoggingTest/logback-access.xml @@ -0,0 +1 @@ +<configuration/> diff --git a/sonar-application/src/test/resources/org/sonar/application/PropsTest/sonar.properties b/sonar-application/src/test/resources/org/sonar/application/PropsTest/sonar.properties new file mode 100644 index 00000000000..b73be15411b --- /dev/null +++ b/sonar-application/src/test/resources/org/sonar/application/PropsTest/sonar.properties @@ -0,0 +1,2 @@ +foo=bar +java.io.tmpdir=/should/be/overridden |