This commit is contained in:
2021-02-28 21:18:03 +03:00
committed by GitHub
parent c6e2201237
commit ad39fb204f
3 changed files with 85 additions and 58 deletions

View File

@@ -11,8 +11,9 @@
} }
], ],
"require": { "require": {
"php": "^8.0.1",
"ext-json": "*", "ext-json": "*",
"nsq/nsq": "0.4.1", "nsq/nsq": "^0.5.1",
"symfony/framework-bundle": "^5.0", "symfony/framework-bundle": "^5.0",
"symfony/messenger": "^5.0" "symfony/messenger": "^5.0"
}, },

View File

@@ -4,42 +4,45 @@ declare(strict_types=1);
namespace Nsq\NsqBundle\Messenger; namespace Nsq\NsqBundle\Messenger;
use Amp\Loop;
use Amp\Promise;
use Generator; use Generator;
use JsonException; use JsonException;
use Nsq\Config\ClientConfig;
use Nsq\Message; use Nsq\Message;
use Nsq\Producer; use Nsq\Producer;
use Nsq\Subscriber; use Nsq\Reader;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use Symfony\Component\Messenger\Envelope; use Symfony\Component\Messenger\Envelope;
use Symfony\Component\Messenger\Exception\MessageDecodingFailedException; use Symfony\Component\Messenger\Exception\MessageDecodingFailedException;
use Symfony\Component\Messenger\Stamp\DelayStamp; use Symfony\Component\Messenger\Stamp\DelayStamp;
use Symfony\Component\Messenger\Stamp\TransportMessageIdStamp; use Symfony\Component\Messenger\Stamp\TransportMessageIdStamp;
use Symfony\Component\Messenger\Transport\Serialization\PhpSerializer;
use Symfony\Component\Messenger\Transport\Serialization\SerializerInterface; use Symfony\Component\Messenger\Transport\Serialization\SerializerInterface;
use Symfony\Component\Messenger\Transport\TransportInterface; use Symfony\Component\Messenger\Transport\TransportInterface;
use Throwable; use function Amp\Promise\wait;
use function json_decode; use function json_decode;
use function json_encode; use function json_encode;
use const JSON_THROW_ON_ERROR; use const JSON_THROW_ON_ERROR;
final class NsqTransport implements TransportInterface final class NsqTransport implements TransportInterface
{ {
private SerializerInterface $serializer; private ?Producer $producer = null;
private ?LoggerInterface $logger; private ?Reader $reader = null;
private ?Generator $generator = null; /**
* @var Promise<Message>|null
*/
private ?Promise $deferred = null;
public function __construct( public function __construct(
private Producer $producer, private string $address,
private Subscriber $subscriber,
private string $topic, private string $topic,
private string $channel, private string $channel,
SerializerInterface $serializer = null, private ClientConfig $clientConfig,
LoggerInterface $logger = null, private SerializerInterface $serializer,
private LoggerInterface $logger,
) { ) {
$this->serializer = $serializer ?? new PhpSerializer();
$this->logger = $logger;
} }
/** /**
@@ -47,6 +50,8 @@ final class NsqTransport implements TransportInterface
*/ */
public function send(Envelope $envelope): Envelope public function send(Envelope $envelope): Envelope
{ {
$producer = $this->getProducer();
$encodedMessage = $this->serializer->encode($envelope->withoutAll(NsqReceivedStamp::class)); $encodedMessage = $this->serializer->encode($envelope->withoutAll(NsqReceivedStamp::class));
$encodedMessage = json_encode($encodedMessage, JSON_THROW_ON_ERROR | JSON_UNESCAPED_UNICODE); $encodedMessage = json_encode($encodedMessage, JSON_THROW_ON_ERROR | JSON_UNESCAPED_UNICODE);
@@ -55,11 +60,13 @@ final class NsqTransport implements TransportInterface
$delay = null !== $delayStamp ? $delayStamp->getDelay() : null; $delay = null !== $delayStamp ? $delayStamp->getDelay() : null;
if (null === $delay) { if (null === $delay) {
$this->producer->pub($this->topic, $encodedMessage); $promise = $producer->publish($this->topic, $encodedMessage);
} else { } else {
$this->producer->dpub($this->topic, $encodedMessage, $delay / 1000); $promise = $producer->defer($this->topic, $encodedMessage, $delay);
} }
wait($promise);
return $envelope; return $envelope;
} }
@@ -68,36 +75,30 @@ final class NsqTransport implements TransportInterface
*/ */
public function get(): iterable public function get(): iterable
{ {
try { $reader = $this->getReader();
$this->producer->receive(0); // keepalive, handle heartbeat messages
} catch (Throwable $e) {
$this->logger->error('Producer keepalive failed.', ['exception' => $e]);
}
$generator = $this->generator; $promise = $this->deferred ??= $reader->consume();
if (null === $generator) {
$this->generator = $generator = $this->subscriber->subscribe($this->topic, $this->channel);
} else {
try {
$generator->next();
} catch (Throwable $e) {
$this->logger->error('Consumer next failed.', ['exception' => $e]);
return []; /** @var Message|null $message */
} $message = null;
} Loop::run(function () use (&$message, $promise): Generator {
Loop::delay(500, static function () {
Loop::stop();
});
/** @var Message|null $nsqMessage */ $message = yield $promise;
$nsqMessage = $generator->current(); });
if (null === $nsqMessage) { if (null === $message) {
return []; return [];
} }
$this->deferred = null;
try { try {
$encodedEnvelope = json_decode($nsqMessage->body, true, 512, JSON_THROW_ON_ERROR); $encodedEnvelope = json_decode($message->body, true, 512, JSON_THROW_ON_ERROR);
} catch (JsonException $e) { } catch (JsonException $e) {
$nsqMessage->finish(); wait($message->finish());
throw new MessageDecodingFailedException('', 0, $e); throw new MessageDecodingFailedException('', 0, $e);
} }
@@ -105,15 +106,15 @@ final class NsqTransport implements TransportInterface
try { try {
$envelope = $this->serializer->decode($encodedEnvelope); $envelope = $this->serializer->decode($encodedEnvelope);
} catch (MessageDecodingFailedException $e) { } catch (MessageDecodingFailedException $e) {
$nsqMessage->finish(); wait($message->finish());
throw $e; throw $e;
} }
return [ return [
$envelope->with( $envelope->with(
new NsqReceivedStamp($nsqMessage), new NsqReceivedStamp($message),
new TransportMessageIdStamp($nsqMessage->id), new TransportMessageIdStamp($message->id),
), ),
]; ];
} }
@@ -125,7 +126,7 @@ final class NsqTransport implements TransportInterface
{ {
$message = NsqReceivedStamp::getMessageFromEnvelope($envelope); $message = NsqReceivedStamp::getMessageFromEnvelope($envelope);
$message->finish(); wait($message->finish());
} }
/** /**
@@ -135,6 +136,38 @@ final class NsqTransport implements TransportInterface
{ {
$message = NsqReceivedStamp::getMessageFromEnvelope($envelope); $message = NsqReceivedStamp::getMessageFromEnvelope($envelope);
$message->finish(); wait($message->finish());
}
private function getProducer(): Producer
{
if (null === $this->producer) {
$this->producer = new Producer(
$this->address,
$this->clientConfig,
$this->logger,
);
}
wait($this->producer->connect());
return $this->producer;
}
private function getReader(): Reader
{
if (null === $this->reader) {
$this->reader = new Reader(
$this->address,
$this->topic,
$this->channel,
$this->clientConfig,
$this->logger,
);
}
wait($this->reader->connect());
return $this->reader;
} }
} }

View File

@@ -4,10 +4,7 @@ declare(strict_types=1);
namespace Nsq\NsqBundle\Messenger; namespace Nsq\NsqBundle\Messenger;
use Nsq\Consumer; use Nsq\Config\ClientConfig;
use Nsq\Producer;
use Nsq\Subscriber;
use Psr\Log\LoggerAwareTrait;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use Psr\Log\NullLogger; use Psr\Log\NullLogger;
use Symfony\Component\Messenger\Exception\InvalidArgumentException; use Symfony\Component\Messenger\Exception\InvalidArgumentException;
@@ -20,7 +17,7 @@ use function sprintf;
final class NsqTransportFactory implements TransportFactoryInterface final class NsqTransportFactory implements TransportFactoryInterface
{ {
use LoggerAwareTrait; private LoggerInterface $logger;
public function __construct(LoggerInterface $logger = null) public function __construct(LoggerInterface $logger = null)
{ {
@@ -42,20 +39,16 @@ final class NsqTransportFactory implements TransportFactoryInterface
} }
$address = sprintf('tcp://%s:%s', $parsedUrl['host'] ?? 'nsqd', $parsedUrl['port'] ?? 4150); $address = sprintf('tcp://%s:%s', $parsedUrl['host'] ?? 'nsqd', $parsedUrl['port'] ?? 4150);
$topic = $nsqOptions['topic'] ?? 'symfony-messenger';
$channel = $nsqOptions['channel'] ?? 'default';
$clientConfig = new ClientConfig();
return new NsqTransport( return new NsqTransport(
new Producer( $address,
address: $address, $topic,
logger: $this->logger, $channel,
), $clientConfig,
new Subscriber(
new Consumer(
address: $address,
logger: $this->logger,
)
),
$nsqOptions['topic'] ?? 'symfony-messenger',
$nsqOptions['channel'] ?? 'default',
$serializer, $serializer,
$this->logger, $this->logger,
); );