Code Coverage
 
Classes and Traits
Functions and Methods
Lines
Total
0.00% covered (danger)
0.00%
0 / 2
12.50% covered (danger)
12.50%
1 / 8
CRAP
71.05% covered (warning)
71.05%
27 / 38
ServerLogHandlerTrait
0.00% covered (danger)
0.00%
0 / 1
14.29% covered (danger)
14.29%
1 / 7
21.97
71.43% covered (warning)
71.43%
25 / 35
 __construct
0.00% covered (danger)
0.00%
0 / 1
2.02
83.33% covered (warning)
83.33%
5 / 6
 handle
0.00% covered (danger)
0.00%
0 / 1
4.25
75.00% covered (warning)
75.00%
6 / 8
 write
0.00% covered (danger)
0.00%
0 / 1
3.33
66.67% covered (warning)
66.67%
6 / 9
 getDefaultFormatter
0.00% covered (danger)
0.00%
0 / 1
2
0.00% covered (danger)
0.00%
0 / 1
 nullErrorHandler
0.00% covered (danger)
0.00%
0 / 1
2
0.00% covered (danger)
0.00%
0 / 1
 createSocket
100.00% covered (success)
100.00%
1 / 1
2
100.00% covered (success)
100.00%
4 / 4
 formatRecord
0.00% covered (danger)
0.00%
0 / 1
3.33
66.67% covered (warning)
66.67%
4 / 6
ServerLogHandler
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
0.00% covered (danger)
0.00%
0 / 1
 getDefaultFormatter
0.00% covered (danger)
0.00%
0 / 1
2
0.00% covered (danger)
0.00%
0 / 1
<?php
/*
 * This file is part of the Symfony package.
 *
 * (c) Fabien Potencier <fabien@symfony.com>
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */
namespace Symfony\Bridge\Monolog\Handler;
use Monolog\Formatter\FormatterInterface;
use Monolog\Handler\AbstractProcessingHandler;
use Monolog\Handler\FormattableHandlerTrait;
use Monolog\Logger;
use Symfony\Bridge\Monolog\Formatter\VarDumperFormatter;
if (trait_exists(FormattableHandlerTrait::class)) {
    class ServerLogHandler extends AbstractProcessingHandler
    {
        use ServerLogHandlerTrait;
        /**
         * {@inheritdoc}
         */
        protected function getDefaultFormatter(): FormatterInterface
        {
            return new VarDumperFormatter();
        }
    }
} else {
    class ServerLogHandler extends AbstractProcessingHandler
    {
        use ServerLogHandlerTrait;
        /**
         * {@inheritdoc}
         */
        protected function getDefaultFormatter()
        {
            return new VarDumperFormatter();
        }
    }
}
/**
 * @author GrĂ©goire Pineau <lyrixx@lyrixx.info>
 */
trait ServerLogHandlerTrait
{
    private $host;
    private $context;
    private $socket;
    public function __construct(string $host, int $level = Logger::DEBUG, bool $bubble = true, array $context = [])
    {
        parent::__construct($level, $bubble);
        if (false === strpos($host, '://')) {
            $host = 'tcp://'.$host;
        }
        $this->host = $host;
        $this->context = stream_context_create($context);
    }
    /**
     * {@inheritdoc}
     */
    public function handle(array $record): bool
    {
        if (!$this->isHandling($record)) {
            return false;
        }
        set_error_handler(self::class.'::nullErrorHandler');
        try {
            if (!$this->socket = $this->socket ?: $this->createSocket()) {
                return false === $this->bubble;
            }
        } finally {
            restore_error_handler();
        }
        return parent::handle($record);
    }
    protected function write(array $record): void
    {
        $recordFormatted = $this->formatRecord($record);
        set_error_handler(self::class.'::nullErrorHandler');
        try {
            if (-1 === stream_socket_sendto($this->socket, $recordFormatted)) {
                stream_socket_shutdown($this->socket, STREAM_SHUT_RDWR);
                // Let's retry: the persistent connection might just be stale
                if ($this->socket = $this->createSocket()) {
                    stream_socket_sendto($this->socket, $recordFormatted);
                }
            }
        } finally {
            restore_error_handler();
        }
    }
    /**
     * {@inheritdoc}
     */
    protected function getDefaultFormatter(): FormatterInterface
    {
        return new VarDumperFormatter();
    }
    private static function nullErrorHandler()
    {
    }
    private function createSocket()
    {
        $socket = stream_socket_client($this->host, $errno, $errstr, 0, STREAM_CLIENT_CONNECT | STREAM_CLIENT_ASYNC_CONNECT | STREAM_CLIENT_PERSISTENT, $this->context);
        if ($socket) {
            stream_set_blocking($socket, false);
        }
        return $socket;
    }
    private function formatRecord(array $record): string
    {
        $recordFormatted = $record['formatted'];
        foreach (['log_uuid', 'uuid', 'uid'] as $key) {
            if (isset($record['extra'][$key])) {
                $recordFormatted['log_id'] = $record['extra'][$key];
                break;
            }
        }
        return base64_encode(serialize($recordFormatted))."\n";
    }
}