| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- <?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 Monolog\Handler;
- use Closure;
- use Monolog\Handler\FingersCrossed\ErrorLevelActivationStrategy;
- use Monolog\Handler\FingersCrossed\ActivationStrategyInterface;
- use Monolog\Level;
- use Monolog\Logger;
- use Monolog\ResettableInterface;
- use Monolog\Formatter\FormatterInterface;
- use Psr\Log\LogLevel;
- use Monolog\LogRecord;
- /**
- * Buffers all records until a certain level is reached
- *
- * The advantage of this approach is that you don't get any clutter in your log files.
- * Only requests which actually trigger an error (or whatever your actionLevel is) will be
- * in the logs, but they will contain all records, not only those above the level threshold.
- *
- * You can then have a passthruLevel as well which means that at the end of the request,
- * even if it did not get activated, it will still send through log records of e.g. at least a
- * warning level.
- *
- * You can find the various activation strategies in the
- * Monolog\Handler\FingersCrossed\ namespace.
- *
- * @author Jordi Boggiano <j.boggiano@seld.be>
- */
- class FingersCrossedHandler extends Handler implements ProcessableHandlerInterface, ResettableInterface, FormattableHandlerInterface
- {
- use ProcessableHandlerTrait;
- /**
- * Handler or factory Closure($record, $this)
- *
- * @phpstan-var (Closure(LogRecord|null, HandlerInterface): HandlerInterface)|HandlerInterface
- */
- protected Closure|HandlerInterface $handler;
- protected ActivationStrategyInterface $activationStrategy;
- protected bool $buffering = true;
- protected int $bufferSize;
- /** @var LogRecord[] */
- protected array $buffer = [];
- protected bool $stopBuffering;
- protected Level|null $passthruLevel = null;
- protected bool $bubble;
- /**
- * @phpstan-param (Closure(LogRecord|null, HandlerInterface): HandlerInterface)|HandlerInterface $handler
- *
- * @param Closure|HandlerInterface $handler Handler or factory Closure($record|null, $fingersCrossedHandler).
- * @param int|string|Level|LogLevel::* $activationStrategy Strategy which determines when this handler takes action, or a level name/value at which the handler is activated
- * @param int $bufferSize How many entries should be buffered at most, beyond that the oldest items are removed from the buffer.
- * @param bool $bubble Whether the messages that are handled can bubble up the stack or not
- * @param bool $stopBuffering Whether the handler should stop buffering after being triggered (default true)
- * @param int|string|Level|LogLevel::*|null $passthruLevel Minimum level to always flush to handler on close, even if strategy not triggered
- *
- * @phpstan-param value-of<Level::VALUES>|value-of<Level::NAMES>|Level|LogLevel::*|ActivationStrategyInterface $activationStrategy
- * @phpstan-param value-of<Level::VALUES>|value-of<Level::NAMES>|Level|LogLevel::* $passthruLevel
- */
- public function __construct(Closure|HandlerInterface $handler, int|string|Level|ActivationStrategyInterface $activationStrategy = null, int $bufferSize = 0, bool $bubble = true, bool $stopBuffering = true, int|string|Level|null $passthruLevel = null)
- {
- if (null === $activationStrategy) {
- $activationStrategy = new ErrorLevelActivationStrategy(Level::Warning);
- }
- // convert simple int activationStrategy to an object
- if (!$activationStrategy instanceof ActivationStrategyInterface) {
- $activationStrategy = new ErrorLevelActivationStrategy($activationStrategy);
- }
- $this->handler = $handler;
- $this->activationStrategy = $activationStrategy;
- $this->bufferSize = $bufferSize;
- $this->bubble = $bubble;
- $this->stopBuffering = $stopBuffering;
- if ($passthruLevel !== null) {
- $this->passthruLevel = Logger::toMonologLevel($passthruLevel);
- }
- }
- /**
- * @inheritDoc
- */
- public function isHandling(LogRecord $record): bool
- {
- return true;
- }
- /**
- * Manually activate this logger regardless of the activation strategy
- */
- public function activate(): void
- {
- if ($this->stopBuffering) {
- $this->buffering = false;
- }
- $this->getHandler(end($this->buffer) ?: null)->handleBatch($this->buffer);
- $this->buffer = [];
- }
- /**
- * @inheritDoc
- */
- public function handle(LogRecord $record): bool
- {
- if (\count($this->processors) > 0) {
- $record = $this->processRecord($record);
- }
- if ($this->buffering) {
- $this->buffer[] = $record;
- if ($this->bufferSize > 0 && \count($this->buffer) > $this->bufferSize) {
- array_shift($this->buffer);
- }
- if ($this->activationStrategy->isHandlerActivated($record)) {
- $this->activate();
- }
- } else {
- $this->getHandler($record)->handle($record);
- }
- return false === $this->bubble;
- }
- /**
- * @inheritDoc
- */
- public function close(): void
- {
- $this->flushBuffer();
- $this->getHandler()->close();
- }
- public function reset(): void
- {
- $this->flushBuffer();
- $this->resetProcessors();
- if ($this->getHandler() instanceof ResettableInterface) {
- $this->getHandler()->reset();
- }
- }
- /**
- * Clears the buffer without flushing any messages down to the wrapped handler.
- *
- * It also resets the handler to its initial buffering state.
- */
- public function clear(): void
- {
- $this->buffer = [];
- $this->reset();
- }
- /**
- * Resets the state of the handler. Stops forwarding records to the wrapped handler.
- */
- private function flushBuffer(): void
- {
- if (null !== $this->passthruLevel) {
- $passthruLevel = $this->passthruLevel;
- $this->buffer = array_filter($this->buffer, static function ($record) use ($passthruLevel) {
- return $passthruLevel->includes($record->level);
- });
- if (\count($this->buffer) > 0) {
- $this->getHandler(end($this->buffer))->handleBatch($this->buffer);
- }
- }
- $this->buffer = [];
- $this->buffering = true;
- }
- /**
- * Return the nested handler
- *
- * If the handler was provided as a factory, this will trigger the handler's instantiation.
- */
- public function getHandler(LogRecord|null $record = null): HandlerInterface
- {
- if (!$this->handler instanceof HandlerInterface) {
- $handler = ($this->handler)($record, $this);
- if (!$handler instanceof HandlerInterface) {
- throw new \RuntimeException("The factory Closure should return a HandlerInterface");
- }
- $this->handler = $handler;
- }
- return $this->handler;
- }
- /**
- * @inheritDoc
- */
- public function setFormatter(FormatterInterface $formatter): HandlerInterface
- {
- $handler = $this->getHandler();
- if ($handler instanceof FormattableHandlerInterface) {
- $handler->setFormatter($formatter);
- return $this;
- }
- throw new \UnexpectedValueException('The nested handler of type '.\get_class($handler).' does not support formatters.');
- }
- /**
- * @inheritDoc
- */
- public function getFormatter(): FormatterInterface
- {
- $handler = $this->getHandler();
- if ($handler instanceof FormattableHandlerInterface) {
- return $handler->getFormatter();
- }
- throw new \UnexpectedValueException('The nested handler of type '.\get_class($handler).' does not support formatters.');
- }
- }
|