feature #29958 introducing native php serialize() support for Messenger transport (weaverryan, xabbuh)
This PR was merged into the 4.3-dev branch. Discussion ---------- introducing native php serialize() support for Messenger transport | Q | A | ------------- | --- | Branch? | master | Bug fix? | yes and no | New feature? | yes and no | BC breaks? | maybe (yes if we change the default) | Deprecations? | no | Tests pass? | yes | Fixed tickets | #29163 | License | MIT | Doc PR | TODO! Messenger currently uses the Serialize to serialize to JSON and then unserialize. This creates a lot of issues: 1) The default serializer requires you to have getter & setter method (or public properties) for them to be serialized. This makes it easy for data to disappear. I've seen several really smart people have this problem. 2) Related to the above, the forced getters/setters (and no required constructor args) force you to design your message classes around this. It's not that the serializer is doing a bad job - it's just not the right use-case for it. This PR proposes simply using `serialize()` and `unserialize()`. This is the behavior we want: we want to put objects to sleep and wake them back up. I believe the original reason we did not do this was so that we could export "generic JSON", in case we wanted other workers (not our Symfony app) to consume the messages. But, that's an edge case, and could still be accomplished by creating your own serializer. Btw, Laravel uses `serialize()` as does Enqueue for (un)serializing Event objects. We're making our life more difficult for no benefit. Cheers! Commits -------97e2e32af4
Changing default serializer in Messenger component to PhpSerializer3111cef9a4
Update src/Symfony/Bundle/FrameworkBundle/Resources/config/messenger.xml4132bfebe7
updating CHANGELOGs and fixing testsb4788e4808
introducing native php serialize() support for Messenger transport
This commit is contained in:
commit
a6d2114277
@ -9,6 +9,12 @@ CHANGELOG
|
|||||||
* Added `ControllerTrait::isFormValid()`
|
* Added `ControllerTrait::isFormValid()`
|
||||||
* Added an `help_html` form option to display the `help` text as HTML
|
* Added an `help_html` form option to display the `help` text as HTML
|
||||||
|
|
||||||
|
* [BC Break] When using Messenger, the default transport changed from
|
||||||
|
using Symfony's serializer service to use `PhpSerializer`, which uses
|
||||||
|
PHP's native `serialize()` and `unserialize()` functions. To use the
|
||||||
|
original serialization method, set the `framework.messenger.serializer.id`
|
||||||
|
config option to `messenger.transport.symfony_serializer`.
|
||||||
|
|
||||||
4.2.0
|
4.2.0
|
||||||
-----
|
-----
|
||||||
|
|
||||||
|
@ -1060,7 +1060,7 @@ class Configuration implements ConfigurationInterface
|
|||||||
})
|
})
|
||||||
->end()
|
->end()
|
||||||
->children()
|
->children()
|
||||||
->scalarNode('id')->defaultValue(!class_exists(FullStack::class) && class_exists(Serializer::class) ? 'messenger.transport.symfony_serializer' : null)->end()
|
->scalarNode('id')->defaultValue('messenger.transport.native_php_serializer')->end()
|
||||||
->scalarNode('format')->defaultValue('json')->end()
|
->scalarNode('format')->defaultValue('json')->end()
|
||||||
->arrayNode('context')
|
->arrayNode('context')
|
||||||
->normalizeKeys(false)
|
->normalizeKeys(false)
|
||||||
|
@ -1529,7 +1529,7 @@ class FrameworkExtension extends Extension
|
|||||||
} else {
|
} else {
|
||||||
if ('messenger.transport.symfony_serializer' === $config['serializer']['id']) {
|
if ('messenger.transport.symfony_serializer' === $config['serializer']['id']) {
|
||||||
if (!$this->isConfigEnabled($container, $serializerConfig)) {
|
if (!$this->isConfigEnabled($container, $serializerConfig)) {
|
||||||
throw new LogicException('The default Messenger serializer cannot be enabled as the Serializer support is not available. Try enabling it or running "composer require symfony/serializer-pack".');
|
throw new LogicException('The Messenger serializer cannot be enabled as the Serializer support is not available. Try enabling it or running "composer require symfony/serializer-pack".');
|
||||||
}
|
}
|
||||||
|
|
||||||
$container->getDefinition('messenger.transport.symfony_serializer')
|
$container->getDefinition('messenger.transport.symfony_serializer')
|
||||||
|
@ -24,6 +24,8 @@
|
|||||||
</service>
|
</service>
|
||||||
<service id="Symfony\Component\Messenger\Transport\Serialization\SerializerInterface" alias="messenger.transport.serializer" />
|
<service id="Symfony\Component\Messenger\Transport\Serialization\SerializerInterface" alias="messenger.transport.serializer" />
|
||||||
|
|
||||||
|
<service id="messenger.transport.native_php_serializer" class="Symfony\Component\Messenger\Transport\Serialization\PhpSerializer" />
|
||||||
|
|
||||||
<!-- Middleware -->
|
<!-- Middleware -->
|
||||||
<service id="messenger.middleware.handle_message" class="Symfony\Component\Messenger\Middleware\HandleMessageMiddleware" abstract="true">
|
<service id="messenger.middleware.handle_message" class="Symfony\Component\Messenger\Middleware\HandleMessageMiddleware" abstract="true">
|
||||||
<argument /> <!-- Bus handler resolver -->
|
<argument /> <!-- Bus handler resolver -->
|
||||||
|
@ -19,7 +19,6 @@ use Symfony\Component\Config\Definition\Exception\InvalidConfigurationException;
|
|||||||
use Symfony\Component\Config\Definition\Processor;
|
use Symfony\Component\Config\Definition\Processor;
|
||||||
use Symfony\Component\Lock\Store\SemaphoreStore;
|
use Symfony\Component\Lock\Store\SemaphoreStore;
|
||||||
use Symfony\Component\Messenger\MessageBusInterface;
|
use Symfony\Component\Messenger\MessageBusInterface;
|
||||||
use Symfony\Component\Serializer\Serializer;
|
|
||||||
|
|
||||||
class ConfigurationTest extends TestCase
|
class ConfigurationTest extends TestCase
|
||||||
{
|
{
|
||||||
@ -295,7 +294,7 @@ class ConfigurationTest extends TestCase
|
|||||||
'routing' => [],
|
'routing' => [],
|
||||||
'transports' => [],
|
'transports' => [],
|
||||||
'serializer' => [
|
'serializer' => [
|
||||||
'id' => !class_exists(FullStack::class) && class_exists(Serializer::class) ? 'messenger.transport.symfony_serializer' : null,
|
'id' => 'messenger.transport.native_php_serializer',
|
||||||
'format' => 'json',
|
'format' => 'json',
|
||||||
'context' => [],
|
'context' => [],
|
||||||
],
|
],
|
||||||
|
@ -660,7 +660,7 @@ abstract class FrameworkExtensionTest extends TestCase
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException \Symfony\Component\DependencyInjection\Exception\LogicException
|
* @expectedException \Symfony\Component\DependencyInjection\Exception\LogicException
|
||||||
* @expectedExceptionMessage The default Messenger serializer cannot be enabled as the Serializer support is not available. Try enabling it or running "composer require symfony/serializer-pack".
|
* @expectedExceptionMessage The Messenger serializer cannot be enabled as the Serializer support is not available. Try enabling it or running "composer require symfony/serializer-pack".
|
||||||
*/
|
*/
|
||||||
public function testMessengerTransportConfigurationWithoutSerializer()
|
public function testMessengerTransportConfigurationWithoutSerializer()
|
||||||
{
|
{
|
||||||
|
@ -1,6 +1,16 @@
|
|||||||
CHANGELOG
|
CHANGELOG
|
||||||
=========
|
=========
|
||||||
|
|
||||||
|
4.3.0
|
||||||
|
-----
|
||||||
|
|
||||||
|
* Added `PhpSerializer` which uses PHP's native `serialize()` and
|
||||||
|
`unserialize()` to serialize messages to a transport
|
||||||
|
|
||||||
|
* [BC BREAK] If no serializer were passed, the default serializer
|
||||||
|
changed from `Serializer` to `PhpSerializer` inside `AmqpReceiver`,
|
||||||
|
`AmqpSender`, `AmqpTransport` and `AmqpTransportFactory`.
|
||||||
|
|
||||||
4.2.0
|
4.2.0
|
||||||
-----
|
-----
|
||||||
|
|
||||||
|
@ -0,0 +1,29 @@
|
|||||||
|
<?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\Messenger\Tests\Transport\Serialization;
|
||||||
|
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use Symfony\Component\Messenger\Envelope;
|
||||||
|
use Symfony\Component\Messenger\Tests\Fixtures\DummyMessage;
|
||||||
|
use Symfony\Component\Messenger\Transport\Serialization\PhpSerializer;
|
||||||
|
|
||||||
|
class PhpSerializerTest extends TestCase
|
||||||
|
{
|
||||||
|
public function testEncodedIsDecodable()
|
||||||
|
{
|
||||||
|
$serializer = new PhpSerializer();
|
||||||
|
|
||||||
|
$envelope = new Envelope(new DummyMessage('Hello'));
|
||||||
|
|
||||||
|
$this->assertEquals($envelope, $serializer->decode($serializer->encode($envelope)));
|
||||||
|
}
|
||||||
|
}
|
@ -13,7 +13,7 @@ namespace Symfony\Component\Messenger\Transport\AmqpExt;
|
|||||||
|
|
||||||
use Symfony\Component\Messenger\Transport\AmqpExt\Exception\RejectMessageExceptionInterface;
|
use Symfony\Component\Messenger\Transport\AmqpExt\Exception\RejectMessageExceptionInterface;
|
||||||
use Symfony\Component\Messenger\Transport\Receiver\ReceiverInterface;
|
use Symfony\Component\Messenger\Transport\Receiver\ReceiverInterface;
|
||||||
use Symfony\Component\Messenger\Transport\Serialization\Serializer;
|
use Symfony\Component\Messenger\Transport\Serialization\PhpSerializer;
|
||||||
use Symfony\Component\Messenger\Transport\Serialization\SerializerInterface;
|
use Symfony\Component\Messenger\Transport\Serialization\SerializerInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -32,7 +32,7 @@ class AmqpReceiver implements ReceiverInterface
|
|||||||
public function __construct(Connection $connection, SerializerInterface $serializer = null)
|
public function __construct(Connection $connection, SerializerInterface $serializer = null)
|
||||||
{
|
{
|
||||||
$this->connection = $connection;
|
$this->connection = $connection;
|
||||||
$this->serializer = $serializer ?? Serializer::create();
|
$this->serializer = $serializer ?? new PhpSerializer();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -13,7 +13,7 @@ namespace Symfony\Component\Messenger\Transport\AmqpExt;
|
|||||||
|
|
||||||
use Symfony\Component\Messenger\Envelope;
|
use Symfony\Component\Messenger\Envelope;
|
||||||
use Symfony\Component\Messenger\Transport\Sender\SenderInterface;
|
use Symfony\Component\Messenger\Transport\Sender\SenderInterface;
|
||||||
use Symfony\Component\Messenger\Transport\Serialization\Serializer;
|
use Symfony\Component\Messenger\Transport\Serialization\PhpSerializer;
|
||||||
use Symfony\Component\Messenger\Transport\Serialization\SerializerInterface;
|
use Symfony\Component\Messenger\Transport\Serialization\SerializerInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -31,7 +31,7 @@ class AmqpSender implements SenderInterface
|
|||||||
public function __construct(Connection $connection, SerializerInterface $serializer = null)
|
public function __construct(Connection $connection, SerializerInterface $serializer = null)
|
||||||
{
|
{
|
||||||
$this->connection = $connection;
|
$this->connection = $connection;
|
||||||
$this->serializer = $serializer ?? Serializer::create();
|
$this->serializer = $serializer ?? new PhpSerializer();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -12,7 +12,7 @@
|
|||||||
namespace Symfony\Component\Messenger\Transport\AmqpExt;
|
namespace Symfony\Component\Messenger\Transport\AmqpExt;
|
||||||
|
|
||||||
use Symfony\Component\Messenger\Envelope;
|
use Symfony\Component\Messenger\Envelope;
|
||||||
use Symfony\Component\Messenger\Transport\Serialization\Serializer;
|
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;
|
||||||
|
|
||||||
@ -31,7 +31,7 @@ class AmqpTransport implements TransportInterface
|
|||||||
public function __construct(Connection $connection, SerializerInterface $serializer = null)
|
public function __construct(Connection $connection, SerializerInterface $serializer = null)
|
||||||
{
|
{
|
||||||
$this->connection = $connection;
|
$this->connection = $connection;
|
||||||
$this->serializer = $serializer ?? Serializer::create();
|
$this->serializer = $serializer ?? new PhpSerializer();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
|
|
||||||
namespace Symfony\Component\Messenger\Transport\AmqpExt;
|
namespace Symfony\Component\Messenger\Transport\AmqpExt;
|
||||||
|
|
||||||
use Symfony\Component\Messenger\Transport\Serialization\Serializer;
|
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\TransportFactoryInterface;
|
use Symfony\Component\Messenger\Transport\TransportFactoryInterface;
|
||||||
use Symfony\Component\Messenger\Transport\TransportInterface;
|
use Symfony\Component\Messenger\Transport\TransportInterface;
|
||||||
@ -28,7 +28,7 @@ class AmqpTransportFactory implements TransportFactoryInterface
|
|||||||
|
|
||||||
public function __construct(SerializerInterface $serializer = null, bool $debug = false)
|
public function __construct(SerializerInterface $serializer = null, bool $debug = false)
|
||||||
{
|
{
|
||||||
$this->serializer = $serializer ?? Serializer::create();
|
$this->serializer = $serializer ?? new PhpSerializer();
|
||||||
$this->debug = $debug;
|
$this->debug = $debug;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,45 @@
|
|||||||
|
<?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\Messenger\Transport\Serialization;
|
||||||
|
|
||||||
|
use Symfony\Component\Messenger\Envelope;
|
||||||
|
use Symfony\Component\Messenger\Exception\InvalidArgumentException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Ryan Weaver<ryan@symfonycasts.com>
|
||||||
|
*
|
||||||
|
* @experimental in 4.2
|
||||||
|
*/
|
||||||
|
class PhpSerializer implements SerializerInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function decode(array $encodedEnvelope): Envelope
|
||||||
|
{
|
||||||
|
if (empty($encodedEnvelope['body'])) {
|
||||||
|
throw new InvalidArgumentException('Encoded envelope should have at least a "body".');
|
||||||
|
}
|
||||||
|
|
||||||
|
return unserialize($encodedEnvelope['body']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
public function encode(Envelope $envelope): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'body' => serialize($envelope),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user