| <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> |
| <!-- |
| | (Unfortunately) copied from the Fluido skin to allow the footer to be centered. |
| --> |
| <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en"> |
| <head> |
| <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /> |
| <meta name="viewport" content="width=device-width, initial-scale=1.0" /> |
| <title> |
| Log4j 2 Thread Context</title> |
| <link rel="stylesheet" href="../css/apache-maven-fluido.min.css" /> |
| <link rel="stylesheet" href="../css/site.css" /> |
| <link rel="stylesheet" href="../css/print.css" media="print" /> |
| |
| |
| <script type="text/javascript" src="../js/apache-maven-fluido.min.js"></script> |
| |
| |
| <meta name="author" content="Ralph Goers" /> |
| <meta name="Date-Revision-yyyymmdd" content="20120921" /> |
| <meta http-equiv="Content-Language" content="en" /> |
| </head> |
| <body class="topBarDisabled"> |
| |
| |
| |
| |
| <div class="container-fluid"> |
| <div id="banner"> |
| <div class="pull-left"> |
| <a href="../../../" id="bannerLeft"> |
| <img src="../images/ls-logo.jpg" alt="Apache Logging Services™"/> |
| </a> |
| </div> |
| <div class="pull-right"> <div id="bannerRight"> |
| <img src="../images/logo.jpg" /> |
| </div> |
| </div> |
| <div class="clear"><hr/></div> |
| </div> |
| |
| <div id="breadcrumbs"> |
| <ul class="breadcrumb"> |
| |
| |
| <li id="publishDate">Last Published: 2012-09-21</li> |
| <li class="divider">|</li> <li id="projectVersion">Version: 2.0-beta1</li> |
| |
| |
| |
| |
| |
| <li class="pull-right"> <a href="../../../" title="Logging Services">Logging Services</a> |
| </li> |
| |
| <li class="divider pull-right">|</li> |
| |
| <li class="pull-right"> <a href="http://www.apache.org/" class="externalLink" title="Apache">Apache</a> |
| </li> |
| |
| <li class="divider pull-right">|</li> |
| |
| <li class="pull-right"> <a href="http://wiki.apache.org/logging" class="externalLink" title="Logging Wiki">Logging Wiki</a> |
| </li> |
| |
| </ul> |
| </div> |
| |
| <div class="row-fluid"> |
| <div id="leftColumn" class="span2"> |
| <div class="well sidebar-nav"> |
| |
| |
| <h3>Apache Log4j™ 2</h3> |
| <ul> |
| <li class="none"> |
| <a href="../index.html" title="About">About</a> |
| </li> |
| <li class="none"> |
| <a href="../download.html" title="Download">Download</a> |
| </li> |
| <li class="none"> |
| <a href="../build.html" title="Build and Install">Build and Install</a> |
| </li> |
| <li class="none"> |
| <a href="../changelog.html" title="Changelog">Changelog</a> |
| </li> |
| </ul> |
| <h3>Manual</h3> |
| <ul> |
| <li class="none"> |
| <a href="../manual/index.html" title="Introduction">Introduction</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/architecture.html" title="Architecture">Architecture</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/migration.html" title="Log4j 1.x Migration">Log4j 1.x Migration</a> |
| </li> |
| <li class="expanded"> |
| <a href="../manual/api.html" title="API">API</a> |
| <ul> |
| <li class="none"> |
| <a href="../manual/api.html#Overview" title="Overview">Overview</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/flowtracing.html" title="Flow Tracing">Flow Tracing</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/markers.html" title="Markers">Markers</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/eventlogging.html" title="Event Logging">Event Logging</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/messages.html" title="Messages">Messages</a> |
| </li> |
| <li class="none"> |
| <strong>ThreadContext</strong> |
| </li> |
| </ul> |
| </li> |
| <li class="collapsed"> |
| <a href="../manual/configuration.html" title="Configuration">Configuration</a> |
| </li> |
| <li class="collapsed"> |
| <a href="../manual/plugins.html" title="Plugins">Plugins</a> |
| </li> |
| <li class="collapsed"> |
| <a href="../manual/lookups.html" title="Lookups">Lookups</a> |
| </li> |
| <li class="collapsed"> |
| <a href="../manual/appenders.html" title="Appenders">Appenders</a> |
| </li> |
| <li class="collapsed"> |
| <a href="../manual/layouts.html" title="Layouts">Layouts</a> |
| </li> |
| <li class="collapsed"> |
| <a href="../manual/filters.html" title="Filters">Filters</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/jmx.html" title="JMX">JMX</a> |
| </li> |
| <li class="none"> |
| <a href="../manual/logsep.html" title="Logging Separation">Logging Separation</a> |
| </li> |
| <li class="collapsed"> |
| <a href="../manual/extending.html" title="Extending Log4j">Extending Log4j</a> |
| </li> |
| </ul> |
| <h3>Components</h3> |
| <ul> |
| <li class="none"> |
| <a href="../log4j-api/index.html" title="API">API</a> |
| </li> |
| <li class="none"> |
| <a href="../log4j-core/index.html" title="Impl">Impl</a> |
| </li> |
| <li class="none"> |
| <a href="../log4j12-api/index.html" title="Log4J 1.2 API">Log4J 1.2 API</a> |
| </li> |
| <li class="none"> |
| <a href="../log4j-jcl/index.html" title="Commons Logging Bridge">Commons Logging Bridge</a> |
| </li> |
| <li class="none"> |
| <a href="../slf4j-impl/index.html" title="SLF4J Binding">SLF4J Binding</a> |
| </li> |
| <li class="none"> |
| <a href="../log4j-flume-ng/index.html" title="Apache Flume">Apache Flume</a> |
| </li> |
| <li class="none"> |
| <a href="../log4j-web/index.html" title="Log4j Web">Log4j Web</a> |
| </li> |
| </ul> |
| <h3>Site Documentation</h3> |
| <ul> |
| <li class="collapsed"> |
| <a href="../project-info.html" title="Project Information">Project Information</a> |
| </li> |
| <li class="collapsed"> |
| <a href="../project-reports.html" title="Project Reports">Project Reports</a> |
| </li> |
| </ul> |
| |
| |
| |
| <hr class="divider" /> |
| |
| <div id="poweredBy"> |
| <div class="clear"></div> |
| <div class="clear"></div> |
| <div class="clear"></div> |
| <a href="http://maven.apache.org/" title="Built by Maven" class="poweredBy"> |
| <img class="poweredBy" alt="Built by Maven" src="../images/logos/maven-feather.png" /> |
| </a> |
| </div> |
| </div> |
| </div> |
| |
| <div id="bodyColumn" class="span10" > |
| |
| <!-- Licensed to the Apache Software Foundation (ASF) under one or more |
| contributor license agreements. See the NOTICE file distributed with |
| this work for additional information regarding copyright ownership. |
| The ASF licenses this file to You under the Apache License, Version 2.0 |
| (the "License"); you may not use this file except in compliance with |
| the License. You may obtain a copy of the License at |
| |
| http://www.apache.org/licenses/LICENSE-2.0 |
| |
| Unless required by applicable law or agreed to in writing, software |
| distributed under the License is distributed on an "AS IS" BASIS, |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| See the License for the specific language governing permissions and |
| limitations under the License. --> |
| |
| <div class="section"><h2>Log4j 2 API<a name="Log4j_2_API"></a></h2> |
| <div class="section"><h3>Thread Context<a name="Thread_Context"></a></h3> |
| <div class="section"><h4>Introduction<a name="Introduction"></a></h4> |
| <p>Log4j introduced the concept of the Mapped Diagnostic Context or MDC. It has been documented and |
| discussed in numerous places including |
| <a class="externalLink" href="http://veerasundar.com/blog/2009/10/log4j-mdc-mapped-diagnostic-context-what-and-why/">Log4j MDC: What and Why</a> and |
| <a class="externalLink" href="http://blog.f12.no/wp/2004/12/09/log4j-and-the-mapped-diagnostic-context/">Log4j and the Mappend Diagnostic Context</a>. |
| In addition, Log4j 1.x provides support for a Nested Diagnostic Context or NDC. It too has been documented |
| and discussed in various places such as |
| <a class="externalLink" href="http://lstierneyltd.com/blog/development/log4j-nested-diagnostic-contexts-ndc/">Log4j NDC</a>. |
| SLF4J/Logback followed with its own implementation of the MDC, which is documented very well at |
| <a class="externalLink" href="http://logback.qos.ch/manual/mdc.html">Mapped Diagnostic Context</a>. |
| </p> |
| <p>Log4j 2 continues with the idea of the MDC and the NDC but merges them into a single Thread Context. |
| The Thread Context Map is the equivalent of the MDC and the Thread Context Stack is the equivalent of the |
| NDC. Although these are frequently used for purposes other than diagnosing problems, they are still |
| frequently referred to as the MDC and NDC in Log4j 2 since they are already well known by those acronyms. |
| </p> |
| </div><div class="section"><h4>Fish Tagging<a name="Fish_Tagging"></a></h4> |
| <p>Most real-world systems have to deal with multiple clients simultaneously. In a typical multithreaded |
| implementation of such asystem, different threads will handle different clients. Logging is |
| especially well suited to trace and debug complex distributed applications. A common approach to |
| differentiate the logging output of one client from another is to instantiate a new separate logger for |
| each client. This promotes the proliferation of loggers and increases the management overhead of logging. |
| </p> |
| <p>A lighter technique is to uniquely stamp each log request initiated from the same client interaction. |
| Neil Harrison described this method in the book "Patterns for Logging Diagnostic Messages," in <i>Pattern |
| Languages of Program Design 3</i>, edited by R. Martin, D. Riehle, and F. Buschmann |
| (Addison-Wesley, 1997). Just as a fish can be tagged and have its movement tracked, stamping log |
| events with a common tag or set of data elements allows the complete flow of a transaction or a request |
| to be tracked. We call this <i>Fish Tagging</i>. |
| </p> |
| <p>Log4j provides two mechanisms for performing Fish Tagging; the Thread Context Map and the Thread |
| Context Stack. The Thread Context Map allows any number of items to be added and be identified |
| using key/value pairs. The Thread Context Stack allows one or more items to be pushed on the |
| Stack and then be identified by their order in the Stack or by the data itself. Since key/value |
| pairs are more flexible, the Thread Context Map is recommended when data items may be added during |
| the processing of the request or when there are more than one or two items. |
| </p> |
| <p>To uniquely stamp each request using the Thread Context Stack, the user pushes contextual information |
| on to the Stack. |
| </p> |
| <div><pre> |
| ThreadContext.push(UUID.randomUUID().toString()); // Add the fishtag; |
| |
| logger.debug("Message 1"); |
| . |
| . |
| . |
| logger.debug("Message 2"); |
| . |
| . |
| ThreadContext.pop();</pre></div> |
| <p> |
| The alternative to the Thread Context Stack is the Thread Context Map. In this case, attributes |
| associated with the request being processed are adding at the beginning and removed at the end |
| as follows: |
| </p> |
| <div><pre> |
| ThreadContext.put("id", UUID.randomUUID().toString(); // Add the fishtag; |
| ThreadContext.put("ipAddress", request.getRemoteAddr()); |
| ThreadContext.put("loginId", session.getAttribute("loginId")); |
| ThreadContext.put("hostName", request.getServerName()); |
| . |
| logger.debug("Message 1"); |
| . |
| . |
| logger.debug("Message 2"); |
| . |
| . |
| ThreadContext.clear();</pre></div> |
| <p>The Stack and the Map are managed per thread and is based on |
| <a class="externalLink" href="http://docs.oracle.com/javase/6/docs/api/java/lang/InheritableThreadLocal.html">InheritableThreadLocal</a>. |
| Thus, in many cases the contents of the Stack and Map will be passed to child threads. However, as |
| discussed in the |
| <a class="externalLink" href="http://docs.oracle.com/javase/1.5.0/docs/api/java/util/concurrent/Executors.html#privilegedThreadFactory()">Executors</a> |
| class and in other cases where thread pooling is utilized, the ThreadContext may not always be |
| automatically passed to worker threads. In those cases the pooling mechanism should provide a means for |
| doing so. The getContext() and cloneStack() methods can be used to obtain copies of the Map and Stack |
| respectively. |
| </p> |
| <p> |
| Note that all methods of the |
| <a href="../log4j2-api/apidocs/org/apache/logging/log4j/ThreadContext.html">ThreadContext</a> |
| class are static. |
| </p> |
| </div><div class="section"><h4>Including the ThreadContext when writing logs<a name="Including_the_ThreadContext_when_writing_logs"></a></h4> |
| <p> |
| The PatternLayout provides mechanisms to print the contents of the ThreadContext Map and Stack. Using |
| %X by itself will cause the full contents of the Map to be included while %X{key} will cause the value |
| of the specified key to be included. %x will include the full contents of the Stack. |
| </p> |
| </div></div> |
| </div> |
| |
| |
| </div> |
| </div> |
| |
| <hr/> |
| |
| <footer> |
| <div class="container-fluid"> |
| <div class="row footer">Copyright © 1999-2012 <a href="http://www.apache.org">Apache Software Foundation</a>. All Rights Reserved.</br /> |
| Apache Logging, Apache Log4j, Log4j, Apache, the Apache feather logo, and the Apache Logging project logo are trademarks of The Apache Software Foundation.</div> |
| </div> |
| </footer> |
| </body> |
| </html> |