This repository is bilingual. Below you will find the English version followed by the Spanish version.
Este repositorio es bilingüe. A continuación encontrarás la versión en inglés seguida de la versión en español.
 
  Software Development with Symfony 7, PHP 8.3, RabbitMQ latest, Redis latest and Hexagonal Architecture
Desarrollo de Software con Symfony 7, PHP 8.3, RabbitMQ latest, Redis latest y Arquitectura Hexagonal
- Requirements / Requisitos
- Installation / Instalación
- Configuration / Configuración
- Usage / Uso
- Architecture / Arquitectura
- Troubleshooting / Solución de Problemas
- Contributing / Contribuir
- Security / Seguridad
- Docker >= 20.10
- Docker Compose >= 2.0
- Git >= 2.30
- PHP >= 8.3
- Composer >= 2.0
- Symfony CLI >= 5.0
Create a .env file in the root directory with the following variables:
Crea un archivo .env en el directorio raíz con las siguientes variables:
# Application
APP_ENV=dev
APP_DEBUG=1
APP_SECRET=your_secret_here
# RabbitMQ
RABBITMQ_DEFAULT_USER=guest
RABBITMQ_DEFAULT_PASS=guest
MESSENGER_TRANSPORT_DSN=amqp://guest:guest@rabbitmq:5672/%2f/messages
# Redis
REDIS_URL=redis://redis:6379This project implements a messaging system with RabbitMQ and Redis in a Symfony environment, following the Hexagonal Architecture and using domain events to decouple business logic from infrastructure. Domain events are triggered within the application core when significant changes occur and are processed by specific handlers, allowing, for example, counters or other processes to be updated asynchronously.
Additionally, the Message module sends messages directly without domain events.
Whereas the User module utilizes domain events. For example, when a user registers, a UserRegisteredEvent is triggered to update the gender count. After this event is processed, a UserEmailSentEvent is dispatched to confirm the user's registration via email.
- 
The user is created in the application. 
- 
A domain event (UserRegisteredEvent) is triggered to update the gender count. 
- 
The event is sent to RabbitMQ via Symfony Messenger. 
- 
A specific handler (UserRegisteredEventHandler) processes the event and updates a counter (via a file, database, etc.). 
- 
After processing UserRegisteredEvent, a new event UserEmailSentEvent is triggered to notify the user via email. 
- 
UserEmailSentEvent is sent to RabbitMQ and processed by its corresponding handler. 
- 
Ensure the consumer is running to process the events: php bin/console messenger:consume async user_registered user_email_sent -vv (If all events are routed to the same transport, the worker will consume and execute the handlers accordingly.) 
git clone https://github.com/Luispfa/sf7-rabbitmq-ha.git
cd rabbit-mqdocker-compose up -d --builddocker exec -it sf7_php_ha bashphp composer installnc -zv sf7_rabbitmq_ha 15672
nc -zv sf7_redis_ha 6379curl -u guest:guest http://sf7_rabbitmq_ha:15672/api/overviewphp bin/console messenger:consume async user_registered user_email_sent -vvphp bin/console messenger:consume async user_registered user_email_sent --daemon9️⃣ If You Are on Windows, Add the Following Line to Your C:\Windows\System32\drivers\etc\hosts File:
127.0.0.1 dev.rabbit-mq.com
The messenger.yml configuration file has been updated to define dedicated queues for UserRegisteredEvent and UserEmailSentEvent.
Updated configuration in messenger.yml:
framework:
  messenger:
    transports:
      async:
        dsn: "%env(MESSENGER_TRANSPORT_DSN)%"
        options:
          exchange:
            name: messages
            type: direct
          queues:
            messages: ~
      user_registered:
        dsn: "%env(MESSENGER_TRANSPORT_DSN)%"
        options:
          exchange:
            name: user_events
            type: direct
          queues:
            user_registered_queue: ~
      user_email_sent:
        dsn: "%env(MESSENGER_TRANSPORT_DSN)%"
        options:
          exchange:
            name: email_events
            type: direct
          queues:
            user_email_sent_queue: ~
    routing:
      'App\Message\Domain\Message\Message': async
      'App\User\Domain\Event\UserRegisteredEvent': user_registered
      'App\User\Domain\Event\UserEmailSentEvent': user_email_sent- Method: POST
- URL: http://dev.rabbit-mq.com/send-message
- Headers: Content-Type: application/json
- Body (JSON):
{
  "message": "Hello RabbitMQ!"
}- Method: POST
- URL: http://dev.rabbit-mq.com/register-user
- Headers: Content-Type: application/json
- Body (JSON):
{
  "name": "Juan",
  "lastname": "Flores",
  "gender": "Male",
  "email": "[email protected]"
}- RabbitMQ is accessible at http://dev.rabbit-mq.com:15672/with username guest and password guest.
- Redis can be accessed at http://dev.rabbit-mq.com:8081/.
- If you are on Windows, add the following line to your C:\Windows\System32\drivers\etc\hosts file:
127.0.0.1 dev.rabbit-mq.com
src/
├── Message/                 # Message Module / Módulo de Mensajes
│   ├── Application/        # Application Services / Servicios de Aplicación
│   ├── Domain/            # Domain Logic / Lógica de Dominio
│   └── Infrastructure/    # Infrastructure Implementation / Implementación de Infraestructura
└── User/                   # User Module / Módulo de Usuarios
    ├── Application/       # Application Services / Servicios de Aplicación
    ├── Domain/           # Domain Logic / Lógica de Dominio
    └── Infrastructure/   # Infrastructure Implementation / Implementación de Infraestructura
