126 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			126 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			PHP
		
	
	
<?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\Component\Mailer\Transport;
 | 
						|
 | 
						|
use Symfony\Component\Mailer\Envelope;
 | 
						|
use Symfony\Component\Mailer\Exception\TransportException;
 | 
						|
use Symfony\Component\Mailer\Exception\TransportExceptionInterface;
 | 
						|
use Symfony\Component\Mailer\SentMessage;
 | 
						|
use Symfony\Component\Mime\RawMessage;
 | 
						|
 | 
						|
/**
 | 
						|
 * Uses several Transports using a round robin algorithm.
 | 
						|
 *
 | 
						|
 * @author Fabien Potencier <fabien@symfony.com>
 | 
						|
 */
 | 
						|
class RoundRobinTransport implements TransportInterface
 | 
						|
{
 | 
						|
    /**
 | 
						|
     * @var \SplObjectStorage<TransportInterface, float>
 | 
						|
     */
 | 
						|
    private \SplObjectStorage $deadTransports;
 | 
						|
    private array $transports = [];
 | 
						|
    private int $retryPeriod;
 | 
						|
    private int $cursor = -1;
 | 
						|
 | 
						|
    /**
 | 
						|
     * @param TransportInterface[] $transports
 | 
						|
     */
 | 
						|
    public function __construct(array $transports, int $retryPeriod = 60)
 | 
						|
    {
 | 
						|
        if (!$transports) {
 | 
						|
            throw new TransportException(sprintf('"%s" must have at least one transport configured.', static::class));
 | 
						|
        }
 | 
						|
 | 
						|
        $this->transports = $transports;
 | 
						|
        $this->deadTransports = new \SplObjectStorage();
 | 
						|
        $this->retryPeriod = $retryPeriod;
 | 
						|
    }
 | 
						|
 | 
						|
    public function send(RawMessage $message, ?Envelope $envelope = null): ?SentMessage
 | 
						|
    {
 | 
						|
        $exception = null;
 | 
						|
 | 
						|
        while ($transport = $this->getNextTransport()) {
 | 
						|
            try {
 | 
						|
                return $transport->send($message, $envelope);
 | 
						|
            } catch (TransportExceptionInterface $e) {
 | 
						|
                $exception ??= new TransportException('All transports failed.');
 | 
						|
                $exception->appendDebug(sprintf("Transport \"%s\": %s\n", $transport, $e->getDebug()));
 | 
						|
                $this->deadTransports[$transport] = microtime(true);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        throw $exception ?? new TransportException('No transports found.');
 | 
						|
    }
 | 
						|
 | 
						|
    public function __toString(): string
 | 
						|
    {
 | 
						|
        return $this->getNameSymbol().'('.implode(' ', array_map('strval', $this->transports)).')';
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Rotates the transport list around and returns the first instance.
 | 
						|
     */
 | 
						|
    protected function getNextTransport(): ?TransportInterface
 | 
						|
    {
 | 
						|
        if (-1 === $this->cursor) {
 | 
						|
            $this->cursor = $this->getInitialCursor();
 | 
						|
        }
 | 
						|
 | 
						|
        $cursor = $this->cursor;
 | 
						|
        while (true) {
 | 
						|
            $transport = $this->transports[$cursor];
 | 
						|
 | 
						|
            if (!$this->isTransportDead($transport)) {
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
            if ((microtime(true) - $this->deadTransports[$transport]) > $this->retryPeriod) {
 | 
						|
                $this->deadTransports->detach($transport);
 | 
						|
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
            if ($this->cursor === $cursor = $this->moveCursor($cursor)) {
 | 
						|
                return null;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        $this->cursor = $this->moveCursor($cursor);
 | 
						|
 | 
						|
        return $transport;
 | 
						|
    }
 | 
						|
 | 
						|
    protected function isTransportDead(TransportInterface $transport): bool
 | 
						|
    {
 | 
						|
        return $this->deadTransports->contains($transport);
 | 
						|
    }
 | 
						|
 | 
						|
    protected function getInitialCursor(): int
 | 
						|
    {
 | 
						|
        // the cursor initial value is randomized so that
 | 
						|
        // when are not in a daemon, we are still rotating the transports
 | 
						|
        return mt_rand(0, \count($this->transports) - 1);
 | 
						|
    }
 | 
						|
 | 
						|
    protected function getNameSymbol(): string
 | 
						|
    {
 | 
						|
        return 'roundrobin';
 | 
						|
    }
 | 
						|
 | 
						|
    private function moveCursor(int $cursor): int
 | 
						|
    {
 | 
						|
        return ++$cursor >= \count($this->transports) ? 0 : $cursor;
 | 
						|
    }
 | 
						|
}
 |