LoggerPropertyConfigurator.php 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650
  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. require_once(LOG4PHP_DIR . '/config/LoggerPropertySetter.php');
  23. require_once(LOG4PHP_DIR . '/helpers/LoggerOptionConverter.php');
  24. require_once(LOG4PHP_DIR . '/or/LoggerObjectRenderer.php');
  25. require_once(LOG4PHP_DIR . '/or/LoggerRendererMap.php');
  26. require_once(LOG4PHP_DIR . '/spi/LoggerConfigurator.php');
  27. require_once(LOG4PHP_DIR . '/spi/LoggerFilter.php');
  28. require_once(LOG4PHP_DIR . '/LoggerAppender.php');
  29. require_once(LOG4PHP_DIR . '/LoggerDefaultCategoryFactory.php');
  30. require_once(LOG4PHP_DIR . '/LoggerLayout.php');
  31. require_once(LOG4PHP_DIR . '/LoggerLevel.php');
  32. require_once(LOG4PHP_DIR . '/LoggerManager.php');
  33. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_CATEGORY_PREFIX', "log4php.category.");
  34. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_PREFIX', "log4php.logger.");
  35. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_FACTORY_PREFIX', "log4php.factory");
  36. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ADDITIVITY_PREFIX', "log4php.additivity.");
  37. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ROOT_CATEGORY_PREFIX', "log4php.rootCategory");
  38. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ROOT_LOGGER_PREFIX', "log4php.rootLogger");
  39. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_APPENDER_PREFIX', "log4php.appender.");
  40. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_RENDERER_PREFIX', "log4php.renderer.");
  41. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_THRESHOLD_PREFIX', "log4php.threshold");
  42. /**
  43. * Key for specifying the {@link LoggerFactory}.
  44. */
  45. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_FACTORY_KEY', "log4php.loggerFactory");
  46. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_DEBUG_KEY', "log4php.debug");
  47. define('LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_INTERNAL_ROOT_NAME', "root");
  48. /**
  49. * Allows the configuration of log4php from an external file.
  50. *
  51. * See {@link doConfigure()} for the expected format.
  52. *
  53. * <p>It is sometimes useful to see how log4php is reading configuration
  54. * files. You can enable log4php internal logging by defining the
  55. * <b>log4php.debug</b> variable.</p>
  56. *
  57. * <p>The <i>LoggerPropertyConfigurator</i> does not handle the
  58. * advanced configuration features supported by the {@link LoggerDOMConfigurator}
  59. * such as support for {@link LoggerFilter},
  60. custom {@link LoggerErrorHandlers}, nested appenders such as the
  61. {@link Logger AsyncAppender},
  62. * etc.
  63. *
  64. * <p>All option <i>values</i> admit variable substitution. The
  65. * syntax of variable substitution is similar to that of Unix
  66. * shells. The string between an opening <b>&quot;${&quot;</b> and
  67. * closing <b>&quot;}&quot;</b> is interpreted as a key. The value of
  68. * the substituted variable can be defined as a system property or in
  69. * the configuration file itself. The value of the key is first
  70. * searched in the defined constants, in the enviroments variables
  71. * and if not found there, it is
  72. * then searched in the configuration file being parsed. The
  73. * corresponding value replaces the ${variableName} sequence.</p>
  74. * <p>For example, if <b>$_ENV['home']</b> env var is set to
  75. * <b>/home/xyz</b>, then every occurrence of the sequence
  76. * <b>${home}</b> will be interpreted as
  77. * <b>/home/xyz</b>. See {@link LoggerOptionConverter::getSystemProperty()}
  78. * for details.</p>
  79. *
  80. * @author VxR <vxr@vxr.it>
  81. * @version $Revision: 1.1 $
  82. * @package log4php
  83. * @since 0.5
  84. */
  85. class LoggerPropertyConfigurator extends LoggerConfigurator {
  86. /**
  87. * @var LoggerFactory
  88. */
  89. var $loggerFactory = null;
  90. /**
  91. * Constructor
  92. */
  93. function LoggerPropertyConfigurator()
  94. {
  95. $this->loggerFactory = new LoggerDefaultCategoryFactory();
  96. }
  97. /**
  98. * Configure the default repository using the resource pointed by <b>url</b>.
  99. * <b>Url</b> is any valid resurce as defined in {@link PHP_MANUAL#file} function.
  100. * Note that the resource will be search with <i>use_include_path</i> parameter
  101. * set to "1".
  102. *
  103. * @param string $url
  104. * @return boolean configuration result
  105. * @static
  106. */
  107. function configure($url = '')
  108. {
  109. $configurator = new LoggerPropertyConfigurator();
  110. $repository =& LoggerManager::getLoggerRepository();
  111. return $configurator->doConfigure($url, $repository);
  112. }
  113. /**
  114. * Read configuration from a file.
  115. *
  116. * <p>The function {@link PHP_MANUAL#parse_ini_file} is used to read the
  117. * file.</p>
  118. *
  119. * <b>The existing configuration is not cleared nor reset.</b>
  120. * If you require a different behavior, then call
  121. * {@link LoggerManager::resetConfiguration()}
  122. * method before calling {@link doConfigure()}.
  123. *
  124. * <p>The configuration file consists of statements in the format
  125. * <b>key=value</b>. The syntax of different configuration
  126. * elements are discussed below.
  127. *
  128. * <p><b>Repository-wide threshold</b></p>
  129. *
  130. * <p>The repository-wide threshold filters logging requests by level
  131. * regardless of logger. The syntax is:
  132. *
  133. * <pre>
  134. * log4php.threshold=[level]
  135. * </pre>
  136. *
  137. * <p>The level value can consist of the string values OFF, FATAL,
  138. * ERROR, WARN, INFO, DEBUG, ALL or a <i>custom level</i> value. A
  139. * custom level value can be specified in the form
  140. * <samp>level#classname</samp>. By default the repository-wide threshold is set
  141. * to the lowest possible value, namely the level <b>ALL</b>.
  142. * </p>
  143. *
  144. *
  145. * <p><b>Appender configuration</b></p>
  146. *
  147. * <p>Appender configuration syntax is:</p>
  148. * <pre>
  149. * ; For appender named <i>appenderName</i>, set its class.
  150. * ; Note: The appender name can contain dots.
  151. * log4php.appender.appenderName=name_of_appender_class
  152. *
  153. * ; Set appender specific options.
  154. *
  155. * log4php.appender.appenderName.option1=value1
  156. * log4php.appender.appenderName.optionN=valueN
  157. * </pre>
  158. *
  159. * For each named appender you can configure its {@link LoggerLayout}. The
  160. * syntax for configuring an appender's layout is:
  161. * <pre>
  162. * log4php.appender.appenderName.layout=name_of_layout_class
  163. * log4php.appender.appenderName.layout.option1=value1
  164. * ....
  165. * log4php.appender.appenderName.layout.optionN=valueN
  166. * </pre>
  167. *
  168. * <p><b>Configuring loggers</b></p>
  169. *
  170. * <p>The syntax for configuring the root logger is:
  171. * <pre>
  172. * log4php.rootLogger=[level], appenderName, appenderName, ...
  173. * </pre>
  174. *
  175. * <p>This syntax means that an optional <i>level</i> can be
  176. * supplied followed by appender names separated by commas.
  177. *
  178. * <p>The level value can consist of the string values OFF, FATAL,
  179. * ERROR, WARN, INFO, DEBUG, ALL or a <i>custom level</i> value. A
  180. * custom level value can be specified in the form</p>
  181. *
  182. * <pre>level#classname</pre>
  183. *
  184. * <p>If a level value is specified, then the root level is set
  185. * to the corresponding level. If no level value is specified,
  186. * then the root level remains untouched.
  187. *
  188. * <p>The root logger can be assigned multiple appenders.
  189. *
  190. * <p>Each <i>appenderName</i> (separated by commas) will be added to
  191. * the root logger. The named appender is defined using the
  192. * appender syntax defined above.
  193. *
  194. * <p>For non-root categories the syntax is almost the same:
  195. * <pre>
  196. * log4php.logger.logger_name=[level|INHERITED|NULL], appenderName, appenderName, ...
  197. * </pre>
  198. *
  199. * <p>The meaning of the optional level value is discussed above
  200. * in relation to the root logger. In addition however, the value
  201. * INHERITED can be specified meaning that the named logger should
  202. * inherit its level from the logger hierarchy.</p>
  203. *
  204. * <p>If no level value is supplied, then the level of the
  205. * named logger remains untouched.</p>
  206. *
  207. * <p>By default categories inherit their level from the
  208. * hierarchy. However, if you set the level of a logger and later
  209. * decide that that logger should inherit its level, then you should
  210. * specify INHERITED as the value for the level value. NULL is a
  211. * synonym for INHERITED.</p>
  212. *
  213. * <p>Similar to the root logger syntax, each <i>appenderName</i>
  214. * (separated by commas) will be attached to the named logger.</p>
  215. *
  216. * <p>See the <i>appender additivity rule</i> in the user manual for
  217. * the meaning of the <b>additivity</b> flag.
  218. *
  219. * <p><b>ObjectRenderers</b></p>
  220. *
  221. * <p>You can customize the way message objects of a given type are
  222. * converted to String before being logged. This is done by
  223. * specifying a {@link LoggerObjectRenderer}
  224. * for the object type would like to customize.</p>
  225. *
  226. * <p>The syntax is:
  227. *
  228. * <pre>
  229. * log4php.renderer.name_of_rendered_class=name_of_rendering.class
  230. * </pre>
  231. *
  232. * As in,
  233. * <pre>
  234. * log4php.renderer.myFruit=myFruitRenderer
  235. * </pre>
  236. *
  237. * <p><b>Logger Factories</b></p>
  238. *
  239. * The usage of custom logger factories is discouraged and no longer
  240. * documented.
  241. *
  242. * <p><b>Example</b></p>
  243. *
  244. * <p>An example configuration is given below. Other configuration
  245. * file examples are given in the <b>tests</b> folder.
  246. *
  247. * <pre>
  248. * ; Set options for appender named "A1".
  249. * ; Appender "A1" will be a SyslogAppender
  250. * log4php.appender.A1=SyslogAppender
  251. *
  252. * ; The syslog daemon resides on www.abc.net
  253. * log4php.appender.A1.SyslogHost=www.abc.net
  254. *
  255. * ; A1's layout is a LoggerPatternLayout, using the conversion pattern
  256. * ; <b>%r %-5p %c{2} %M.%L %x - %m%n</b>. Thus, the log output will
  257. * ; include the relative time since the start of the application in
  258. * ; milliseconds, followed by the level of the log request,
  259. * ; followed by the two rightmost components of the logger name,
  260. * ; followed by the callers method name, followed by the line number,
  261. * ; the nested disgnostic context and finally the message itself.
  262. * ; Refer to the documentation of LoggerPatternLayout} for further information
  263. * ; on the syntax of the ConversionPattern key.
  264. * log4php.appender.A1.layout=LoggerPatternLayout
  265. * log4php.appender.A1.layout.ConversionPattern="%-4r %-5p %c{2} %M.%L %x - %m%n"
  266. *
  267. * ; Set options for appender named "A2"
  268. * ; A2 should be a LoggerAppenderRollingFile, with maximum file size of 10 MB
  269. * ; using at most one backup file. A2's layout is TTCC, using the
  270. * ; ISO8061 date format with context printing enabled.
  271. * log4php.appender.A2=LoggerAppenderRollingFile
  272. * log4php.appender.A2.MaxFileSize=10MB
  273. * log4php.appender.A2.MaxBackupIndex=1
  274. * log4php.appender.A2.layout=LoggerLayoutTTCC
  275. * log4php.appender.A2.layout.ContextPrinting="true"
  276. * log4php.appender.A2.layout.DateFormat="%c"
  277. *
  278. * ; Root logger set to DEBUG using the A2 appender defined above.
  279. * log4php.rootLogger=DEBUG, A2
  280. *
  281. * ; Logger definitions:
  282. * ; The SECURITY logger inherits is level from root. However, it's output
  283. * ; will go to A1 appender defined above. It's additivity is non-cumulative.
  284. * log4php.logger.SECURITY=INHERIT, A1
  285. * log4php.additivity.SECURITY=false
  286. *
  287. * ; Only warnings or above will be logged for the logger "SECURITY.access".
  288. * ; Output will go to A1.
  289. * log4php.logger.SECURITY.access=WARN
  290. *
  291. *
  292. * ; The logger "class.of.the.day" inherits its level from the
  293. * ; logger hierarchy. Output will go to the appender's of the root
  294. * ; logger, A2 in this case.
  295. * log4php.logger.class.of.the.day=INHERIT
  296. * </pre>
  297. *
  298. * <p>Refer to the <b>setOption</b> method in each Appender and
  299. * Layout for class specific options.</p>
  300. *
  301. * <p>Use the <b>&quot;;&quot;</b> character at the
  302. * beginning of a line for comments.</p>
  303. *
  304. * @param string $url The name of the configuration file where the
  305. * configuration information is stored.
  306. * @param LoggerHierarchy &$repository the repository to apply the configuration
  307. */
  308. function doConfigure($url, &$repository)
  309. {
  310. $properties = @parse_ini_file($url);
  311. if ($properties === false) {
  312. LoggerLog::warn("LoggerPropertyConfigurator::doConfigure() cannot load '$url' configuration.");
  313. return false;
  314. }
  315. return $this->doConfigureProperties($properties, $repository);
  316. }
  317. /**
  318. * Read configuration options from <b>properties</b>.
  319. *
  320. * @see doConfigure().
  321. * @param array $properties
  322. * @param LoggerHierarchy &$hierarchy
  323. */
  324. function doConfigureProperties($properties, &$hierarchy)
  325. {
  326. $value = @$properties[LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_DEBUG_KEY];
  327. if (!empty($value)) {
  328. LoggerLog::internalDebugging(LoggerOptionConverter::toBoolean($value, LoggerLog::internalDebugging()));
  329. }
  330. $thresholdStr = @$properties[LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_THRESHOLD_PREFIX];
  331. $hierarchy->setThreshold(LoggerOptionConverter::toLevel($thresholdStr, LoggerLevel::getLevelAll()));
  332. $this->configureRootCategory($properties, $hierarchy);
  333. $this->configureLoggerFactory($properties);
  334. $this->parseCatsAndRenderers($properties, $hierarchy);
  335. LoggerLog::debug("LoggerPropertyConfigurator::doConfigureProperties() Finished configuring.");
  336. return true;
  337. }
  338. // --------------------------------------------------------------------------
  339. // Internal stuff
  340. // --------------------------------------------------------------------------
  341. /**
  342. * Check the provided <b>Properties</b> object for a
  343. * {@link LoggerFactory} entry specified by
  344. * {@link LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_FACTORY_KEY}.
  345. *
  346. * If such an entry exists, an attempt is made to create an instance using
  347. * the default constructor.
  348. * This instance is used for subsequent Category creations
  349. * within this configurator.
  350. *
  351. * @see parseCatsAndRenderers()
  352. * @param array $props array of properties
  353. */
  354. function configureLoggerFactory($props)
  355. {
  356. $factoryFqcn = @$props[LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_FACTORY_KEY];
  357. if(!empty($factoryFqcn)) {
  358. $factoryClassName = basename($factoryFqcn);
  359. LoggerLog::debug(
  360. "LoggerPropertyConfigurator::configureLoggerFactory() Trying to load factory [" .
  361. $factoryClassName .
  362. "]."
  363. );
  364. if (!class_exists($factoryClassName))
  365. @include_once("{$factoryFqcn}.php");
  366. if (class_exists($factoryClassName)) {
  367. $loggerFactory = new $factoryClassName();
  368. } else {
  369. LoggerLog::debug(
  370. "LoggerPropertyConfigurator::configureLoggerFactory() Unable to load factory [" .
  371. $factoryClassName .
  372. "]. Using default."
  373. );
  374. $loggerFactory = $this->loggerFactory;
  375. }
  376. LoggerLog::debug(
  377. "LoggerPropertyConfigurator::configureLoggerFactory() ".
  378. "Setting properties for category factory [" . get_class($loggerFactory) . "]."
  379. );
  380. LoggerPropertySetter::setPropertiesByObject($loggerFactory, $props, LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_FACTORY_PREFIX . ".");
  381. }
  382. }
  383. /**
  384. * @param array $props array of properties
  385. * @param LoggerHierarchy &$hierarchy
  386. */
  387. function configureRootCategory($props, &$hierarchy)
  388. {
  389. $effectivePrefix = LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ROOT_LOGGER_PREFIX;
  390. $value = @$props[LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ROOT_LOGGER_PREFIX];
  391. if(empty($value)) {
  392. $value = @$props[LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ROOT_CATEGORY_PREFIX];
  393. $effectivePrefix = LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ROOT_CATEGORY_PREFIX;
  394. }
  395. if (empty($value)) {
  396. LoggerLog::debug(
  397. "LoggerPropertyConfigurator::configureRootCategory() ".
  398. "Could not find root logger information. Is this OK?"
  399. );
  400. } else {
  401. $root =& $hierarchy->getRootLogger();
  402. // synchronized(root) {
  403. $this->parseCategory(
  404. $props,
  405. $root,
  406. $effectivePrefix,
  407. LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_INTERNAL_ROOT_NAME,
  408. $value
  409. );
  410. // }
  411. }
  412. }
  413. /**
  414. * Parse non-root elements, such non-root categories and renderers.
  415. *
  416. * @param array $props array of properties
  417. * @param LoggerHierarchy &$hierarchy
  418. */
  419. function parseCatsAndRenderers($props, &$hierarchy)
  420. {
  421. while(list($key,$value) = each($props)) {
  422. if( strpos($key, LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_CATEGORY_PREFIX) === 0 or
  423. strpos($key, LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_PREFIX) === 0) {
  424. if(strpos($key, LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_CATEGORY_PREFIX) === 0) {
  425. $loggerName = substr($key, strlen(LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_CATEGORY_PREFIX));
  426. } elseif (strpos($key, LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_PREFIX) === 0) {
  427. $loggerName = substr($key, strlen(LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_LOGGER_PREFIX));
  428. }
  429. $logger =& $hierarchy->getLogger($loggerName, $this->loggerFactory);
  430. // synchronized(logger) {
  431. $this->parseCategory($props, $logger, $key, $loggerName, $value);
  432. $this->parseAdditivityForLogger($props, $logger, $loggerName);
  433. // }
  434. } elseif (strpos($key, LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_RENDERER_PREFIX) === 0) {
  435. $renderedClass = substr($key, strlen(LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_RENDERER_PREFIX));
  436. $renderingClass = $value;
  437. if (method_exists($hierarchy, 'addrenderer')) {
  438. LoggerRendererMap::addRenderer($hierarchy, $renderedClass, $renderingClass);
  439. }
  440. }
  441. }
  442. }
  443. /**
  444. * Parse the additivity option for a non-root category.
  445. *
  446. * @param array $props array of properties
  447. * @param Logger &$cat
  448. * @param string $loggerName
  449. */
  450. function parseAdditivityForLogger($props, &$cat, $loggerName)
  451. {
  452. $value = LoggerOptionConverter::findAndSubst(
  453. LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ADDITIVITY_PREFIX . $loggerName,
  454. $props
  455. );
  456. LoggerLog::debug(
  457. "LoggerPropertyConfigurator::parseAdditivityForLogger() ".
  458. "Handling " . LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_ADDITIVITY_PREFIX . $loggerName . "=[{$value}]"
  459. );
  460. // touch additivity only if necessary
  461. if(!empty($value)) {
  462. $additivity = LoggerOptionConverter::toBoolean($value, true);
  463. LoggerLog::debug(
  464. "LoggerPropertyConfigurator::parseAdditivityForLogger() ".
  465. "Setting additivity for [{$loggerName}] to [{$additivity}]"
  466. );
  467. $cat->setAdditivity($additivity);
  468. }
  469. }
  470. /**
  471. * This method must work for the root category as well.
  472. *
  473. * @param array $props array of properties
  474. * @param Logger &$logger
  475. * @param string $optionKey
  476. * @param string $loggerName
  477. * @param string $value
  478. * @return Logger
  479. */
  480. function &parseCategory($props, &$logger, $optionKey, $loggerName, $value)
  481. {
  482. LoggerLog::debug(
  483. "LoggerPropertyConfigurator::parseCategory() ".
  484. "Parsing for [{$loggerName}] with value=[{$value}]."
  485. );
  486. // We must skip over ',' but not white space
  487. $st = explode(',', $value);
  488. // If value is not in the form ", appender.." or "", then we should set
  489. // the level of the loggeregory.
  490. if(!(@$value{0} == ',' || empty($value))) {
  491. // just to be on the safe side...
  492. if(sizeof($st) == 0)
  493. return;
  494. $levelStr = current($st);
  495. LoggerLog::debug(
  496. "LoggerPropertyConfigurator::parseCategory() ".
  497. "Level token is [$levelStr]."
  498. );
  499. // If the level value is inherited, set category level value to
  500. // null. We also check that the user has not specified inherited for the
  501. // root category.
  502. if('INHERITED' == strtoupper($levelStr) || 'NULL' == strtoupper($levelStr)) {
  503. if ($loggerName == LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_INTERNAL_ROOT_NAME) {
  504. LoggerLog::warn(
  505. "LoggerPropertyConfigurator::parseCategory() ".
  506. "The root logger cannot be set to null."
  507. );
  508. } else {
  509. $logger->setLevel(null);
  510. }
  511. } else {
  512. $logger->setLevel(LoggerOptionConverter::toLevel($levelStr, LoggerLevel::getLevelDebug()));
  513. }
  514. }
  515. // Begin by removing all existing appenders.
  516. $logger->removeAllAppenders();
  517. while($appenderName = next($st)) {
  518. $appenderName = trim($appenderName);
  519. if(empty($appenderName))
  520. continue;
  521. LoggerLog::debug(
  522. "LoggerPropertyConfigurator::parseCategory() ".
  523. "Parsing appender named [{$appenderName}]."
  524. );
  525. $appender =& $this->parseAppender($props, $appenderName);
  526. if($appender !== null) {
  527. $logger->addAppender($appender);
  528. }
  529. }
  530. }
  531. /**
  532. * @param array $props array of properties
  533. * @param string $appenderName
  534. * @return LoggerAppender
  535. */
  536. function &parseAppender($props, $appenderName)
  537. {
  538. $appender =& LoggerAppender::singleton($appenderName);
  539. if($appender !== null) {
  540. LoggerLog::debug(
  541. "LoggerPropertyConfigurator::parseAppender() ".
  542. "Appender [{$appenderName}] was already parsed."
  543. );
  544. return $appender;
  545. }
  546. // Appender was not previously initialized.
  547. $prefix = LOG4PHP_LOGGER_PROPERTY_CONFIGURATOR_APPENDER_PREFIX . $appenderName;
  548. $layoutPrefix = $prefix . ".layout";
  549. $appenderClass = @$props[$prefix];
  550. if (!empty($appenderClass)) {
  551. $appender =& LoggerAppender::singleton($appenderName, $appenderClass);
  552. if($appender === null) {
  553. LoggerLog::warn(
  554. "LoggerPropertyConfigurator::parseAppender() ".
  555. "Could not instantiate appender named [$appenderName]."
  556. );
  557. return null;
  558. }
  559. } else {
  560. LoggerLog::warn(
  561. "LoggerPropertyConfigurator::parseAppender() ".
  562. "Could not instantiate appender named [$appenderName] with null className."
  563. );
  564. return null;
  565. }
  566. $appender->setName($appenderName);
  567. if( $appender->requiresLayout() ) {
  568. LoggerLog::debug(
  569. "LoggerPropertyConfigurator::parseAppender() ".
  570. "Parsing layout section for [$appenderName]."
  571. );
  572. $layoutClass = @$props[$layoutPrefix];
  573. $layoutClass = LoggerOptionConverter::substVars($layoutClass, $props);
  574. if (empty($layoutClass)) {
  575. LoggerLog::warn(
  576. "LoggerPropertyConfigurator::parseAppender() ".
  577. "layout class is empty in '$layoutPrefix'. Using Simple layout"
  578. );
  579. $layout = LoggerLayout::factory('LoggerLayoutSimple');
  580. } else {
  581. $layout = LoggerLayout::factory($layoutClass);
  582. if($layout === null) {
  583. LoggerLog::warn(
  584. "LoggerPropertyConfigurator::parseAppender() ".
  585. "cannot create layout '$layoutClass'. Using Simple layout"
  586. );
  587. $layout = LoggerLayout::factory('LoggerLayoutSimple');
  588. }
  589. }
  590. LoggerLog::debug(
  591. "LoggerPropertyConfigurator::parseAppender() ".
  592. "Parsing layout options for [$appenderName]."
  593. );
  594. LoggerPropertySetter::setPropertiesByObject($layout, $props, $layoutPrefix . ".");
  595. LoggerLog::debug(
  596. "LoggerPropertyConfigurator::parseAppender() ".
  597. "End Parsing layout options for [$appenderName]."
  598. );
  599. $appender->setLayout($layout);
  600. }
  601. LoggerPropertySetter::setPropertiesByObject($appender, $props, $prefix . ".");
  602. LoggerLog::debug(
  603. "LoggerPropertyConfigurator::parseAppender() ".
  604. "Parsed [{$appenderName}] options."
  605. );
  606. return $appender;
  607. }
  608. }
  609. ?>