Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
32.63% covered (danger)
32.63%
31 / 95
27.78% covered (danger)
27.78%
5 / 18
CRAP
0.00% covered (danger)
0.00%
0 / 1
common_Logger
32.63% covered (danger)
32.63%
31 / 95
27.78% covered (danger)
27.78%
5 / 18
722.41
0.00% covered (danger)
0.00%
0 / 1
 singleton
100.00% covered (success)
100.00%
3 / 3
100.00% covered (success)
100.00%
1 / 1
2
 __construct
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 register
0.00% covered (danger)
0.00%
0 / 2
0.00% covered (danger)
0.00%
0 / 1
2
 log
100.00% covered (success)
100.00%
11 / 11
100.00% covered (success)
100.00%
1 / 1
6
 enable
0.00% covered (danger)
0.00%
0 / 2
0.00% covered (danger)
0.00%
0 / 1
2
 disable
100.00% covered (success)
100.00%
2 / 2
100.00% covered (success)
100.00%
1 / 1
1
 restore
66.67% covered (warning)
66.67%
2 / 3
0.00% covered (danger)
0.00%
0 / 1
2.15
 t
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 d
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 i
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 w
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 e
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 addTrace
0.00% covered (danger)
0.00%
0 / 6
0.00% covered (danger)
0.00%
0 / 1
12
 f
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 handleException
0.00% covered (danger)
0.00%
0 / 12
0.00% covered (danger)
0.00%
0 / 1
6
 handlePHPErrors
0.00% covered (danger)
0.00%
0 / 27
0.00% covered (danger)
0.00%
0 / 1
240
 handlePHPShutdown
0.00% covered (danger)
0.00%
0 / 7
0.00% covered (danger)
0.00%
0 / 1
20
 getContext