sequenceDiagram
    participant Client
    participant Controller
    participant Domain
    participant RabbitMQ
    participant Handler
    Client->>Controller: POST /register-user
    Controller->>Domain: Create User
    Domain->>Domain: Trigger UserRegisteredEvent
    Domain->>RabbitMQ: Send Event
    RabbitMQ->>Handler: Process Event
    Handler->>Domain: Update Counters
    Domain->>Domain: Trigger UserEmailSentEvent
    Domain->>RabbitMQ: Send Event
    RabbitMQ->>Handler: Process Event
    Handler->>Client: Send Email
    - 
RabbitMQ Connection Issues / Problemas de Conexión con RabbitMQ - Check if RabbitMQ is running: docker ps | grep rabbitmq
- Verify ports are open: nc -zv sf7_rabbitmq_ha 5672
- Check RabbitMQ logs: docker logs sf7_rabbitmq_ha
 
- Check if RabbitMQ is running: 
- 
Redis Connection Issues / Problemas de Conexión con Redis - Check if Redis is running: docker ps | grep redis
- Verify Redis connection: redis-cli -h sf7_redis_ha ping
- Check Redis logs: docker logs sf7_redis_ha
 
- Check if Redis is running: 
- 
Consumer Issues / Problemas con el Consumidor - Ensure consumer is running: ps aux | grep messenger:consume
- Check consumer logs: docker logs sf7_php_ha
- Restart consumer if needed: php bin/console messenger:consume async user_registered user_email_sent -vv
 
- Ensure consumer is running: 
- Fork the repository / Haz un fork del repositorio
- Create your feature branch / Crea tu rama de características
- Commit your changes / Haz commit de tus cambios
- Push to the branch / Haz push a la rama
- Create a new Pull Request / Crea un nuevo Pull Request
- 
Environment Variables / Variables de Entorno - Never commit .envfiles / Nunca commits archivos.env
- Use different secrets for development and production / Usa diferentes secretos para desarrollo y producción
- Rotate secrets regularly / Rota los secretos regularmente
 
- Never commit 
- 
API Security / Seguridad de la API - Use HTTPS in production / Usa HTTPS en producción
- Implement rate limiting / Implementa límites de tasa
- Validate all input / Valida todas las entradas
 
- 
Message Queue Security / Seguridad de Cola de Mensajes - Use strong passwords / Usa contraseñas fuertes
- Enable SSL/TLS / Habilita SSL/TLS
- Monitor queue access / Monitorea el acceso a las colas
 
Este proyecto implementa un sistema de mensajería con RabbitMQ y Redis en un entorno Symfony, siguiendo la Arquitectura Hexagonal y utilizando eventos de dominio para desacoplar la lógica de negocio de la infraestructura. Los eventos de dominio se activan en el núcleo de la aplicación cuando ocurren cambios significativos y son procesados por manejadores específicos, permitiendo, por ejemplo, actualizar contadores u otros procesos de manera asíncrona.
Además, el módulo Mensaje envía mensajes directamente sin eventos de dominio.
Mientras que el módulo Usuario utiliza eventos de dominio. Por ejemplo, cuando un usuario se registra, se activa un UserRegisteredEvent para actualizar el conteo de género. Después de que se procesa este evento, se envía un UserEmailSentEvent para confirmar el registro del usuario por correo electrónico.
- 
Se crea el usuario en la aplicación. 
- 
Se dispara un evento de dominio (UserRegisteredEvent) para actualizar el conteo de género. 
- 
El evento se envía a RabbitMQ a través de Symfony Messenger. 
- 
Un manejador específico (UserRegisteredEventHandler) procesa el evento y actualiza un contador (mediante un archivo, base de datos, etc.). 
- 
Después de procesar UserRegisteredEvent, se activa un nuevo evento UserEmailSentEvent para notificar al usuario por correo electrónico. 
- 
UserEmailSentEvent se envía a RabbitMQ y es procesado por su manejador correspondiente. 
- 
Asegúrate de que el consumidor esté en ejecución para procesar los eventos: php bin/console messenger:consume async user_registered user_email_sent -vv (Si todos los eventos se enrutan al mismo transporte, el trabajador los consumirá y ejecutará los manejadores correspondientes.) 
git clone https://github.com/Luispfa/sf7-rabbitmq-ha.git
cd rabbit-mqdocker-compose up -d --builddocker exec -it sf7_php_ha bashphp composer installnc -zv sf7_rabbitmq_ha 15672
nc -zv sf7_redis_ha 6379curl -u guest:guest http://sf7_rabbitmq_ha:15672/api/overviewphp bin/console messenger:consume async user_registered user_email_sent -vvphp bin/console messenger:consume async user_registered user_email_sent --daemon9️⃣ Si Estás en Windows, Agrega la Siguiente Línea a tu Archivo C:\Windows\System32\drivers\etc\hosts:
127.0.0.1 dev.rabbit-mq.com
El archivo de configuración messenger.yml ha sido actualizado para definir colas dedicadas para UserRegisteredEvent y UserEmailSentEvent.
Configuración actualizada en messenger.yml:
framework:
  messenger:
    transports:
      async:
        dsn: "%env(MESSENGER_TRANSPORT_DSN)%"
        options:
          exchange:
            name: messages
            type: direct
          queues:
            messages: ~
      user_registered:
        dsn: "%env(MESSENGER_TRANSPORT_DSN)%"
        options:
          exchange:
            name: user_events
            type: direct
          queues:
            user_registered_queue: ~
      user_email_sent:
        dsn: "%env(MESSENGER_TRANSPORT_DSN)%"
        options:
          exchange:
            name: email_events
            type: direct
          queues:
            user_email_sent_queue: ~
    routing:
      'App\Message\Domain\Message\Message': async
      'App\User\Domain\Event\UserRegisteredEvent': user_registered
      'App\User\Domain\Event\UserEmailSentEvent': user_email_sent- Método: POST
