AbstractSenderHandler   A
last analyzed

Complexity

Total Complexity 12

Size/Duplication

Total Lines 130
Duplicated Lines 0 %

Test Coverage

Coverage 0%

Importance

Changes 1
Bugs 0 Features 0
Metric Value
wmc 12
eloc 33
c 1
b 0
f 0
dl 0
loc 130
ccs 0
cts 36
cp 0
rs 10

3 Methods

Rating   Name   Duplication   Size   Complexity  
A __construct() 0 5 1
B handle() 0 61 8
A resolveStrategies() 0 16 3
1
<?php
2
3
declare(strict_types=1);
4
5
/**
6
 * LibreDTE: Biblioteca PHP (Núcleo).
7
 * Copyright (C) LibreDTE <https://www.libredte.cl>
8
 *
9
 * Este programa es software libre: usted puede redistribuirlo y/o modificarlo
10
 * bajo los términos de la Licencia Pública General Affero de GNU publicada por
11
 * la Fundación para el Software Libre, ya sea la versión 3 de la Licencia, o
12
 * (a su elección) cualquier versión posterior de la misma.
13
 *
14
 * Este programa se distribuye con la esperanza de que sea útil, pero SIN
15
 * GARANTÍA ALGUNA; ni siquiera la garantía implícita MERCANTIL o de APTITUD
16
 * PARA UN PROPÓSITO DETERMINADO. Consulte los detalles de la Licencia Pública
17
 * General Affero de GNU para obtener una información más detallada.
18
 *
19
 * Debería haber recibido una copia de la Licencia Pública General Affero de
20
 * GNU junto a este programa.
21
 *
22
 * En caso contrario, consulte <http://www.gnu.org/licenses/agpl.html>.
23
 */
24
25
namespace libredte\lib\Core\Package\Billing\Component\Exchange\Abstract;
26
27
use Derafu\Lib\Core\Foundation\Abstract\AbstractHandler;
28
use libredte\lib\Core\Package\Billing\Component\Exchange\Contract\EnvelopeInterface;
29
use libredte\lib\Core\Package\Billing\Component\Exchange\Contract\ExchangeBagInterface;
30
use libredte\lib\Core\Package\Billing\Component\Exchange\Contract\ExchangeHandlerInterface;
31
use libredte\lib\Core\Package\Billing\Component\Exchange\Contract\SenderStrategyInterface;
32
use libredte\lib\Core\Package\Billing\Component\Exchange\Contract\SenderWorkerInterface;
33
use libredte\lib\Core\Package\Billing\Component\Exchange\Exception\ExchangeException;
34
use LogicException;
35
36
/**
37
 * Clase base para los handlers de envío del proceso de intercambio.
38
 */
39
abstract class AbstractSenderHandler extends AbstractHandler implements ExchangeHandlerInterface
40
{
41
    /**
42
     * Constructor del handler.
43
     *
44
     * @param SenderWorkerInterface $senderWorker
45
     * @param array $strategies Estrategias que este handler puede manejar.
46
     */
47
    public function __construct(
48
        private SenderWorkerInterface $senderWorker,
49
        iterable $strategies = []
50
    ) {
51
        parent::__construct($strategies);
52
    }
53
54
    /**
55
     * {@inheritDoc}
56
     */
57
    public function handle(ExchangeBagInterface $bag): array
58
    {
59
        // Si no hay sobres error.
60
        if (!$bag->hasEnvelopes()) {
61
            throw new LogicException(
62
                'La bolsa de intercambio que se desea enviar no posee sobres.'
63
            );
64
        }
65
66
        // Procesar cada sobre.
67
        foreach ($bag->getEnvelopes() as $envelope) {
68
            // Si no hay documentos error.
69
            if (!$envelope->countDocuments()) {
70
                throw new LogicException(
71
                    'El sobre que se desea enviar no posee documentos.'
72
                );
73
            }
74
75
            // Verificar si este sobre debe ser procesado por el handler.
76
            // Esta es una revisión general del handler. Evita solicitar la
77
            // estrategia si no se debe procesar por este handler.
78
            if (!$this->shouldProcess($envelope)) {
79
                continue;
80
            }
81
82
            // Verificar que el sobre tenga lo necesario.
83
            // Esta es una revisión general del handler. Evita solicitar la
84
            // estrategia si no se debe procesar por este handler.
85
            if (!$this->hasRequiredData($envelope)) {
86
                continue;
87
            }
88
89
            // Determinar qué estrategias se ejecutarán.
90
            $strategies = $this->resolveStrategies($envelope);
91
            if (empty($strategies)) {
92
                continue;
93
            }
94
95
            // Iterar las estrategias.
96
            foreach ($strategies as $strategy) {
97
                // Armar la bolsa para el worker. Se reutilizan las opciones y
98
                // se pasa solo el sobre que se está procesando.
99
                $bagClass = get_class($bag);
100
                $bagForWorker = new $bagClass($bag->getOptions()->all());
101
                assert($bagForWorker instanceof ExchangeBagInterface);
102
                $bagForWorker->addEnvelope($envelope);
103
                $bagForWorker->getOptions()->set('strategy', $strategy);
104
105
                // Enviar el sobre usando el método send() del worker.
106
                $workerResults = $this->senderWorker->send($bagForWorker);
107
108
                // La llamada previa a send() entrega un arreglo, pero como se
109
                // pasó solo un sobre habrá solo un resultado del worker. Este
110
                // resultado se agrega a los resultados generales del handler.
111
                $bag->addResult($workerResults[0]);
112
            }
113
        }
114
115
        // Entregar los resultados del envío de todos los sobres con todas las
116
        // estrategias que se hayan ejecutado.
117
        return $bag->getResults();
118
    }
119
120
    /**
121
     * Determina si el sobre debe ser procesado por el handler y pasado a una
122
     * estrategia si tiene los datos necesarios y se encuentra una estrategia
123
     * válida.
124
     *
125
     * @param EnvelopeInterface $envelope
126
     * @return bool
127
     */
128
    abstract protected function shouldProcess(EnvelopeInterface $envelope): bool;
129
130
    /**
131
     * Determina si el sobre tiene los datos mínimos necesarios.
132
     *
133
     * Estos datos mínimos son independientes de la estrategia que use el
134
     * handler, pero están relacionados. Por ejemplo, estrategias que envían el
135
     * sobre por correo electrónico requerirán un correo electrónico en el
136
     * destinatario.
137
     *
138
     * @param EnvelopeInterface $envelope
139
     * @return bool
140
     */
141
    abstract protected function hasRequiredData(EnvelopeInterface $envelope): bool;
142
143
    /**
144
     * Entrega las estrategias que efectivamente se pueden ejecutar con el sobre
145
     * que se ha pasado.
146
     *
147
     * Este método revisa cada estrategia pasando el sobre para saber si la
148
     * estrategia lo puede procesar.
149
     *
150
     * @param EnvelopeInterface $envelope
151
     * @return string[] Códigos de las estrategias que se pueden ejecutar.
152
     */
153
    protected function resolveStrategies(EnvelopeInterface $envelope): array
154
    {
155
        $strategies = [];
156
157
        foreach ($this->getStrategies() as $name => $strategy) {
158
            assert($strategy instanceof SenderStrategyInterface);
159
            try {
160
                $strategy->canSend($envelope);
161
                $strategies[] = $name;
162
            } catch (ExchangeException $e) {
163
                // Falla silenciosamente pues no se puede procesar con esta
164
                // estrategia.
165
            }
166
        }
167
168
        return $strategies;
169
    }
170
}
171