Passed
Push — main ( 1bfc55...29d6ba )
by Fractal
02:09
created

DefaultContextExtractor   A

Complexity

Total Complexity 7

Size/Duplication

Total Lines 41
Duplicated Lines 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
eloc 20
dl 0
loc 41
rs 10
c 1
b 0
f 0
wmc 7

3 Methods

Rating   Name   Duplication   Size   Complexity  
A extract() 0 25 5
A getType() 0 3 1
A __construct() 0 3 1
1
<?php
2
3
declare(strict_types=1);
4
5
/**
6
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
7
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
8
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
9
 *
10
 * Copyright (c) 2024 Mykhailo Shtanko [email protected]
11
 *
12
 * For the full copyright and license information, please view the LICENSE.MD
13
 * file that was distributed with this source code.
14
 */
15
16
namespace FRZB\Component\MetricsPower\Logger\ContextExtractor;
17
18
use FRZB\Component\DependencyInjection\Attribute\AsService;
19
use FRZB\Component\DependencyInjection\Attribute\AsTagged;
20
use FRZB\Component\MetricsPower\Attribute\OptionsInterface;
21
use FRZB\Component\MetricsPower\Helper\ClassHelper;
22
use FRZB\Component\MetricsPower\Logger\Data\Context;
23
use Symfony\Component\Serializer\Encoder\JsonEncoder;
24
use Symfony\Component\Serializer\SerializerInterface;
25
26
/**
27
 * @implements ContextExtractorInterface<object, OptionsInterface>
28
 */
29
#[AsService, AsTagged(ContextExtractorInterface::class)]
30
class DefaultContextExtractor implements ContextExtractorInterface
31
{
32
    public const DEFAULT_TYPE = 'Default';
33
    private const MESSAGE_INFO = '[MetricsPower] [INFO] [MESSAGE: Operation succeed] [OPTIONS_CLASS: {options_class}] [TARGET_CLASS: {target_class}]';
34
    private const MESSAGE_ERROR = '[MetricsPower] [ERROR] [MESSAGE: Operation failed] [OPTIONS_CLASS: {options_class}] [MESSAGE_CLASS: {message_class}] [EXCEPTION_CLASS: {exception_class}] [EXCEPTION_MESSAGE: {exception_message}] [OPTIONS_VALUES: {option_values}]';
35
36
    public function __construct(
37
        private readonly SerializerInterface $serializer,
38
    ) {}
39
40
    public function extract(mixed $target, ?OptionsInterface $options = null, ?\Throwable $exception = null): Context
41
    {
42
        $message = $exception ? self::MESSAGE_ERROR : self::MESSAGE_INFO;
43
        $context = ['target_class' => ClassHelper::getShortName($target)];
44
45
        if ($options?->isSerializable()) {
46
            $context += ['target_values' => $this->serializer->serialize($target, JsonEncoder::FORMAT)];
47
        }
48
49
        if ($options) {
50
            $context += [
51
                'options_class' => ClassHelper::getShortName($options),
52
                'option_values' => ClassHelper::getProperties($options),
53
            ];
54
        }
55
56
        if ($exception) {
57
            $context += [
58
                'exception_class' => ClassHelper::getShortName($exception),
59
                'exception_message' => $exception->getMessage(),
60
                'exception_trace' => $exception->getTrace(),
61
            ];
62
        }
63
64
        return new Context($message, $context);
65
    }
66
67
    public static function getType(): string
68
    {
69
        return self::DEFAULT_TYPE;
70
    }
71
}
72