Skip to content
This repository has been archived by the owner on Oct 6, 2021. It is now read-only.

arnou/tomcat-slf4j-logback

 
 

Repository files navigation

Tomcat + SLF4J + Logback

Introduction

This project allows using SLF4J and Logback in Apache Tomcat absolutely without the need for commons-logging, log4j and java.util.logging.

This project's main and only goal is to allow the following:

  • redirect all org.apache.commons.logging (repackaged to org.apache.juli.logging) calls to org.slf4j (repackaged to org.apache.juli.logging.org.slf4j) - i.e. handle internal tomcat logging with slf4j with logback binding,
  • make still possible to use logback-access with logback-access.xml config - using standard functionality of logback-access,
  • make possible to use independent configuration of slf4j+logback from all web applications which may carry their own slf4j-api, logback-core and logback-classic in their WEB-INF/lib directory.

Using only ANT's build.xml file (based on the file provided with Tomcat), proper source JARs are downloaded from maven repository and unpacked. Then all classes are refactored under org.apache.juli.logging package and subpackages and then compiled.

To allow web applications to use their own slf4j-api and logback-classic, classes used by Tomcat (particularly jcl-over-slf4j) must go into different, non-standard packages. According to Tomcat Documentation web application looks up classes in their WEB-INF/classes directory and WEB-INF/lib/*.jar files before looking them in $TOMCAT_HOME/lib, but after looking them in system class loader. So Tomcat needs not only to have tomcat-juli replaced with tweaked jcl-over-slf4j but also the remaining JARs (slf4j-api, logback-core and logback-classic) must be refactored into different packages. Also, to put the three remaining JARs under Tomcat's system class loader, they must be referenced from tomcat-juli.jar using META-INF/MANIFEST.MF's Class-Path mechanism.

Finally, in order to keep the classpath clean, I've chosen the method of selecting Logback's configuration file using juli-logback.configurationFile system property. It is renamed in source files during refactoring phase. Leaving standard logback.configurationFile property would cause selecting this file in all web applications despite of having dedicated, classpath-based logback.xml configuration files.

There are four JARs involved in the process:

  • jcl-over-slf4j - this JAR is transformed into org.apache.juli.logging exactly the same way as commons-logging is transformed in Tomcat's build process. It is eventually compiled into tomcat-juli.jar - this name is mandatory, because it is directly referenced during Tomcat's startup process while constructing system class loader. This JAR is transformed and placed in $TOMCAT_HOME/bin/tomcat-juli.jar file.
  • slf4j-api - main SLF4J JAR. Transformed into $TOMCAT_HOME/bin/tomcat-juli-slf4j-api-<version>.jar file.
  • logback-core - core Logback JAR. Transformed into $TOMCAT_HOME/bin/tomcat-juli-logback-core-<version>.jar file.
  • logback-classic - actual SLF4J binding. Transformed into $TOMCAT_HOME/bin/tomcat-juli-logback-classic-<version>.jar file.

Installation

Before launching the build, make sure Ivy is installed (place proper ivy-X.Y.Z.jar into $ANT_HOME/lib directory). Place tomcat-juli.jar from $TOMCAT_HOME/bin of the Tomcat installation of your choice into _external directory - this is required, because we need few (namely: one) classes from the jar that is going to be replaced.

Type:

ant

And move four JARs from _dist directory to $TOMCAT_HOME/bin directory.

Notice: running ant on clean working copy will result in error about lack of tomcat-juli.jar file, which is needed during the process of tranforming JARs.

More detailed instruction:

  1. edit file build.properties, which may contain custom values for properties hardcoded in build.properties.default. e.g. tomcat.version
  2. run ant
  3. place correct version of tomcat-juli.jar in _external
  4. run ant again
  5. move JARs from _dist directory to $TOMCAT_HOME/bin.

After changing versions (e.g. for Tomcat), run ant clean.

Running Tomcat now will use default (very verbose) configuration of Logback. To change Logback's configuration, run Tomcat with the following system variable (using your favorite method of setting such variables - in catalina.sh, setenv.sh or other):

-Djuli-logback.configurationFile=file:<logback.xml location>

Configuration

Now you can configure whatever logging technology you want for your web applications. I recommend SLF4J and Logback because from now on, it will not collide with Tomcat's logging configuration.

While configuring Tomcat's logging, keep in mind that you have to use renamed packages in logback.xml config file, e.g.:

<configuration>
	<appender name="CONSOLE" class="org.apache.juli.logging.ch.qos.logback.core.ConsoleAppender">
		<encoder>
			<pattern>%d{HH:mm:ss.SSS} %-5level {%thread} [%logger{20}] : %msg%n</pattern>
		</encoder>
	</appender>
	<logger name="org.apache.catalina.core.ContainerBase.[Catalina].[localhost]" level="INFO" additivity="false">
		<appender-ref ref="FILE-LOCALHOST" />
	</logger>
	<root level="INFO">
		<appender-ref ref="CONSOLE" />
	</root>
</configuration>

Configuration of logback-access doesn't require renamed packages, as the required JARs are loaded from common class loader.

Sample logback.xml reflecting the configuration from standard $TOMCAT_HOME/conf/logging.properties can be found here.

Tomcat Customization

Tomcat 6.0.x

After unpacking apache-tomcat-6.0.x.tgz, one can run Tomcat by executing $TOMCAT_HOME/bin/startup.sh. This will cause running Tomcat with standard java.util.logging enabled. The standard commandline is:

"java" \
	-Djava.util.logging.config.file="$TOMCAT_HOME/conf/logging.properties"
	-Djava.util.logging.manager=org.apache.juli.ClassLoaderLogManager
	-Djava.endorsed.dirs="$TOMCAT_HOME/endorsed"
	-classpath "$TOMCAT_HOME\bin\bootstrap.jar"
	-Dcatalina.base="$TOMCAT_HOME"
	-Dcatalina.home="$TOMCAT_HOME"
	-Djava.io.tmpdir="$TOMCAT_HOME"
	org.apache.catalina.startup.Bootstrap start

Deleting $TOMCAT_HOME/conf/logging.properties will replace -Djava.util.logging.config.file with -Dnop - first step to remove j.u.logging. To get rid of -Djava.util.logging.manager we must explicitely set the following environment property in setenv.sh:

LOGGING_MANAGER=-Dnop

Finally we must configure our tomcat-slf4j-logback integration:

  • place all 4 JARs in $TOMCAT_HOME/bin
  • add -Djuli-logback.configurationFile=file:<logback.xml location> to $JAVA_OPTS in setenv.sh

Now Tomcat's internal logging goes through org.apache.juli.logging.org.slf4j and org.apache.juli.logging.ch.qos.logback to appenders configured in $TOMCAT_HOME/conf/logback.xml (or whatever file you set juli-logback.configurationFile variable to).

The final step is to configure logback-access. Now we don't have to deal with package manipulation. Just add:

<Valve className="ch.qos.logback.access.tomcat.LogbackValve" quiet="true"
	filename="${catalina.home}/conf/logback-access-localhost.xml" />

to $TOMCAT_HOME/conf/server.xml, place properly configured logback-access-localhost.xml on $TOMCAT_HOME/conf and place logback-core and logback-access JARs into $TOMCAT_HOME/lib. This won't cause problems with individual WARs' slf4j+logback configuration, because logback.xml is read by logback-classic which is recommended to reside in WEB-INF/lib. The only additional benefit is that WARs will see logback-core through common class loader.

Tomcat 7.0.x

Until LBACCESS-17 is resolved, there's a fix in build.xml which changes one source file in logback-access-0.9.29 sources to make them compatible with changed Valves API from Tomcat 7.

This results in logback-access-0.9.29-tomcat-7.0.16.jar being created in _dist directory which must be copied to $TOMCAT_HOME/lib. For Tomcat version below 7.0.0, the fix is not produced, and one can place original logback-access-0.9.29.jar into $TOMCAT_HOME/lib.

Besides that everything else is working exactly as with Tomcat 6.

About

Tomcat, SLF4J and Logback integration

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published