2014-02-26 10:47:02 +01:00
|
|
|
<?php
|
2015-02-04 10:46:36 +01:00
|
|
|
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
2014-02-26 10:47:02 +01:00
|
|
|
|
2014-10-31 10:27:17 +01:00
|
|
|
namespace Icinga\Application;
|
2014-02-26 10:47:02 +01:00
|
|
|
|
2014-04-22 14:31:57 +02:00
|
|
|
use Exception;
|
2014-11-18 13:11:52 +01:00
|
|
|
use Icinga\Data\ConfigObject;
|
2014-10-31 10:27:17 +01:00
|
|
|
use Icinga\Application\Logger\Writer\FileWriter;
|
|
|
|
use Icinga\Application\Logger\Writer\SyslogWriter;
|
2014-02-26 10:47:02 +01:00
|
|
|
use Icinga\Exception\ConfigurationError;
|
2015-07-24 16:19:20 +02:00
|
|
|
use Icinga\Exception\IcingaException;
|
2014-02-26 10:47:02 +01:00
|
|
|
|
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* Logger
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
|
|
|
class Logger
|
|
|
|
{
|
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* Debug message
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-10-16 15:54:13 +02:00
|
|
|
const DEBUG = 1;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Informational message
|
|
|
|
*/
|
|
|
|
const INFO = 2;
|
2014-02-26 10:47:02 +01:00
|
|
|
|
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* Warning message
|
|
|
|
*/
|
|
|
|
const WARNING = 4;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Error message
|
|
|
|
*/
|
|
|
|
const ERROR = 8;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Log levels
|
2014-02-26 10:47:02 +01:00
|
|
|
*
|
2014-10-16 15:54:13 +02:00
|
|
|
* @var array
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-10-16 15:54:13 +02:00
|
|
|
public static $levels = array(
|
|
|
|
Logger::DEBUG => 'DEBUG',
|
|
|
|
Logger::INFO => 'INFO',
|
|
|
|
Logger::WARNING => 'WARNING',
|
|
|
|
Logger::ERROR => 'ERROR'
|
|
|
|
);
|
2014-02-26 10:47:02 +01:00
|
|
|
|
2014-09-05 15:28:39 +02:00
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* This logger's instance
|
2014-09-05 15:28:39 +02:00
|
|
|
*
|
2014-10-16 15:54:13 +02:00
|
|
|
* @var static
|
2014-09-05 15:28:39 +02:00
|
|
|
*/
|
2014-10-16 15:54:13 +02:00
|
|
|
protected static $instance;
|
2014-09-05 15:28:39 +02:00
|
|
|
|
2014-02-26 10:47:02 +01:00
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* Log writer
|
2014-02-26 10:47:02 +01:00
|
|
|
*
|
2014-10-31 10:27:17 +01:00
|
|
|
* @var \Icinga\Application\Logger\LogWriter
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-10-16 15:54:13 +02:00
|
|
|
protected $writer;
|
2014-02-26 10:47:02 +01:00
|
|
|
|
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* Maximum level to emit
|
|
|
|
*
|
|
|
|
* @var int
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-10-16 15:54:13 +02:00
|
|
|
protected $level;
|
2014-02-26 10:47:02 +01:00
|
|
|
|
2015-04-13 17:11:42 +02:00
|
|
|
/**
|
|
|
|
* Error messages to be displayed prior to any other log message
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $configErrors = array();
|
|
|
|
|
2014-02-26 10:47:02 +01:00
|
|
|
/**
|
|
|
|
* Create a new logger object
|
|
|
|
*
|
2014-11-18 13:11:52 +01:00
|
|
|
* @param ConfigObject $config
|
2014-10-16 15:54:13 +02:00
|
|
|
*
|
|
|
|
* @throws ConfigurationError If the logging configuration directive 'log' is missing or if the logging level is
|
|
|
|
* not defined
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-11-18 13:11:52 +01:00
|
|
|
public function __construct(ConfigObject $config)
|
2014-02-26 10:47:02 +01:00
|
|
|
{
|
2014-10-16 15:54:13 +02:00
|
|
|
if ($config->log === null) {
|
|
|
|
throw new ConfigurationError('Required logging configuration directive \'log\' missing');
|
|
|
|
}
|
|
|
|
|
2015-04-13 17:09:49 +02:00
|
|
|
$this->setLevel($config->get('level', static::ERROR));
|
2014-02-26 10:47:02 +01:00
|
|
|
|
2014-10-16 15:54:13 +02:00
|
|
|
if (strtolower($config->get('log', 'syslog')) !== 'none') {
|
2014-09-03 11:34:21 +02:00
|
|
|
$this->writer = $this->createWriter($config);
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-13 17:09:49 +02:00
|
|
|
/**
|
|
|
|
* Set the logging level to use
|
|
|
|
*
|
|
|
|
* @param mixed $level
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*
|
|
|
|
* @throws ConfigurationError In case the given level is invalid
|
|
|
|
*/
|
|
|
|
public function setLevel($level)
|
|
|
|
{
|
|
|
|
if (is_numeric($level)) {
|
|
|
|
$level = (int) $level;
|
|
|
|
if (! isset(static::$levels[$level])) {
|
|
|
|
throw new ConfigurationError(
|
|
|
|
'Can\'t set logging level %d. Logging level is invalid. Use one of %s or one of the'
|
|
|
|
. ' Logger\'s constants.',
|
|
|
|
$level,
|
|
|
|
implode(', ', array_keys(static::$levels))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->level = $level;
|
|
|
|
} else {
|
|
|
|
$level = strtoupper($level);
|
|
|
|
$levels = array_flip(static::$levels);
|
|
|
|
if (! isset($levels[$level])) {
|
|
|
|
throw new ConfigurationError(
|
|
|
|
'Can\'t set logging level "%s". Logging level is invalid. Use one of %s.',
|
|
|
|
$level,
|
|
|
|
implode(', ', array_keys($levels))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->level = $levels[$level];
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2015-11-11 09:59:28 +01:00
|
|
|
/**
|
|
|
|
* Return the logging level being used
|
|
|
|
*
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
public function getLevel()
|
|
|
|
{
|
|
|
|
return $this->level;
|
|
|
|
}
|
|
|
|
|
2015-04-13 17:11:42 +02:00
|
|
|
/**
|
|
|
|
* Register the given message as config error
|
|
|
|
*
|
|
|
|
* Config errors are logged every time a log message is being logged.
|
|
|
|
*
|
|
|
|
* @param mixed $arg,... A string, exception or format-string + substitutions
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function registerConfigError()
|
|
|
|
{
|
|
|
|
if (func_num_args() > 0) {
|
|
|
|
$this->configErrors[] = static::formatMessage(func_get_args());
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-02-26 10:47:02 +01:00
|
|
|
/**
|
|
|
|
* Create a new logger object
|
|
|
|
*
|
2014-11-18 13:11:52 +01:00
|
|
|
* @param ConfigObject $config
|
2014-10-16 15:54:13 +02:00
|
|
|
*
|
|
|
|
* @return static
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-11-18 13:11:52 +01:00
|
|
|
public static function create(ConfigObject $config)
|
2014-02-26 10:47:02 +01:00
|
|
|
{
|
|
|
|
static::$instance = new static($config);
|
2014-10-16 15:54:13 +02:00
|
|
|
return static::$instance;
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* Create a log writer
|
2014-02-26 10:47:02 +01:00
|
|
|
*
|
2014-11-18 13:11:52 +01:00
|
|
|
* @param ConfigObject $config The configuration to initialize the writer with
|
2014-02-26 10:47:02 +01:00
|
|
|
*
|
2014-10-31 10:27:17 +01:00
|
|
|
* @return \Icinga\Application\Logger\LogWriter The requested log writer
|
|
|
|
* @throws ConfigurationError If the requested writer cannot be found
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-11-18 13:11:52 +01:00
|
|
|
protected function createWriter(ConfigObject $config)
|
2014-02-26 10:47:02 +01:00
|
|
|
{
|
2014-10-31 10:27:17 +01:00
|
|
|
$class = 'Icinga\\Application\\Logger\\Writer\\' . ucfirst(strtolower($config->log)) . 'Writer';
|
2014-10-16 15:54:13 +02:00
|
|
|
if (! class_exists($class)) {
|
2014-08-22 11:46:11 +02:00
|
|
|
throw new ConfigurationError(
|
|
|
|
'Cannot find log writer of type "%s"',
|
2014-10-16 15:54:13 +02:00
|
|
|
$config->log
|
2014-08-22 11:46:11 +02:00
|
|
|
);
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
return new $class($config);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-10-16 15:54:13 +02:00
|
|
|
* Log a message
|
2014-08-22 11:05:20 +02:00
|
|
|
*
|
2014-10-16 15:54:13 +02:00
|
|
|
* @param int $level The logging level
|
|
|
|
* @param string $message The log message
|
2014-02-26 10:47:02 +01:00
|
|
|
*/
|
2014-10-16 15:54:13 +02:00
|
|
|
public function log($level, $message)
|
2014-02-26 10:47:02 +01:00
|
|
|
{
|
2014-10-29 16:51:17 +01:00
|
|
|
if ($this->writer !== null && $this->level <= $level) {
|
2015-04-13 17:11:42 +02:00
|
|
|
foreach ($this->configErrors as $error_message) {
|
|
|
|
$this->writer->log(static::ERROR, $error_message);
|
|
|
|
}
|
|
|
|
|
2014-10-16 15:54:13 +02:00
|
|
|
$this->writer->log($level, $message);
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a string representation of the passed arguments
|
|
|
|
*
|
|
|
|
* This method provides three different processing techniques:
|
|
|
|
* - If the only passed argument is a string it is returned unchanged
|
|
|
|
* - If the only passed argument is an exception it is formatted as follows:
|
|
|
|
* <name> in <file>:<line> with message: <message>[ <- <name> ...]
|
|
|
|
* - If multiple arguments are passed the first is interpreted as format-string
|
|
|
|
* that gets substituted with the remaining ones which can be of any type
|
|
|
|
*
|
|
|
|
* @param array $arguments The arguments to format
|
|
|
|
*
|
|
|
|
* @return string The formatted result
|
|
|
|
*/
|
|
|
|
protected static function formatMessage(array $arguments)
|
|
|
|
{
|
|
|
|
if (count($arguments) === 1) {
|
|
|
|
$message = $arguments[0];
|
|
|
|
|
|
|
|
if ($message instanceof Exception) {
|
|
|
|
$messages = array();
|
|
|
|
$error = $message;
|
|
|
|
do {
|
2015-07-24 16:19:20 +02:00
|
|
|
$messages[] = IcingaException::describe($error);
|
2014-02-26 10:47:02 +01:00
|
|
|
} while ($error = $error->getPrevious());
|
|
|
|
$message = implode(' <- ', $messages);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $message;
|
|
|
|
}
|
|
|
|
|
|
|
|
return vsprintf(
|
|
|
|
array_shift($arguments),
|
|
|
|
array_map(
|
2015-05-13 10:46:34 +02:00
|
|
|
function ($a) {
|
2015-10-05 12:54:04 +02:00
|
|
|
return is_string($a) ? $a : ($a instanceof Exception
|
|
|
|
? IcingaException::describe($a)
|
|
|
|
: json_encode($a));
|
2015-05-13 10:46:34 +02:00
|
|
|
},
|
2014-02-26 10:47:02 +01:00
|
|
|
$arguments
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Log a message with severity ERROR
|
|
|
|
*
|
|
|
|
* @param mixed $arg,... A string, exception or format-string + substitutions
|
|
|
|
*/
|
|
|
|
public static function error()
|
|
|
|
{
|
|
|
|
if (static::$instance !== null && func_num_args() > 0) {
|
2014-10-16 15:54:13 +02:00
|
|
|
static::$instance->log(static::ERROR, static::formatMessage(func_get_args()));
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Log a message with severity WARNING
|
|
|
|
*
|
|
|
|
* @param mixed $arg,... A string, exception or format-string + substitutions
|
|
|
|
*/
|
|
|
|
public static function warning()
|
|
|
|
{
|
|
|
|
if (static::$instance !== null && func_num_args() > 0) {
|
2014-10-16 15:54:13 +02:00
|
|
|
static::$instance->log(static::WARNING, static::formatMessage(func_get_args()));
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Log a message with severity INFO
|
|
|
|
*
|
|
|
|
* @param mixed $arg,... A string, exception or format-string + substitutions
|
|
|
|
*/
|
|
|
|
public static function info()
|
|
|
|
{
|
|
|
|
if (static::$instance !== null && func_num_args() > 0) {
|
2014-10-16 15:54:13 +02:00
|
|
|
static::$instance->log(static::INFO, static::formatMessage(func_get_args()));
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Log a message with severity DEBUG
|
|
|
|
*
|
|
|
|
* @param mixed $arg,... A string, exception or format-string + substitutions
|
|
|
|
*/
|
|
|
|
public static function debug()
|
|
|
|
{
|
|
|
|
if (static::$instance !== null && func_num_args() > 0) {
|
2014-10-16 15:54:13 +02:00
|
|
|
static::$instance->log(static::DEBUG, static::formatMessage(func_get_args()));
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|
|
|
|
}
|
2014-09-03 12:02:11 +02:00
|
|
|
|
2014-09-03 13:53:46 +02:00
|
|
|
/**
|
|
|
|
* Get the log writer to use
|
|
|
|
*
|
2014-10-31 10:27:17 +01:00
|
|
|
* @return \Icinga\Application\Logger\LogWriter
|
2014-09-03 13:53:46 +02:00
|
|
|
*/
|
2014-09-03 12:02:11 +02:00
|
|
|
public function getWriter()
|
|
|
|
{
|
|
|
|
return $this->writer;
|
|
|
|
}
|
|
|
|
|
2014-10-16 15:54:13 +02:00
|
|
|
/**
|
|
|
|
* Is the logger writing to Syslog?
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2014-09-05 15:28:39 +02:00
|
|
|
public static function writesToSyslog()
|
|
|
|
{
|
2015-04-07 10:31:57 +02:00
|
|
|
return static::$instance && static::$instance->getWriter() instanceof SyslogWriter;
|
2014-09-05 15:28:39 +02:00
|
|
|
}
|
|
|
|
|
2014-10-16 15:54:13 +02:00
|
|
|
/**
|
|
|
|
* Is the logger writing to a file?
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2014-09-05 15:28:39 +02:00
|
|
|
public static function writesToFile()
|
|
|
|
{
|
2015-04-07 10:31:57 +02:00
|
|
|
return static::$instance && static::$instance->getWriter() instanceof FileWriter;
|
2014-09-05 15:28:39 +02:00
|
|
|
}
|
|
|
|
|
2014-09-03 13:53:46 +02:00
|
|
|
/**
|
|
|
|
* Get this' instance
|
|
|
|
*
|
2014-10-16 15:54:13 +02:00
|
|
|
* @return static
|
2014-09-03 13:53:46 +02:00
|
|
|
*/
|
2014-09-03 12:02:11 +02:00
|
|
|
public static function getInstance()
|
|
|
|
{
|
|
|
|
return static::$instance;
|
|
|
|
}
|
2014-02-26 10:47:02 +01:00
|
|
|
}
|