Flag of Ukraine
SymfonyCasts stands united with the people of Ukraine

Middleware de transacción y validación de Doctrine

Keep on Learning!

If you liked what you've learned so far, dive in!
Subscribe to get access to this tutorial plus
video, code and script downloads.

Start your All-Access Pass
Buy just this tutorial for $12.00

With a Subscription, click any sentence in the script to jump to that part of the video!

Login Subscribe

Ahora estamos utilizando tanto un patrón de bus de comandos, en el que creamos comandos y manejadores de comandos, como el patrón de bus de eventos: tenemos nuestro primer evento y manejador de eventos. La diferencia entre un comando y un evento... es un poco sutil. Cada comando debe tener exactamente un manejador: estamos ordenando que algo realice una acción concreta: AddPonkaToImage. Pero un evento es algo que suele despacharse después de que se realice esa acción, y el propósito es permitir que cualquier otra persona realice alguna acción secundaria: reaccionar a la acción.

Dos autobuses... ¿Por qué?

Obviamente, el propio Messenger es una herramienta lo suficientemente genérica como para poder utilizarla en estos dos casos de uso. Abre config/packages/messenger.yaml. Hemos decidido registrar un servicio de bus que estamos utilizando como nuestro bus de comandos y otro servicio de bus que estamos utilizando como nuestro bus de eventos. Pero... ¡en realidad no hay casi ninguna diferencia entre estos dos buses! Un bus no es más que una colección de middleware... así que las únicas diferencias son que el primero tieneAuditMiddleware... que también podríamos añadir al segundo... y que le dijimos al HandleMessageMiddleware del bus de eventos que permitiera "sin manejadores" para un mensaje: si un evento tiene cero manejadores, no lanzará una excepción.

Pero realmente... esto es tan poco importante que si quisieras utilizar un solo bus para todo, funcionaría muy bien.

Middleware de Validación, Transacción Doctrine, etc

Sin embargo, hay algunas personas que hacen sus buses de comandos y eventos un poco más diferentes. Busca en Google "Symfony Messenger multiple buses" para encontrar un artículo que habla de cómo gestionar varios buses. En este ejemplo, los documentos muestran tres buses diferentes: el bus de comandos, un bus de consultas -del que hablaremos en un minuto- y un bus de eventos. Pero cada bus tiene un middleware ligeramente diferente.

Estos dos middleware - validation y doctrine_transaction - vienen automáticamente con Symfony pero no están activados por defecto. Si añades el middleware validation, cuando envíes un mensaje, ese middleware validará el propio objeto mensaje a través del validador de Symfony. Si la validación falla, lanzará unValidationFailedException que puedes atrapar en tu código para leer los errores.

Esto es genial... pero no lo vamos a utilizar porque prefiero validar mis datos antes de enviarlos al bus. Simplemente tiene más sentido para mí y parece un poco más sencillo que una capa, en cierto modo, "invisible" que haga la validación por nosotros. Pero, es algo totalmente válido para usar.

El middleware doctrine_transaction es similar. Si activas este middleware, envolverá tu manejador dentro de una transacción Doctrine. Si el manejador lanza una excepción, revertirá la transacción. Y si no se lanza ninguna excepción, la confirmará. Esto significa que tu gestor no tendrá que llamar a flush() en el EntityManager: el middleware lo hace por ti.

Esto es genial... pero me parece bien crear y gestionar yo mismo las transacciones de Doctrine si las necesito. Así que éste es otro bonito middleware que me gusta, pero que no utilizo.

De todos modos, si utilizas más middleware del que estamos utilizando, entonces tus diferentes buses podrían empezar a ser... realmente más diferentes... y utilizar múltiples servicios de bus tendría más sentido. Como con todo, si el enfoque más sencillo -usar un solo bus para todo- te funciona, ¡genial! Hazlo. Si necesitas flexibilidad para tener diferentes middleware en diferentes buses, genial. Configura varios buses.

Dado que los buses múltiples son el caso de uso más complejo... y que estamos profundizando en Messenger, mantengamos nuestra configuración de buses múltiples y organicemos mejor nuestro código en torno a este concepto.

