LoggerNDC.php 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. <?php
  2. /**
  3. * log4php is a PHP port of the log4j java logging package.
  4. *
  5. * <p>This framework is based on log4j (see {@link http://jakarta.apache.org/log4j log4j} for details).</p>
  6. * <p>Design, strategies and part of the methods documentation are developed by log4j team
  7. * (Ceki Gülcü as log4j project founder and
  8. * {@link http://jakarta.apache.org/log4j/docs/contributors.html contributors}).</p>
  9. *
  10. * <p>PHP port, extensions and modifications by VxR. All rights reserved.<br>
  11. * For more information, please see {@link http://www.vxr.it/log4php/}.</p>
  12. *
  13. * <p>This software is published under the terms of the LGPL License
  14. * a copy of which has been included with this distribution in the LICENSE file.</p>
  15. *
  16. * @package log4php
  17. */
  18. /**
  19. * @ignore
  20. */
  21. if (!defined('LOG4PHP_DIR')) define('LOG4PHP_DIR', dirname(__FILE__));
  22. /**
  23. */
  24. require_once(LOG4PHP_DIR . '/LoggerLog.php');
  25. define('LOGGER_NDC_HT_SIZE', 7);
  26. /**
  27. * This is the global repository of NDC stack
  28. */
  29. $GLOBALS['log4php.LoggerNDC.ht'] = array();
  30. /**
  31. * This is the max depth of NDC stack
  32. */
  33. $GLOBALS['log4php.LoggerNDC.maxDepth'] = LOGGER_NDC_HT_SIZE;
  34. /**
  35. * The NDC class implements <i>nested diagnostic contexts</i> as
  36. * defined by Neil Harrison in the article "Patterns for Logging
  37. * Diagnostic Messages" part of the book "<i>Pattern Languages of
  38. * Program Design 3</i>" edited by Martin et al.
  39. *
  40. * <p>A Nested Diagnostic Context, or NDC in short, is an instrument
  41. * to distinguish interleaved log output from different sources. Log
  42. * output is typically interleaved when a server handles multiple
  43. * clients near-simultaneously.
  44. *
  45. * <p>Interleaved log output can still be meaningful if each log entry
  46. * from different contexts had a distinctive stamp. This is where NDCs
  47. * come into play.
  48. *
  49. * <p><i><b>Note that NDCs are managed on a per thread
  50. * basis</b></i>. NDC operations such as {@link push()}, {@link pop()},
  51. * {@link clear()}, {@link getDepth()} and {@link setMaxDepth()}
  52. * affect the NDC of the <i>current</i> thread only. NDCs of other
  53. * threads remain unaffected.
  54. *
  55. * <p>For example, a servlet can build a per client request NDC
  56. * consisting the clients host name and other information contained in
  57. * the the request. <i>Cookies</i> are another source of distinctive
  58. * information. To build an NDC one uses the {@link push()}
  59. * operation.</p>
  60. *
  61. * Simply put,
  62. *
  63. * - Contexts can be nested.
  64. * - When entering a context, call
  65. * <code>LoggerNDC::push()</code>
  66. * As a side effect, if there is no nested diagnostic context for the
  67. * current thread, this method will create it.
  68. * - When leaving a context, call
  69. * <code>LoggerNDC::pop()</code>
  70. * - <b>When exiting a thread make sure to call {@link remove()}</b>
  71. *
  72. * <p>There is no penalty for forgetting to match each
  73. * <code>push</code> operation with a corresponding <code>pop</code>,
  74. * except the obvious mismatch between the real application context
  75. * and the context set in the NDC.</p>
  76. *
  77. * <p>If configured to do so, {@link LoggerPatternLayout} and {@link LoggerLayoutTTCC}
  78. * instances automatically retrieve the nested diagnostic
  79. * context for the current thread without any user intervention.
  80. * Hence, even if a servlet is serving multiple clients
  81. * simultaneously, the logs emanating from the same code (belonging to
  82. * the same category) can still be distinguished because each client
  83. * request will have a different NDC tag.</p>
  84. *
  85. *
  86. * @author VxR <vxr@vxr.it>
  87. * @version $Revision: 1.1 $
  88. * @package log4php
  89. * @since 0.3
  90. */
  91. class LoggerNDC {
  92. /**
  93. * Clear any nested diagnostic information if any. This method is
  94. * useful in cases where the same thread can be potentially used
  95. * over and over in different unrelated contexts.
  96. *
  97. * <p>This method is equivalent to calling the {@link setMaxDepth()}
  98. * method with a zero <var>maxDepth</var> argument.
  99. *
  100. * @static
  101. */
  102. function clear()
  103. {
  104. LoggerLog::debug("LoggerNDC::clear()");
  105. $GLOBALS['log4php.LoggerNDC.ht'] = array();
  106. }
  107. /**
  108. * Never use this method directly, use the {@link LoggerLoggingEvent::getNDC()} method instead.
  109. * @static
  110. * @return array
  111. */
  112. function get()
  113. {
  114. LoggerLog::debug("LoggerNDC::get()");
  115. return $GLOBALS['log4php.LoggerNDC.ht'];
  116. }
  117. /**
  118. * Get the current nesting depth of this diagnostic context.
  119. *
  120. * @see setMaxDepth()
  121. * @return integer
  122. * @static
  123. */
  124. function getDepth()
  125. {
  126. LoggerLog::debug("LoggerNDC::getDepth()");
  127. return sizeof($GLOBALS['log4php.LoggerNDC.ht']);
  128. }
  129. /**
  130. * Clients should call this method before leaving a diagnostic
  131. * context.
  132. *
  133. * <p>The returned value is the value that was pushed last. If no
  134. * context is available, then the empty string "" is returned.</p>
  135. *
  136. * @return string The innermost diagnostic context.
  137. * @static
  138. */
  139. function pop()
  140. {
  141. LoggerLog::debug("LoggerNDC::pop()");
  142. if (sizeof($GLOBALS['log4php.LoggerNDC.ht']) > 0) {
  143. return array_pop($GLOBALS['log4php.LoggerNDC.ht']);
  144. } else {
  145. return '';
  146. }
  147. }
  148. /**
  149. * Looks at the last diagnostic context at the top of this NDC
  150. * without removing it.
  151. *
  152. * <p>The returned value is the value that was pushed last. If no
  153. * context is available, then the empty string "" is returned.</p>
  154. * @return string The innermost diagnostic context.
  155. * @static
  156. */
  157. function peek()
  158. {
  159. LoggerLog::debug("LoggerNDC::peek()");
  160. if (sizeof($GLOBALS['log4php.LoggerNDC.ht']) > 0) {
  161. return end($GLOBALS['log4php.LoggerNDC.ht']);
  162. } else {
  163. return '';
  164. }
  165. }
  166. /**
  167. * Push new diagnostic context information for the current thread.
  168. *
  169. * <p>The contents of the <var>message</var> parameter is
  170. * determined solely by the client.
  171. *
  172. * @param string $message The new diagnostic context information.
  173. * @static
  174. */
  175. function push($message)
  176. {
  177. LoggerLog::debug("LoggerNDC::push()");
  178. array_push($GLOBALS['log4php.LoggerNDC.ht'], (string)$message);
  179. }
  180. /**
  181. * Remove the diagnostic context for this thread.
  182. * @static
  183. */
  184. function remove()
  185. {
  186. LoggerLog::debug("LoggerNDC::remove()");
  187. LoggerNDC::clear();
  188. }
  189. /**
  190. * Set maximum depth of this diagnostic context. If the current
  191. * depth is smaller or equal to <var>maxDepth</var>, then no
  192. * action is taken.
  193. *
  194. * <p>This method is a convenient alternative to multiple
  195. * {@link pop()} calls. Moreover, it is often the case that at
  196. * the end of complex call sequences, the depth of the NDC is
  197. * unpredictable. The {@link setMaxDepth()} method circumvents
  198. * this problem.
  199. *
  200. * @param integer $maxDepth
  201. * @see getDepth()
  202. * @static
  203. */
  204. function setMaxDepth($maxDepth)
  205. {
  206. LoggerLog::debug("LoggerNDC::setMaxDepth() maxDepth='$maxDepth'");
  207. $maxDepth = (int)$maxDepth;
  208. if ($maxDepth <= LOGGER_NDC_HT_SIZE) {
  209. if (LoggerNDC::getDepth() > $maxDepth) {
  210. $GLOBALS['log4php.LoggerNDC.ht'] = array_slice($GLOBALS['log4php.LoggerNDC.ht'], $maxDepth);
  211. }
  212. $GLOBALS['log4php.LoggerNDC.maxDepth'] = $maxDepth;
  213. }
  214. }
  215. }
  216. ?>