Вход Регистрация
Файл: vendor/monolog/monolog/src/Monolog/Formatter/JsonFormatter.php
Строк: 244
<?php declare(strict_types=1);

/*
 * This file is part of the Monolog package.
 *
 * (c) Jordi Boggiano <j.boggiano@seld.be>
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */

namespace MonologFormatter;

use 
Stringable;
use 
Throwable;
use 
MonologLogRecord;

/**
 * Encodes whatever record data is passed to it as json
 *
 * This can be useful to log to databases or remote APIs
 *
 * @author Jordi Boggiano <j.boggiano@seld.be>
 */
class JsonFormatter extends NormalizerFormatter
{
    public const 
BATCH_MODE_JSON 1;
    public const 
BATCH_MODE_NEWLINES 2;

    
/** @var self::BATCH_MODE_* */
    
protected int $batchMode;

    protected 
bool $appendNewline;

    protected 
bool $ignoreEmptyContextAndExtra;

    protected 
bool $includeStacktraces false;

    
/**
     * @param self::BATCH_MODE_* $batchMode
     */
    
public function __construct(int $batchMode self::BATCH_MODE_JSONbool $appendNewline truebool $ignoreEmptyContextAndExtra falsebool $includeStacktraces false)
    {
        
$this->batchMode $batchMode;
        
$this->appendNewline $appendNewline;
        
$this->ignoreEmptyContextAndExtra $ignoreEmptyContextAndExtra;
        
$this->includeStacktraces $includeStacktraces;

        
parent::__construct();
    }

    
/**
     * The batch mode option configures the formatting style for
     * multiple records. By default, multiple records will be
     * formatted as a JSON-encoded array. However, for
     * compatibility with some API endpoints, alternative styles
     * are available.
     */
    
public function getBatchMode(): int
    
{
        return 
$this->batchMode;
    }

    
/**
     * True if newlines are appended to every formatted record
     */
    
public function isAppendingNewlines(): bool
    
{
        return 
$this->appendNewline;
    }

    
/**
     * @inheritDoc
     */
    
public function format(LogRecord $record): string
    
{
        
$normalized $this->normalizeRecord($record);

        return 
$this->toJson($normalizedtrue) . ($this->appendNewline "n" '');
    }

    
/**
     * @inheritDoc
     */
    
public function formatBatch(array $records): string
    
{
        return 
match ($this->batchMode) {
            static::
BATCH_MODE_NEWLINES => $this->formatBatchNewlines($records),
            default => 
$this->formatBatchJson($records),
        };
    }

    
/**
     * @return $this
     */
    
public function includeStacktraces(bool $include true): self
    
{
        
$this->includeStacktraces $include;

        return 
$this;
    }

    
/**
     * @return array<array<mixed>|bool|float|int|stdClass|string|null>
     */
    
protected function normalizeRecord(LogRecord $record): array
    {
        
$normalized parent::normalizeRecord($record);

        if (isset(
$normalized['context']) && $normalized['context'] === []) {
            if (
$this->ignoreEmptyContextAndExtra) {
                unset(
$normalized['context']);
            } else {
                
$normalized['context'] = new stdClass;
            }
        }
        if (isset(
$normalized['extra']) && $normalized['extra'] === []) {
            if (
$this->ignoreEmptyContextAndExtra) {
                unset(
$normalized['extra']);
            } else {
                
$normalized['extra'] = new stdClass;
            }
        }

        return 
$normalized;
    }

    
/**
     * Return a JSON-encoded array of records.
     *
     * @phpstan-param LogRecord[] $records
     */
    
protected function formatBatchJson(array $records): string
    
{
        
$formatted array_map(fn (LogRecord $record) => $this->normalizeRecord($record), $records);

        return 
$this->toJson($formattedtrue);
    }

    
/**
     * Use new lines to separate records instead of a
     * JSON-encoded array.
     *
     * @phpstan-param LogRecord[] $records
     */
    
protected function formatBatchNewlines(array $records): string
    
{
        
$oldNewline $this->appendNewline;
        
$this->appendNewline false;
        
$formatted array_map(fn (LogRecord $record) => $this->format($record), $records);
        
$this->appendNewline $oldNewline;

        return 
implode("n"$formatted);
    }

    
/**
     * Normalizes given $data.
     *
     * @return null|scalar|array<mixed[]|scalar|null|object>|object
     */
    
protected function normalize(mixed $dataint $depth 0): mixed
    
{
        if (
$depth $this->maxNormalizeDepth) {
            return 
'Over '.$this->maxNormalizeDepth.' levels deep, aborting normalization';
        }

        if (
is_array($data)) {
            
$normalized = [];

            
$count 1;
            foreach (
$data as $key => $value) {
                if (
$count++ > $this->maxNormalizeItemCount) {
                    
$normalized['...'] = 'Over '.$this->maxNormalizeItemCount.' items ('.count($data).' total), aborting normalization';
                    break;
                }

                
$normalized[$key] = $this->normalize($value$depth 1);
            }

            return 
$normalized;
        }

        if (
is_object($data)) {
            if (
$data instanceof DateTimeInterface) {
                return 
$this->formatDate($data);
            }

            if (
$data instanceof Throwable) {
                return 
$this->normalizeException($data$depth);
            }

            
// if the object has specific json serializability we want to make sure we skip the __toString treatment below
            
if ($data instanceof JsonSerializable) {
                return 
$data;
            }

            if (
$data instanceof Stringable) {
                return 
$data->__toString();
            }

            if (
get_class($data) === '__PHP_Incomplete_Class') {
                return new 
ArrayObject($data);
            }

            return 
$data;
        }

        if (
is_resource($data)) {
            return 
parent::normalize($data);
        }

        return 
$data;
    }

    
/**
     * Normalizes given exception with or without its own stack trace based on
     * `includeStacktraces` property.
     *
     * @return array<array-key, string|int|array<string|int|array<string>>>
     */
    
protected function normalizeException(Throwable $eint $depth 0): array
    {
        
$data parent::normalizeException($e$depth);
        if (!
$this->includeStacktraces) {
            unset(
$data['trace']);
        }

        return 
$data;
    }
}
Онлайн: 0
Реклама