diff options
author | Sébastien Lesaint <sebastien.lesaint@sonarsource.com> | 2016-11-14 14:39:22 +0100 |
---|---|---|
committer | Sébastien Lesaint <sebastien.lesaint@sonarsource.com> | 2016-11-16 10:09:21 +0100 |
commit | ad40128e3d6b300eef57e24180fc22d2e374f125 (patch) | |
tree | 8e5b955e96a969d52b1c997e711b364b33f19fb7 /server | |
parent | 7c6a909d760e457af4259bf49bfbc6e1902d631b (diff) | |
download | sonarqube-ad40128e3d6b300eef57e24180fc22d2e374f125.tar.gz sonarqube-ad40128e3d6b300eef57e24180fc22d2e374f125.zip |
SONAR-8333 make APP logging configuration support 4 log files
add support for option sonar.wrapped=true to detect when SQ is started by JWP (wrapper) and correctly support all logging use cases
include refactoring at LogBackHelper level to simplify code
refactor applying log level property to make all process support only INFO, DEBUG and TRACE
Diffstat (limited to 'server')
12 files changed, 274 insertions, 143 deletions
diff --git a/server/sonar-process/src/main/java/org/sonar/process/LogbackHelper.java b/server/sonar-process/src/main/java/org/sonar/process/LogbackHelper.java index d4496f747c1..a669c91a9ce 100644 --- a/server/sonar-process/src/main/java/org/sonar/process/LogbackHelper.java +++ b/server/sonar-process/src/main/java/org/sonar/process/LogbackHelper.java @@ -30,33 +30,40 @@ import ch.qos.logback.classic.spi.LoggerContextListener; import ch.qos.logback.core.ConsoleAppender; import ch.qos.logback.core.Context; import ch.qos.logback.core.FileAppender; -import ch.qos.logback.core.filter.Filter; import ch.qos.logback.core.joran.spi.JoranException; import ch.qos.logback.core.rolling.FixedWindowRollingPolicy; import ch.qos.logback.core.rolling.RollingFileAppender; import ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy; import ch.qos.logback.core.rolling.TimeBasedRollingPolicy; import java.io.File; +import java.util.Arrays; +import java.util.HashSet; +import java.util.Set; import javax.annotation.CheckForNull; import org.apache.commons.lang.StringUtils; import org.slf4j.LoggerFactory; +import static java.util.Collections.unmodifiableSet; import static java.util.Objects.requireNonNull; +import static org.slf4j.Logger.ROOT_LOGGER_NAME; /** * Helps to configure Logback in a programmatic way, without using XML. */ public class LogbackHelper { - public static final String ROLLING_POLICY_PROPERTY = "sonar.log.rollingPolicy"; - public static final String MAX_FILES_PROPERTY = "sonar.log.maxFiles"; + private static final String ALL_LOGS_TO_CONSOLE_PROPERTY = "sonar.log.console"; + private static final String SONAR_LOG_LEVEL_PROPERTY = "sonar.log.level"; + private static final String ROLLING_POLICY_PROPERTY = "sonar.log.rollingPolicy"; + private static final String MAX_FILES_PROPERTY = "sonar.log.maxFiles"; private static final String PROCESS_NAME_PLACEHOLDER = "XXXX"; private static final String THREAD_ID_PLACEHOLDER = "ZZZZ"; private static final String LOG_FORMAT = "%d{yyyy.MM.dd HH:mm:ss} %-5level " + PROCESS_NAME_PLACEHOLDER + "[" + THREAD_ID_PLACEHOLDER + "][%logger{20}] %msg%n"; + public static final Set<Level> ALLOWED_ROOT_LOG_LEVELS = unmodifiableSet(setOf(Level.TRACE, Level.DEBUG, Level.INFO)); public LoggerContext getRootContext() { org.slf4j.Logger logger; - while (!((logger = LoggerFactory.getLogger(Logger.ROOT_LOGGER_NAME)) instanceof Logger)) { + while (!((logger = LoggerFactory.getLogger(ROOT_LOGGER_NAME)) instanceof Logger)) { // It occurs when the initialization step is still not finished because of a race condition // on ILoggerFactory.getILoggerFactory // http://jira.qos.ch/browse/SLF4J-167 @@ -83,14 +90,12 @@ public class LogbackHelper { public static final class RootLoggerConfig { private final String processName; private final String threadIdFieldPattern; - private final String fileName; - private final boolean detachConsole; + private final String fileNamePrefix; - public RootLoggerConfig(Builder builder) { + private RootLoggerConfig(Builder builder) { this.processName = builder.processName; this.threadIdFieldPattern = builder.threadIdFieldPattern; - this.fileName = builder.fileName; - this.detachConsole = builder.detachConsole; + this.fileNamePrefix = builder.fileNamePrefix; } public static Builder newRootLoggerConfigBuilder() { @@ -101,16 +106,12 @@ public class LogbackHelper { return processName; } - public String getThreadIdFieldPattern() { + String getThreadIdFieldPattern() { return threadIdFieldPattern; } - public String getFileName() { - return fileName; - } - - public boolean isDetachConsole() { - return detachConsole; + String getFileNamePrefix() { + return fileNamePrefix; } public static final class Builder { @@ -118,8 +119,7 @@ public class LogbackHelper { public String processName; private String threadIdFieldPattern = ""; @CheckForNull - private String fileName; - private boolean detachConsole = true; + private String fileNamePrefix; private Builder() { // prevents instantiation outside RootLoggerConfig, use static factory method @@ -136,15 +136,15 @@ public class LogbackHelper { return this; } - public Builder setFileName(String fileName) { - checkFileName(fileName); - this.fileName = fileName; + public Builder setFileNamePrefix(String fileNamePrefix) { + checkFileName(fileNamePrefix); + this.fileNamePrefix = fileNamePrefix; return this; } private static void checkFileName(String fileName) { - if (requireNonNull(fileName, "fileName can't be null").isEmpty()) { - throw new IllegalArgumentException("fileName can't be empty"); + if (requireNonNull(fileName, "fileNamePrefix can't be null").isEmpty()) { + throw new IllegalArgumentException("fileNamePrefix can't be empty"); } } @@ -154,60 +154,140 @@ public class LogbackHelper { } } - public Builder setDetachConsole(boolean detachConsole) { - this.detachConsole = detachConsole; - return this; - } - public RootLoggerConfig build() { checkProcessName(this.processName); - checkFileName(this.fileName); + checkFileName(this.fileNamePrefix); return new RootLoggerConfig(this); } } } - public void configureRootLogger(LoggerContext ctx, Props props, RootLoggerConfig config) { - String logFormat = LOG_FORMAT - .replace(PROCESS_NAME_PLACEHOLDER, config.getProcessName()) - .replace(THREAD_ID_PLACEHOLDER, config.getThreadIdFieldPattern()); - // configure appender - LogbackHelper.RollingPolicy rollingPolicy = createRollingPolicy(ctx, props, config.getFileName()); - FileAppender<ILoggingEvent> fileAppender = rollingPolicy.createAppender("file"); - fileAppender.setContext(ctx); - PatternLayoutEncoder fileEncoder = new PatternLayoutEncoder(); - fileEncoder.setContext(ctx); - fileEncoder.setPattern(logFormat); - fileEncoder.start(); - fileAppender.setEncoder(fileEncoder); - fileAppender.start(); - - // configure logger - Logger rootLogger = ctx.getLogger(ch.qos.logback.classic.Logger.ROOT_LOGGER_NAME); - rootLogger.addAppender(fileAppender); -// if (config.isDetachConsole()) { -// rootLogger.detachAppender("console"); -// } - rootLogger.addAppender(newConsoleAppender(ctx, "console", logFormat)); + public String buildLogPattern(LogbackHelper.RootLoggerConfig config) { + return LOG_FORMAT + .replace(PROCESS_NAME_PLACEHOLDER, config.getProcessName()) + .replace(THREAD_ID_PLACEHOLDER, config.getThreadIdFieldPattern()); } - public ConsoleAppender newConsoleAppender(Context loggerContext, String name, String pattern, Filter... filters) { + /** + * Creates a new {@link ConsoleAppender} to {@code System.out} with the specified name and log pattern. + * + * @see #buildLogPattern(RootLoggerConfig) + */ + public ConsoleAppender<ILoggingEvent> newConsoleAppender(Context loggerContext, String name, String logPattern) { PatternLayoutEncoder consoleEncoder = new PatternLayoutEncoder(); consoleEncoder.setContext(loggerContext); - consoleEncoder.setPattern(pattern); + consoleEncoder.setPattern(logPattern); consoleEncoder.start(); - ConsoleAppender consoleAppender = new ConsoleAppender(); + ConsoleAppender<ILoggingEvent> consoleAppender = new ConsoleAppender<>(); consoleAppender.setContext(loggerContext); consoleAppender.setEncoder(consoleEncoder); consoleAppender.setName(name); consoleAppender.setTarget("System.out"); - for (Filter filter : filters) { - consoleAppender.addFilter(filter); - } consoleAppender.start(); return consoleAppender; } + /** + * Make logback configuration for a process to push all its logs to a log file. + * <p> + * <ul> + * <li>the file's name will use the prefix defined in {@link RootLoggerConfig#getFileNamePrefix()}.</li> + * <li>the file will follow the rotation policy defined in property {@link #ROLLING_POLICY_PROPERTY} and + * the max number of files defined in property {@link #MAX_FILES_PROPERTY}</li> + * <li>the logs will follow the specified log pattern</li> + * </ul> + * </p> + * + * @see #buildLogPattern(RootLoggerConfig) + */ + public FileAppender<ILoggingEvent> configureGlobalFileLog(Props props, RootLoggerConfig config, String logPattern) { + LoggerContext ctx = getRootContext(); + Logger rootLogger = ctx.getLogger(ROOT_LOGGER_NAME); + FileAppender<ILoggingEvent> fileAppender = newFileAppender(ctx, props, config, logPattern); + rootLogger.addAppender(fileAppender); + return fileAppender; + } + + public FileAppender<ILoggingEvent> newFileAppender(LoggerContext ctx, Props props, LogbackHelper.RootLoggerConfig config, String logPattern) { + RollingPolicy rollingPolicy = createRollingPolicy(ctx, props, config.getFileNamePrefix()); + FileAppender<ILoggingEvent> fileAppender = rollingPolicy.createAppender("file_" + config.getFileNamePrefix()); + fileAppender.setContext(ctx); + PatternLayoutEncoder fileEncoder = new PatternLayoutEncoder(); + fileEncoder.setContext(ctx); + fileEncoder.setPattern(logPattern); + fileEncoder.start(); + fileAppender.setEncoder(fileEncoder); + fileAppender.start(); + return fileAppender; + } + + /** + * Make the logback configuration for a sub process to correctly push all its logs to be read by a stream gobbler + * on the sub process's System.out. + * + * @see #buildLogPattern(RootLoggerConfig) + */ + public void configureForSubprocessGobbler(Props props, String logPattern) { + if (isAllLogsToConsoleEnabled(props)) { + LoggerContext ctx = getRootContext(); + ctx.getLogger(ROOT_LOGGER_NAME).addAppender(newConsoleAppender(ctx, "root_console", logPattern)); + } + } + + /** + * Finds out whether we are in testing environment (usually ITs) and logs of all processes must be forward to + * App's System.out. This is specified by the value of property {@link #ALL_LOGS_TO_CONSOLE_PROPERTY}. + */ + public boolean isAllLogsToConsoleEnabled(Props props) { + return props.valueAsBoolean(ALL_LOGS_TO_CONSOLE_PROPERTY, false); + } + + @SafeVarargs + private static <T> Set<T> setOf(T... args) { + Set<T> res = new HashSet<>(args.length); + res.addAll(Arrays.asList(args)); + return res; + } + + /** + * Configure the log level of the root logger reading the value of property {@link #SONAR_LOG_LEVEL_PROPERTY}. + * + * @throws IllegalArgumentException if the value of {@link #SONAR_LOG_LEVEL_PROPERTY} is not one of {@link #ALLOWED_ROOT_LOG_LEVELS} + */ + public Level configureRootLogLevel(Props props) { + return configureRootLogLevel(props, SONAR_LOG_LEVEL_PROPERTY); + } + + /** + * Configure the log level of the root logger reading the value of specified property. + * + * @throws IllegalArgumentException if the value of the specified property is not one of {@link #ALLOWED_ROOT_LOG_LEVELS} + */ + public Level configureRootLogLevel(Props props, String propertyKey) { + Level newLevel = Level.toLevel(props.value(propertyKey, Level.INFO.toString()), Level.INFO); + return configureRootLogLevel(newLevel); + } + + /** + * Configure the log level of the root logger to the specified level. + * + * @throws IllegalArgumentException if the specified level is not one of {@link #ALLOWED_ROOT_LOG_LEVELS} + */ + public Level configureRootLogLevel(Level newLevel) { + Logger rootLogger = getRootContext().getLogger(ROOT_LOGGER_NAME); + if (!ALLOWED_ROOT_LOG_LEVELS.contains(newLevel)) { + throw new IllegalArgumentException(String.format("%s log level is not supported (allowed levels are %s)", newLevel, ALLOWED_ROOT_LOG_LEVELS)); + } + rootLogger.setLevel(newLevel); + return newLevel; + } + + /** + * Configure the log level of the specified logger to specified level. + * <p> + * Any level is allowed. + * </p> + */ public Logger configureLogger(String loggerName, Level level) { Logger logger = getRootContext().getLogger(loggerName); logger.setLevel(level); @@ -246,31 +326,31 @@ public class LogbackHelper { public abstract static class RollingPolicy { protected final Context context; - protected final String filenamePrefix; - protected final File logsDir; - protected final int maxFiles; + final String filenamePrefix; + final File logsDir; + final int maxFiles; - public RollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles) { + RollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles) { this.context = context; this.filenamePrefix = filenamePrefix; this.logsDir = logsDir; this.maxFiles = maxFiles; } - public abstract FileAppender createAppender(String appenderName); + public abstract FileAppender<ILoggingEvent> createAppender(String appenderName); } /** * Log files are not rotated, for example for unix command logrotate is in place. */ - static class NoRollingPolicy extends RollingPolicy { - NoRollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles) { + private static class NoRollingPolicy extends RollingPolicy { + private NoRollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles) { super(context, filenamePrefix, logsDir, maxFiles); } @Override - public FileAppender createAppender(String appenderName) { - FileAppender appender = new FileAppender<>(); + public FileAppender<ILoggingEvent> createAppender(String appenderName) { + FileAppender<ILoggingEvent> appender = new FileAppender<>(); appender.setContext(context); appender.setFile(new File(logsDir, filenamePrefix + ".log").getAbsolutePath()); appender.setName(appenderName); @@ -282,17 +362,17 @@ public class LogbackHelper { * Log files are rotated according to time (one file per day, month or year). * See http://logback.qos.ch/manual/appenders.html#TimeBasedRollingPolicy */ - static class TimeRollingPolicy extends RollingPolicy { + private static class TimeRollingPolicy extends RollingPolicy { private final String datePattern; - public TimeRollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles, String datePattern) { + private TimeRollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles, String datePattern) { super(context, filenamePrefix, logsDir, maxFiles); this.datePattern = datePattern; } @Override - public FileAppender createAppender(String appenderName) { - RollingFileAppender appender = new RollingFileAppender(); + public FileAppender<ILoggingEvent> createAppender(String appenderName) { + RollingFileAppender<ILoggingEvent> appender = new RollingFileAppender<>(); appender.setContext(context); appender.setName(appenderName); String filePath = new File(logsDir, filenamePrefix + ".log").getAbsolutePath(); @@ -314,23 +394,23 @@ public class LogbackHelper { * Log files are rotated according to their size. * See http://logback.qos.ch/manual/appenders.html#FixedWindowRollingPolicy */ - static class SizeRollingPolicy extends RollingPolicy { + private static class SizeRollingPolicy extends RollingPolicy { private final String size; - SizeRollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles, String parameter) { + private SizeRollingPolicy(Context context, String filenamePrefix, File logsDir, int maxFiles, String parameter) { super(context, filenamePrefix, logsDir, maxFiles); this.size = parameter; } @Override - public FileAppender createAppender(String appenderName) { - RollingFileAppender appender = new RollingFileAppender(); + public FileAppender<ILoggingEvent> createAppender(String appenderName) { + RollingFileAppender<ILoggingEvent> appender = new RollingFileAppender<>(); appender.setContext(context); appender.setName(appenderName); String filePath = new File(logsDir, filenamePrefix + ".log").getAbsolutePath(); appender.setFile(filePath); - SizeBasedTriggeringPolicy trigger = new SizeBasedTriggeringPolicy(size); + SizeBasedTriggeringPolicy<ILoggingEvent> trigger = new SizeBasedTriggeringPolicy<>(size); trigger.setContext(context); trigger.start(); appender.setTriggeringPolicy(trigger); diff --git a/server/sonar-process/src/test/java/org/sonar/process/LogbackHelperTest.java b/server/sonar-process/src/test/java/org/sonar/process/LogbackHelperTest.java index 389a4c4f395..3ffc86c64e9 100644 --- a/server/sonar-process/src/test/java/org/sonar/process/LogbackHelperTest.java +++ b/server/sonar-process/src/test/java/org/sonar/process/LogbackHelperTest.java @@ -27,7 +27,6 @@ import ch.qos.logback.classic.spi.LoggerContextListener; import ch.qos.logback.core.Appender; import ch.qos.logback.core.ConsoleAppender; import ch.qos.logback.core.FileAppender; -import ch.qos.logback.core.filter.Filter; import ch.qos.logback.core.rolling.FixedWindowRollingPolicy; import ch.qos.logback.core.rolling.RollingFileAppender; import ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy; @@ -42,7 +41,6 @@ import org.junit.rules.TemporaryFolder; import static org.assertj.core.api.Assertions.assertThat; import static org.junit.Assert.fail; -import static org.mockito.Mockito.mock; public class LogbackHelperTest { @@ -92,15 +90,6 @@ public class LogbackHelperTest { } @Test - public void newConsoleAppender_with_filter() { - Filter filter = mock(Filter.class); - LoggerContext ctx = underTest.getRootContext(); - ConsoleAppender<?> appender = underTest.newConsoleAppender(ctx, "MY_APPENDER", "%msg%n", filter); - - assertThat(appender.getCopyOfAttachedFiltersList()).containsOnly(filter); - } - - @Test public void configureLogger() { Logger logger = underTest.configureLogger("my_logger", Level.WARN); diff --git a/server/sonar-search/src/main/java/org/sonar/search/SearchLogging.java b/server/sonar-search/src/main/java/org/sonar/search/SearchLogging.java index 4667d67f051..62c710e929a 100644 --- a/server/sonar-search/src/main/java/org/sonar/search/SearchLogging.java +++ b/server/sonar-search/src/main/java/org/sonar/search/SearchLogging.java @@ -33,8 +33,14 @@ public class SearchLogging { LoggerContext ctx = helper.getRootContext(); ctx.reset(); - helper.configureRootLogger(ctx, props, newRootLoggerConfigBuilder().setProcessName("es").setFileName("es").build()); + LogbackHelper.RootLoggerConfig config = newRootLoggerConfigBuilder().setProcessName("es").setFileNamePrefix("es").build(); + + String logPattern = helper.buildLogPattern(config); + helper.configureGlobalFileLog(props, config, logPattern); + helper.configureForSubprocessGobbler(props, logPattern); + helper.configureRootLogLevel(props); return ctx; } + } diff --git a/server/sonar-search/src/test/java/org/sonar/search/SearchLoggingTest.java b/server/sonar-search/src/test/java/org/sonar/search/SearchLoggingTest.java index ab229e74a76..cd4533691b9 100644 --- a/server/sonar-search/src/test/java/org/sonar/search/SearchLoggingTest.java +++ b/server/sonar-search/src/test/java/org/sonar/search/SearchLoggingTest.java @@ -73,7 +73,7 @@ public class SearchLoggingTest { LoggerContext ctx = underTest.configure(props); Logger root = ctx.getLogger(Logger.ROOT_LOGGER_NAME); - Appender<ILoggingEvent> appender = root.getAppender("file"); + Appender<ILoggingEvent> appender = root.getAppender("file_es"); assertThat(appender).isInstanceOf(FileAppender.class); FileAppender fileAppender = (FileAppender) appender; assertThat(fileAppender.getFile()).isEqualTo(new File(logDir, "es.log").getAbsolutePath()); diff --git a/server/sonar-server/src/main/java/org/sonar/ce/log/CeProcessLogging.java b/server/sonar-server/src/main/java/org/sonar/ce/log/CeProcessLogging.java index 28dc43956c2..201878865df 100644 --- a/server/sonar-server/src/main/java/org/sonar/ce/log/CeProcessLogging.java +++ b/server/sonar-server/src/main/java/org/sonar/ce/log/CeProcessLogging.java @@ -19,6 +19,7 @@ */ package org.sonar.ce.log; +import org.sonar.process.Props; import org.sonar.server.app.ServerProcessLogging; /** @@ -30,4 +31,8 @@ public class CeProcessLogging extends ServerProcessLogging { super("ce", "%X{ceTaskUuid}"); } + @Override + protected void extendConfiguration(Props props) { + // no configuration extension to do + } } diff --git a/server/sonar-server/src/main/java/org/sonar/server/app/ServerProcessLogging.java b/server/sonar-server/src/main/java/org/sonar/server/app/ServerProcessLogging.java index 72aea06c92a..ec9f4e565d4 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/app/ServerProcessLogging.java +++ b/server/sonar-server/src/main/java/org/sonar/server/app/ServerProcessLogging.java @@ -20,10 +20,6 @@ package org.sonar.server.app; import ch.qos.logback.classic.LoggerContext; -import java.util.logging.LogManager; -import org.slf4j.bridge.SLF4JBridgeHandler; -import org.sonar.api.utils.MessageException; -import org.sonar.api.utils.log.LoggerLevel; import org.sonar.process.LogbackHelper; import org.sonar.process.Props; import org.sonar.server.platform.ServerLogging; @@ -31,7 +27,6 @@ import org.sonar.server.platform.ServerLogging; import static org.sonar.process.LogbackHelper.RootLoggerConfig.newRootLoggerConfigBuilder; public abstract class ServerProcessLogging { - private static final String LOG_LEVEL_PROPERTY = "sonar.log.level"; private final String processName; private final String threadIdFieldPattern; private final LogbackHelper helper = new LogbackHelper(); @@ -46,32 +41,28 @@ public abstract class ServerProcessLogging { ctx.reset(); helper.enableJulChangePropagation(ctx); - helper.configureRootLogger(ctx, props, - newRootLoggerConfigBuilder() - .setProcessName(processName) - .setThreadIdFieldPattern(threadIdFieldPattern) - .setFileName(processName) - .build()); - configureLevels(props); + configureRootLogger(props); + + extendConfiguration(props); - // Configure java.util.logging, used by Tomcat, in order to forward to slf4j - LogManager.getLogManager().reset(); - SLF4JBridgeHandler.install(); return ctx; } - private void configureLevels(Props props) { - String levelCode = props.value(LOG_LEVEL_PROPERTY, "INFO"); - LoggerLevel level; - if ("TRACE".equals(levelCode)) { - level = LoggerLevel.TRACE; - } else if ("DEBUG".equals(levelCode)) { - level = LoggerLevel.DEBUG; - } else if ("INFO".equals(levelCode)) { - level = LoggerLevel.INFO; - } else { - throw MessageException.of(String.format("Unsupported log level: %s. Please check property %s", levelCode, LOG_LEVEL_PROPERTY)); - } - ServerLogging.configureLevels(helper, level); + protected abstract void extendConfiguration(Props props); + + private void configureRootLogger(Props props) { + LogbackHelper.RootLoggerConfig config = newRootLoggerConfigBuilder() + .setProcessName(processName) + .setThreadIdFieldPattern(threadIdFieldPattern) + .setFileNamePrefix(processName) + .build(); + String logPattern = helper.buildLogPattern(config); + + helper.configureGlobalFileLog(props, config, logPattern); + helper.configureForSubprocessGobbler(props, logPattern); + + helper.configureRootLogLevel(props); + ServerLogging.configureHardcodedLevels(helper); } + } diff --git a/server/sonar-server/src/main/java/org/sonar/server/app/WebServerProcessLogging.java b/server/sonar-server/src/main/java/org/sonar/server/app/WebServerProcessLogging.java index 511aba2c5e5..f10f249db8d 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/app/WebServerProcessLogging.java +++ b/server/sonar-server/src/main/java/org/sonar/server/app/WebServerProcessLogging.java @@ -19,6 +19,10 @@ */ package org.sonar.server.app; +import java.util.logging.LogManager; +import org.slf4j.bridge.SLF4JBridgeHandler; +import org.sonar.process.Props; + /** * Configure logback for the Web Server process. Logs are written to file "web.log" in SQ's log directory. */ @@ -28,4 +32,10 @@ public class WebServerProcessLogging extends ServerProcessLogging { super("web", "%X{UID}"); } + @Override + protected void extendConfiguration(Props props) { + // Configure java.util.logging, used by Tomcat, in order to forward to slf4j + LogManager.getLogManager().reset(); + SLF4JBridgeHandler.install(); + } } diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/ServerLogging.java b/server/sonar-server/src/main/java/org/sonar/server/platform/ServerLogging.java index 2c15b75735f..5fe0cbb7d3b 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/platform/ServerLogging.java +++ b/server/sonar-server/src/main/java/org/sonar/server/platform/ServerLogging.java @@ -21,13 +21,11 @@ package org.sonar.server.platform; import ch.qos.logback.classic.Level; import ch.qos.logback.classic.Logger; -import com.google.common.base.Preconditions; -import com.google.common.collect.Sets; +import com.google.common.annotations.VisibleForTesting; import java.io.File; -import java.util.Set; import org.slf4j.LoggerFactory; -import org.sonar.api.config.Settings; import org.sonar.api.ce.ComputeEngineSide; +import org.sonar.api.config.Settings; import org.sonar.api.server.ServerSide; import org.sonar.api.utils.log.LoggerLevel; import org.sonar.api.utils.log.Loggers; @@ -38,17 +36,23 @@ import org.sonar.process.ProcessProperties; @ComputeEngineSide public class ServerLogging { - public static final Set<LoggerLevel> ALLOWED_ROOT_LOG_LEVELS = Sets.immutableEnumSet(LoggerLevel.TRACE, LoggerLevel.DEBUG, LoggerLevel.INFO); - - private final LogbackHelper helper = new LogbackHelper(); + private final LogbackHelper helper; private final Settings settings; public ServerLogging(Settings settings) { + this(new LogbackHelper(), settings); + } + + @VisibleForTesting + ServerLogging(LogbackHelper helper, Settings settings) { + this.helper = helper; this.settings = settings; } public void changeLevel(LoggerLevel level) { - configureLevels(helper, level); + Level logbackLevel = Level.toLevel(level.name()); + helper.configureRootLogLevel(logbackLevel); + configureHardcodedLevels(helper); LoggerFactory.getLogger(ServerLogging.class).info("Level of logs changed to {}", level); } @@ -56,9 +60,7 @@ public class ServerLogging { return Loggers.get(Logger.ROOT_LOGGER_NAME).getLevel(); } - public static void configureLevels(LogbackHelper helper, LoggerLevel level) { - Preconditions.checkArgument(ALLOWED_ROOT_LOG_LEVELS.contains(level), "%s log level is not supported (allowed levels are %s)", level, ALLOWED_ROOT_LOG_LEVELS); - helper.configureLogger(Logger.ROOT_LOGGER_NAME, Level.toLevel(level.name())); + public static void configureHardcodedLevels(LogbackHelper helper) { helper.configureLogger("rails", Level.WARN); helper.configureLogger("org.apache.ibatis", Level.WARN); helper.configureLogger("java.sql", Level.WARN); diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/ws/ChangeLogLevelAction.java b/server/sonar-server/src/main/java/org/sonar/server/platform/ws/ChangeLogLevelAction.java index 4948a15bfae..e64eac0db8f 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/platform/ws/ChangeLogLevelAction.java +++ b/server/sonar-server/src/main/java/org/sonar/server/platform/ws/ChangeLogLevelAction.java @@ -28,6 +28,8 @@ import org.sonar.db.Database; import org.sonar.server.platform.ServerLogging; import org.sonar.server.user.UserSession; +import static org.sonar.process.LogbackHelper.ALLOWED_ROOT_LOG_LEVELS; + public class ChangeLogLevelAction implements SystemWsAction { private static final String PARAM_LEVEL = "level"; @@ -55,7 +57,7 @@ public class ChangeLogLevelAction implements SystemWsAction { newAction.createParam(PARAM_LEVEL) .setDescription("The new level. Be cautious: DEBUG, and even more TRACE, may have performance impacts.") - .setPossibleValues(ServerLogging.ALLOWED_ROOT_LOG_LEVELS) + .setPossibleValues(ALLOWED_ROOT_LOG_LEVELS) .setRequired(true); } diff --git a/server/sonar-server/src/test/java/org/sonar/ce/log/CeProcessLoggingTest.java b/server/sonar-server/src/test/java/org/sonar/ce/log/CeProcessLoggingTest.java index b2e39b3e5d8..6b2dbfd704d 100644 --- a/server/sonar-server/src/test/java/org/sonar/ce/log/CeProcessLoggingTest.java +++ b/server/sonar-server/src/test/java/org/sonar/ce/log/CeProcessLoggingTest.java @@ -77,7 +77,7 @@ public class CeProcessLoggingTest { LoggerContext ctx = underTest.configure(props); Logger root = ctx.getLogger(Logger.ROOT_LOGGER_NAME); - Appender<ILoggingEvent> appender = root.getAppender("file"); + Appender<ILoggingEvent> appender = root.getAppender("file_ce"); assertThat(appender).isInstanceOf(FileAppender.class); FileAppender fileAppender = (FileAppender) appender; assertThat(fileAppender.getFile()).isEqualTo(new File(logDir, "ce.log").getAbsolutePath()); diff --git a/server/sonar-server/src/test/java/org/sonar/server/app/WebServerProcessLoggingTest.java b/server/sonar-server/src/test/java/org/sonar/server/app/WebServerProcessLoggingTest.java index 170ea8b6c11..0ab5bc96715 100644 --- a/server/sonar-server/src/test/java/org/sonar/server/app/WebServerProcessLoggingTest.java +++ b/server/sonar-server/src/test/java/org/sonar/server/app/WebServerProcessLoggingTest.java @@ -77,7 +77,7 @@ public class WebServerProcessLoggingTest { LoggerContext ctx = underTest.configure(props); Logger root = ctx.getLogger(Logger.ROOT_LOGGER_NAME); - Appender<ILoggingEvent> appender = root.getAppender("file"); + Appender<ILoggingEvent> appender = root.getAppender("file_web"); assertThat(appender).isInstanceOf(FileAppender.class); FileAppender fileAppender = (FileAppender) appender; assertThat(fileAppender.getFile()).isEqualTo(new File(logDir, "web.log").getAbsolutePath()); diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/ServerLoggingTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/ServerLoggingTest.java index d31ce54dfc8..c3b6f9b8d68 100644 --- a/server/sonar-server/src/test/java/org/sonar/server/platform/ServerLoggingTest.java +++ b/server/sonar-server/src/test/java/org/sonar/server/platform/ServerLoggingTest.java @@ -20,7 +20,6 @@ package org.sonar.server.platform; import ch.qos.logback.classic.Level; -import ch.qos.logback.classic.Logger; import java.io.File; import java.io.IOException; import org.apache.commons.io.FileUtils; @@ -28,8 +27,8 @@ import org.junit.Rule; import org.junit.Test; import org.junit.rules.ExpectedException; import org.junit.rules.TemporaryFolder; -import org.sonar.api.config.Settings; import org.sonar.api.config.MapSettings; +import org.sonar.api.config.Settings; import org.sonar.api.utils.log.LogTester; import org.sonar.api.utils.log.LoggerLevel; import org.sonar.process.LogbackHelper; @@ -37,18 +36,19 @@ import org.sonar.process.ProcessProperties; import static org.assertj.core.api.Assertions.assertThat; import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.spy; import static org.mockito.Mockito.verify; public class ServerLoggingTest { @Rule public TemporaryFolder temp = new TemporaryFolder(); - @Rule public ExpectedException expectedException = ExpectedException.none(); - Settings settings = new MapSettings(); - ServerLogging underTest = new ServerLogging(settings); + private LogbackHelper logbackHelper = spy(new LogbackHelper()); + private Settings settings = new MapSettings(); + private ServerLogging underTest = new ServerLogging(logbackHelper, settings); @Rule public LogTester logTester = new LogTester(); @@ -78,20 +78,66 @@ public class ServerLoggingTest { } @Test - public void configureLevels() { + public void configureHardcodedLevels() { LogbackHelper logbackHelper = mock(LogbackHelper.class); - ServerLogging.configureLevels(logbackHelper, LoggerLevel.TRACE); + ServerLogging.configureHardcodedLevels(logbackHelper); - verify(logbackHelper).configureLogger(Logger.ROOT_LOGGER_NAME, Level.TRACE); - verify(logbackHelper).configureLogger("java.sql", Level.WARN); + verifyHardcodedLevels(logbackHelper); } @Test - public void configureLevels_unsupported_level() { + public void changeLevel_throws_IAE_if_level_is_WARN() { expectedException.expect(IllegalArgumentException.class); - expectedException.expectMessage("ERROR log level is not supported (allowed levels are [TRACE, DEBUG, INFO])"); + expectedException.expectMessage("WARN log level is not supported (allowed levels are ["); - LogbackHelper logbackHelper = mock(LogbackHelper.class); - ServerLogging.configureLevels(logbackHelper, LoggerLevel.ERROR); + underTest.changeLevel(LoggerLevel.WARN); + } + + @Test + public void changeLevel_throws_IAE_if_level_is_ERROR() { + expectedException.expect(IllegalArgumentException.class); + expectedException.expectMessage("ERROR log level is not supported (allowed levels are ["); + + underTest.changeLevel(LoggerLevel.ERROR); + } + + @Test + public void changeLevel_changes_root_logger_level_to_INFO() { + underTest.changeLevel(LoggerLevel.INFO); + + verify(logbackHelper).configureRootLogLevel(Level.INFO); + verifyHardcodedLevels(logbackHelper); + } + + @Test + public void changeLevel_changes_root_logger_level_to_DEBUG() { + underTest.changeLevel(LoggerLevel.DEBUG); + + verify(logbackHelper).configureRootLogLevel(Level.DEBUG); + verifyHardcodedLevels(logbackHelper); + } + + @Test + public void changeLevel_changes_root_logger_level_to_TRACE() { + underTest.changeLevel(LoggerLevel.TRACE); + + verify(logbackHelper).configureRootLogLevel(Level.TRACE); + verifyHardcodedLevels(logbackHelper); + } + + private void verifyHardcodedLevels(LogbackHelper logbackHelper) { + verify(logbackHelper).configureLogger("rails", Level.WARN); + verify(logbackHelper).configureLogger("org.apache.ibatis", Level.WARN); + verify(logbackHelper).configureLogger("java.sql", Level.WARN); + verify(logbackHelper).configureLogger("java.sql.ResultSet", Level.WARN); + verify(logbackHelper).configureLogger("org.sonar.MEASURE_FILTER", Level.WARN); + verify(logbackHelper).configureLogger("org.elasticsearch", Level.INFO); + verify(logbackHelper).configureLogger("org.elasticsearch.node", Level.INFO); + verify(logbackHelper).configureLogger("org.elasticsearch.http", Level.INFO); + verify(logbackHelper).configureLogger("ch.qos.logback", Level.WARN); + verify(logbackHelper).configureLogger("org.apache.catalina", Level.INFO); + verify(logbackHelper).configureLogger("org.apache.coyote", Level.INFO); + verify(logbackHelper).configureLogger("org.apache.jasper", Level.INFO); + verify(logbackHelper).configureLogger("org.apache.tomcat", Level.INFO); } } |