* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Component\Messenger; use Psr\Container\ContainerInterface; use Symfony\Component\Messenger\Exception\InvalidArgumentException; use Symfony\Component\Messenger\Stamp\BusNameStamp; /** * Bus of buses that is routable using a BusNameStamp. * * This is useful when passed to Worker: messages received * from the transport can be sent to the correct bus. * * @experimental in 4.3 * * @author Ryan Weaver */ class RoutableMessageBus implements MessageBusInterface { private $busLocator; private $fallbackBus; /** * @param ContainerInterface $busLocator A locator full of MessageBusInterface objects */ public function __construct(ContainerInterface $busLocator, MessageBusInterface $fallbackBus = null) { $this->busLocator = $busLocator; $this->fallbackBus = $fallbackBus; } public function dispatch($envelope, array $stamps = []): Envelope { if (!$envelope instanceof Envelope) { throw new InvalidArgumentException('Messages passed to RoutableMessageBus::dispatch() must be inside an Envelope'); } return $this->getMessageBus($envelope)->dispatch($envelope, $stamps); } private function getMessageBus(Envelope $envelope): MessageBusInterface { /** @var BusNameStamp|null $busNameStamp */ $busNameStamp = $envelope->last(BusNameStamp::class); if (null === $busNameStamp) { if (null === $this->fallbackBus) { throw new InvalidArgumentException(sprintf('Envelope is missing a BusNameStamp and no fallback message bus is configured on RoutableMessageBus.')); } return $this->fallbackBus; } $busName = $busNameStamp->getBusName(); if (!$this->busLocator->has($busName)) { throw new InvalidArgumentException(sprintf('Bus named "%s" does not exist.', $busName)); } return $this->busLocator->get($busName); } }