Mensajes enviados al bus equivocado

Busca tu terminal y ejecuta:

php bin/console debug:messenger

Ah... Ahora que tenemos varios autobuses, desglosa la información autobús por autobús. Dice que los siguientes mensajes pueden ser enviados a nuestro bus de comandos y... eh... estos mismos mensajes pueden ser enviados al bus de eventos.

Eso está... bien... pero no es lo que realmente queremos. Sabemos que ciertos mensajes son órdenes y se enviarán al bus de órdenes y otros son eventos. Pero cuando configuramos nuestros manejadores, nunca le dijimos a Messenger que este manejador sólo debe ser utilizado por este bus. Así, Messenger se asegura de que todos los buses conozcan todos los manejadores. Esto no es un gran problema, pero significa que si accidentalmente tomáramos este comando y lo enviáramos al bus de eventos, ¡funcionaría! Y si tomáramos este evento y lo enviáramos al bus de comandos, funcionaría. Si confiamos en que cada bus tenga un middleware bastante diferente, probablemente no querremos cometer ese error.

Así que... vamos a hacer algo totalmente opcional... pero agradable, cuando se utilizan eventos y comandos. Mira dentro de los directorios Message y MessageHandler: tenemos una mezcla de eventos y comandos. Claro, he puesto el evento en un subdirectorio Event/, pero no hemos hecho lo mismo con los comandos.

Hagamos eso a continuación: organicemos mejor nuestros mensajes y manejadores de mensajes. Una vez hecho esto, podemos utilizar un truco de configuración de servicios para asegurarnos de que el bus de comandos sólo conoce los manejadores de comandos y el bus de eventos sólo conoce los manejadores de eventos.

Leave a comment!

0
Login or Register to join the conversation
Cat in space

"Houston: no signs of life"
Start the conversation!

Este tutorial está construido con Symfony 4.3, pero funcionará bien en Symfony 4.4 o 5.

What PHP libraries does this tutorial use?

// composer.json
{
    "require": {
        "php": "^7.1.3",
        "ext-ctype": "*",
        "ext-iconv": "*",
        "composer/package-versions-deprecated": "^1.11", // 1.11.99
        "doctrine/annotations": "^1.0", // v1.8.0
        "doctrine/doctrine-bundle": "^1.6.10", // 1.11.2
        "doctrine/doctrine-migrations-bundle": "^1.3|^2.0", // v2.0.0
        "doctrine/orm": "^2.5.11", // v2.6.3
        "intervention/image": "^2.4", // 2.4.2
        "league/flysystem-bundle": "^1.0", // 1.1.0
        "phpdocumentor/reflection-docblock": "^3.0|^4.0", // 4.3.1
        "sensio/framework-extra-bundle": "^5.3", // v5.3.1
        "symfony/console": "4.3.*", // v4.3.2
        "symfony/dotenv": "4.3.*", // v4.3.2
        "symfony/flex": "^1.9", // v1.18.7
        "symfony/framework-bundle": "4.3.*", // v4.3.2
        "symfony/messenger": "4.3.*", // v4.3.4
        "symfony/property-access": "4.3.*", // v4.3.2
        "symfony/property-info": "4.3.*", // v4.3.2
        "symfony/serializer": "4.3.*", // v4.3.2
        "symfony/validator": "4.3.*", // v4.3.2
        "symfony/webpack-encore-bundle": "^1.5", // v1.6.2
        "symfony/yaml": "4.3.*" // v4.3.2
    },
    "require-dev": {
        "easycorp/easy-log-handler": "^1.0.7", // v1.0.7
        "symfony/debug-bundle": "4.3.*", // v4.3.2
        "symfony/maker-bundle": "^1.0", // v1.12.0
        "symfony/monolog-bundle": "^3.0", // v3.4.0
        "symfony/stopwatch": "4.3.*", // v4.3.2
        "symfony/twig-bundle": "4.3.*", // v4.3.2
        "symfony/var-dumper": "4.3.*", // v4.3.2
        "symfony/web-profiler-bundle": "4.3.*" // v4.3.2
    }
}
userVoice