84.62% covered (warning)
84.62%
11 / 13
0.00% covered (danger)
0.00%
0 / 1
3.03
1<?php
2
3/**
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; under version 2
7 * of the License (non-upgradable).
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
17 *
18 * Copyright (c) 2002-2008 (original work) Public Research Centre Henri Tudor & University of Luxembourg
19 *                         (under the project TAO & TAO2);
20 *               2008-2010 (update and modification) Deutsche Institut für Internationale Pädagogische Forschung
21 *                         (under the project TAO-TRANSFER);
22 *               2009-2012 (update and modification) Public Research Centre Henri Tudor
23 *                         (under the project TAO-SUSTAIN & TAO-DEV);
24 *               2013 (update and modification) Open Assessment Technologies SA (under the project TAO-PRODUCT);
25 *
26 */
27
28/**
29 * Abstraction for the System Logger
30 *
31 * @access public
32 * @author Joel Bout, <joel.bout@tudor.lu>
33 * @package generis
34 */
35class common_Logger
36{
37    use \oat\oatbox\log\LoggerAwareTrait;
38
39    /**
40     * whenever or not the Logger is enabled
41     *
42     * @access private
43     * @var boolean
44     */
45    private $enabled = true;
46
47    /**
48     * a history of past states, to allow a restoration of the previous state
49     *
50     * @access private
51     * @var array
52     */
53    private $stateStack = [];
54
55    /**
56     * instance of the class Logger, to implement the singleton pattern
57     *
58     * @access private
59     * @var Logger
60     */
61    private static $instance = null;
62
63    /**
64     * The dispatcher of the Logger
65     *
66     * @access private
67     * @var Appender
68     */
69    private $dispatcher = null;
70
71    /**
72     * the lowest level of events representing the finest-grained processes
73     *
74     * @access public
75     * @var int
76     */
77    public const TRACE_LEVEL = 0;
78
79    /**
80     * the level of events representing fine grained informations for debugging
81     *
82     * @access public
83     * @var int
84     */
85    public const DEBUG_LEVEL = 1;
86
87    /**
88     * the level of information events that represent high level system events
89     *
90     * @access public
91     * @var int
92     */
93    public const INFO_LEVEL = 2;
94
95    /**
96     * the level of warning events that represent potential problems
97     *
98     * @access public
99     * @var int
100     */
101    public const WARNING_LEVEL = 3;
102
103    /**
104     * the level of error events that allow the system to continue
105     *
106     * @access public
107     * @var int
108     */
109    public const ERROR_LEVEL = 4;
110
111    /**
112     * the level of very severe error events that prevent the system to continue
113     *
114     * @access public
115     * @var int
116     */
117    public const FATAL_LEVEL = 5;
118
119    public const CONTEXT_ERROR_FILE = 'file';
120
121    public const CONTEXT_ERROR_LINE = 'line';
122
123    public const CONTEXT_TRACE = 'trace';
124
125    public const CONTEXT_EXCEPTION = 'exception';
126
127    /**
128     * Warnings that are acceptable in our projects
129     * invoked by the way generis/tao use abstract functions
130     *
131     * @access private
132     * @var array
133     */
134    private $ACCEPTABLE_WARNINGS = [];
135
136    // --- OPERATIONS ---
137
138    /**
139     * returns the existing Logger instance or instantiates a new one
140     *
141     * @access public
142     * @author Joel Bout, <joel.bout@tudor.lu>
143     * @return common_Logger
144     */
145    public static function singleton()
146    {
147        if (is_null(self::$instance)) {
148            self::$instance = new self();
149        }
150        return self::$instance;
151    }
152
153    /**
154     * Private constructor for singleton pattern
155     *
156     * @access private
157     * @author Joel Bout, <joel.bout@tudor.lu>
158     * @return mixed
159     */
160    private function __construct()
161    {
162    }
163
164    /**
165     * register the logger as errorhandler
166     * and shutdown function
167     *
168     * @access public
169     * @author Joel Bout, <joel.bout@tudor.lu>
170     * @return mixed
171     */
172    public function register()
173    {
174        // initialize the logger here to prevent fatal errors that occure if:
175        // while autoloading any class, an error gets thrown
176        // the logger initializes to handle this error,  and failes to autoload his files
177        set_error_handler([$this, 'handlePHPErrors']);
178        register_shutdown_function([$this, 'handlePHPShutdown']);
179    }
180
181    /**
182     * Short description of method log
183     *
184     * @access public
185     * @author Joel Bout, <joel.bout@tudor.lu>
186     * @param  int $level
187     * @param  string $message
188     * @param  array $tags
189     * @return mixed
190     */
191    public function log($level, $message, $tags = [])
192    {
193        if ($this->enabled) {
194            $this->disable();
195            try {
196                if (defined('CONFIG_PATH')) {
197                    // Gets the log context.
198                    $context = $this->getContext();
199                    $context = array_merge($context, $tags);
200                    if (!empty($context['file']) && !empty($context['line'])) {
201                        $tags['file'] = $context['file'];
202                        $tags['line'] = $context['line'];
203                    }
204
205                    $this->getLogger()->log(common_log_Logger2Psr::getPsrLevelFromCommon($level), $message, $tags);
206                }
207            } catch (\Exception $e) {
208                // Unable to use the logger service to retrieve the logger
209            }
210            $this->restore();
211        }
212    }
213
214    /**
215     * enables the logger, should not be used to restore a previous logger state
216     *
217     * @access public
218     * @author Joel Bout, <joel.bout@tudor.lu>
219     * @return mixed
220     */
221    public function enable()
222    {
223
224        $this->stateStack[] = self::singleton()->enabled;
225        $this->enabled = true;
226    }
227
228    /**
229     * disables the logger, should not be used to restore a previous logger
230     *
231     * @access public
232     * @author Joel Bout, <joel.bout@tudor.lu>
233     * @return mixed
234     */
235    public function disable()
236    {
237
238        $this->stateStack[] = self::singleton()->enabled;
239        $this->enabled = false;
240    }
241
242    /**
243     * restores the logger after its state was modified by enable() or disable()
244     *
245     * @access public
246     * @author Joel Bout, <joel.bout@tudor.lu>
247     * @return mixed
248     */
249    public function restore()
250    {
251
252        if (count($this->stateStack) > 0) {
253            $this->enabled = array_pop($this->stateStack);
254        } else {
255            self::e("Tried to restore Log state that was never changed");
256        }
257    }
258
259    /**
260     * trace logs finest-grained processes informations
261     *
262     * @access public
263     * @author Joel Bout, <joel.bout@tudor.lu>
264     * @param  string $message
265     * @param  array $tags
266     * @return mixed
267     */
268    public static function t($message, $tags = [])
269    {
270
271        self::singleton()->log(self::TRACE_LEVEL, $message, $tags);
272    }
273
274    /**
275     * debug logs fine grained informations for debugging
276     *
277     * @access public
278     * @author Joel Bout, <joel.bout@tudor.lu>
279     * @param  string $message
280     * @param  array $tags
281     * @return mixed
282     */
283    public static function d($message, $tags = [])
284    {
285
286        self::singleton()->log(self::DEBUG_LEVEL, $message, $tags);
287    }
288
289
290
291    /**
292     * info logs high level system events
293     *
294     * @access public
295     * @author Joel Bout, <joel.bout@tudor.lu>
296     * @param  string $message
297     * @param  array $tags
298     * @return mixed
299     */
300    public static function i($message, $tags = [])
301    {
302
303        self::singleton()->log(self::INFO_LEVEL, $message, $tags);
304    }
305
306    /**
307     * warning logs events that represent potential problems
308     *
309     * @access public
310     * @author Joel Bout, <joel.bout@tudor.lu>
311     * @param  string $message
312     * @param  array $tags
313     * @return mixed
314     */
315    public static function w($message, $tags = [])
316    {
317
318        self::singleton()->log(self::WARNING_LEVEL, $message, $tags);
319    }
320
321    /**
322     * error logs events that allow the system to continue
323     *
324     * @access public
325     * @author Joel Bout, <joel.bout@tudor.lu>
326     * @param  string $message
327     * @param  array $tags
328     * @return mixed
329     */
330    public static function e($message, $tags = [])
331    {
332        self::singleton()->log(self::ERROR_LEVEL, $message, self::addTrace($tags));
333    }
334
335    private static function addTrace(array $tags = [])
336    {
337        if (!isset($tags[self::CONTEXT_TRACE])) {
338            $trace = defined('DEBUG_BACKTRACE_IGNORE_ARGS')
339                ? debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS)
340                : debug_backtrace(false);
341
342            // remove 2 last traces which are error handler itself. to make logs more readable and reduce size of a log
343            $tags[self::CONTEXT_TRACE] = array_slice($trace, 2);
344        }
345
346        return $tags;
347    }
348
349    /**
350     * fatal logs very severe error events that prevent the system to continue
351     *
352     * @access public
353     * @author Joel Bout, <joel.bout@tudor.lu>
354     * @param  string $message
355     * @param  array $tags
356     * @return mixed
357     */
358    public static function f($message, $tags = [])
359    {
360        self::singleton()->log(self::FATAL_LEVEL, $message, self::addTrace($tags));
361    }
362
363    /**
364     * Short description of method handleException
365     *
366     * @access public
367     * @author Joel Bout, <joel.bout@tudor.lu>
368     * @param  Exception $exception
369     */
370    public function handleException(Exception $exception)
371    {
372        $severity = method_exists($exception, 'getSeverity') ? $exception->getSeverity() : self::ERROR_LEVEL;
373        self::singleton()
374            ->log(
375                $severity,
376                $exception->getMessage(),
377                [
378                    self::CONTEXT_EXCEPTION => get_class($exception),
379                    self::CONTEXT_ERROR_FILE => $exception->getFile(),
380                    self::CONTEXT_ERROR_LINE => $exception->getLine(),
381                    self::CONTEXT_TRACE => $exception->getTrace()
382                ]
383            );
384    }
385
386    /**
387     * A handler for php errors, should never be called manually
388     *
389     * @access public
390     * @author Joel Bout, <joel.bout@tudor.lu>
391     *
392     * @param  int $errorNumber
393     * @param  string $errorString
394     * @param  string $errorFile
395     * @param  mixed $errorLine
396     * @param  array $errorContext
397     *
398     * @return boolean
399     */
400    public function handlePHPErrors(
401        $errorNumber,
402        $errorString,
403        $errorFile = null,
404        $errorLine = null,
405        $errorContext = []
406    ) {
407        if (error_reporting() !== 0) {
408            if ($errorNumber === E_STRICT) {
409                foreach ($this->ACCEPTABLE_WARNINGS as $pattern) {
410                    if (preg_match($pattern, $errorString) > 0) {
411                        return false;
412                    }
413                }
414            }
415
416            switch ($errorNumber) {
417                case E_USER_ERROR:
418                case E_RECOVERABLE_ERROR:
419                    $severity = self::FATAL_LEVEL;
420                    break;
421                case E_WARNING:
422                case E_USER_WARNING:
423                    $severity = self::ERROR_LEVEL;
424                    break;
425                case E_NOTICE:
426                case E_USER_NOTICE:
427                    $severity = self::WARNING_LEVEL;
428                    break;
429                case E_DEPRECATED:
430                case E_USER_DEPRECATED:
431                case E_STRICT:
432                    $severity = self::DEBUG_LEVEL;
433                    break;
434                default:
435                    self::d('Unsupported PHP error type: ' . $errorNumber, 'common_Logger');
436                    $severity = self::ERROR_LEVEL;
437                    break;
438            }
439
440            self::singleton()->log(
441                $severity,
442                sprintf('php error(%s): %s', $errorNumber, $errorString),
443                [
444                    'PHPERROR',
445                    self::CONTEXT_ERROR_FILE => $errorFile,
446                    self::CONTEXT_ERROR_LINE => $errorLine,
447                    self::CONTEXT_TRACE      => self::addTrace()
448                ]
449            );
450        }
451
452        return false;
453    }
454
455    /**
456     * a workaround to catch fatal errors by handling the php shutdown,
457     * should never be called manually
458     *
459     * @access public
460     * @author Joel Bout, <joel.bout@tudor.lu>
461     * @return mixed
462     */
463    public function handlePHPShutdown()
464    {
465        $error = error_get_last();
466
467        if ($error !== null && ($error['type'] & (E_COMPILE_ERROR | E_ERROR | E_PARSE | E_CORE_ERROR)) !== 0) {
468            $msg = (isset($error['file'], $error['line']))
469                ? 'php error(' . $error['type'] . ') in ' . $error['file'] . '@' . $error['line'] . ': '
470                    . $error['message']
471                : 'php error(' . $error['type'] . '): ' . $error['message'];
472            self::singleton()->log(self::FATAL_LEVEL, $msg, ['PHPERROR']);
473        }
474    }
475
476    /**
477     * Returns the calling context.
478     *
479     * @return array
480     */
481    protected function getContext()
482    {
483        $trace = debug_backtrace();
484
485        $file = isset($trace[2]['file'])
486            ? $trace[2]['file']
487            : ''
488        ;
489
490        $line = isset($trace[2]['line'])
491            ? $trace[2]['line']
492            : ''
493        ;
494
495        return [
496            'file' => $file,
497            'line' => $line,
498        ];
499    }
500}