- URL: http://dev.rabbit-mq.com/send-message
- Encabezados: Content-Type: application/json
- Cuerpo (JSON):
{
  "message": "¡Hola RabbitMQ!"
}- Método: POST
- URL: http://dev.rabbit-mq.com/register-user
- Encabezados: Content-Type: application/json
- Cuerpo (JSON):
{
  "name": "Juan",
  "lastname": "Flores",
  "gender": "Male",
  "email": "[email protected]"
}- RabbitMQ es accesible en http://dev.rabbit-mq.com:15672/con usuario guest y contraseña guest.
- Redis puede ser accedido en http://dev.rabbit-mq.com:8081/.
- Si estás en Windows, agrega la siguiente línea a tu archivo C:\Windows\System32\drivers\etc\hosts:
127.0.0.1 dev.rabbit-mq.com
src/
├── Message/                 # Módulo de Mensajes
│   ├── Application/        # Servicios de Aplicación
│   ├── Domain/            # Lógica de Dominio
│   └── Infrastructure/    # Implementación de Infraestructura
└── User/                   # Módulo de Usuarios
    ├── Application/       # Servicios de Aplicación
    ├── Domain/           # Lógica de Dominio
    └── Infrastructure/   # Implementación de Infraestructura
sequenceDiagram
    participant Client
    participant Controller
    participant Domain
    participant RabbitMQ
    participant Handler
    Client->>Controller: POST /register-user
    Controller->>Domain: Create User
    Domain->>Domain: Trigger UserRegisteredEvent
    Domain->>RabbitMQ: Send Event
    RabbitMQ->>Handler: Process Event
    Handler->>Domain: Update Counters
    Domain->>Domain: Trigger UserEmailSentEvent
    Domain->>RabbitMQ: Send Event
    RabbitMQ->>Handler: Process Event
    Handler->>Client: Send Email
    - 
Problemas de Conexión con RabbitMQ - Verificar si RabbitMQ está ejecutándose: docker ps | grep rabbitmq
- Verificar si los puertos están abiertos: nc -zv sf7_rabbitmq_ha 5672
- Revisar los logs de RabbitMQ: docker logs sf7_rabbitmq_ha
 
- Verificar si RabbitMQ está ejecutándose: 
- 
Problemas de Conexión con Redis - Verificar si Redis está ejecutándose: docker ps | grep redis
- Verificar la conexión con Redis: redis-cli -h sf7_redis_ha ping
- Revisar los logs de Redis: docker logs sf7_redis_ha
 
- Verificar si Redis está ejecutándose: 
- 
Problemas con el Consumidor - Asegurar que el consumidor está ejecutándose: ps aux | grep messenger:consume
- Revisar los logs del consumidor: docker logs sf7_php_ha
- Reiniciar el consumidor si es necesario: php bin/console messenger:consume async user_registered user_email_sent -vv
 
- Asegurar que el consumidor está ejecutándose: 
- Haz un fork del repositorio
- Crea tu rama de características
- Haz commit de tus cambios
- Haz push a la rama
- Crea un nuevo Pull Request
- 
Variables de Entorno - Nunca commits archivos .env
- Usa diferentes secretos para desarrollo y producción
- Rota los secretos regularmente
 
- Nunca commits archivos 
- 
Seguridad de la API - Usa HTTPS en producción
- Implementa límites de tasa
- Valida todas las entradas
 
- 
Seguridad de Cola de Mensajes - Usa contraseñas fuertes
- Habilita SSL/TLS
- Monitorea el acceso a las colas