ody / cqrs
CQRS module for ody API framework
Requires
- php: >=8.3
- ext-swoole: ^6.0
Requires (Dev)
- phpunit/phpunit: ^9.6
- ramsey/uuid: ^3.7|^4.0
This package is auto-updated.
Last update: 2025-04-01 20:00:47 UTC
README
A robust and flexible CQRS (Command Query Responsibility Segregation) implementation for the ODY PHP framework.
Overview
ODY CQRS provides a clean separation between commands (that modify state) and queries (that return data) in your application. This implementation focuses on a synchronous approach for reliable and predictable execution of your domain operations.
Key features:
- Command Bus: Process state-changing operations
- Query Bus: Handle data retrieval operations
- Event Bus: Broadcast and handle domain events
- Middleware Support: Extend functionality with custom middleware
- Attribute-based Registration: Simple handler declaration with PHP 8 attributes
Installation
composer require ody/cqrs
Quick Start
1. Define Commands, Queries, and Events
<?php // Command to modify state namespace App\Commands; use Ody\CQRS\Message\Command; class CreateUserCommand extends Command { public function __construct( private string $name, private string $email, private string $password ) {} public function getName(): string { return $this->name; } public function getEmail(): string { return $this->email; } public function getPassword(): string { return $this->password; } } // Query to retrieve data namespace App\Queries; use Ody\CQRS\Message\Query; class GetUserById extends Query { public function __construct( private string $id ) {} public function getId(): string { return $this->id; } } // Event to notify about domain changes namespace App\Events; use Ody\CQRS\Message\Event; class UserWasCreated extends Event { public function __construct( private string $id ) {} public function getId(): string { return $this->id; } }
2. Create Handlers
<?php namespace App\Services; use App\Commands\CreateUserCommand; use App\Events\UserWasCreated; use App\Models\User; use App\Queries\GetUserById; use Ody\CQRS\Attributes\CommandHandler; use Ody\CQRS\Attributes\EventHandler; use Ody\CQRS\Attributes\QueryHandler; use Ody\CQRS\Interfaces\EventBusInterface; class UserService { #[CommandHandler] public function createUser(CreateUserCommand $command, EventBusInterface $eventBus) { $user = User::create([ 'name' => $command->getName(), 'email' => $command->getEmail(), 'password' => $command->getPassword(), ]); $eventBus->publish(new UserWasCreated($user->id)); } #[QueryHandler] public function getUserById(GetUserById $query) { return User::findOrFail($query->getId()); } #[EventHandler] public function when(UserWasCreated $event): void { logger()->info("User was created: " . $event->getId()); } }
3. Use in Controllers
<?php namespace App\Controllers; use App\Commands\CreateUserCommand; use App\Queries\GetUserById; use Ody\CQRS\Interfaces\CommandBusInterface; use Ody\CQRS\Interfaces\QueryBusInterface; use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; class UserController { public function __construct( private readonly CommandBusInterface $commandBus, private readonly QueryBusInterface $queryBus ) {} public function createUser(ServerRequestInterface $request, ResponseInterface $response): ResponseInterface { $data = $request->getParsedBody(); $this->commandBus->dispatch( new CreateUserCommand( name: $data['name'], email: $data['email'], password: $data['password'] ) ); return $response->json([ 'status' => 'success' ]); } public function getUser(ServerRequestInterface $request, ResponseInterface $response, array $args): ResponseInterface { $user = $this->queryBus->dispatch( new GetUserById( id: $args['id'] ) ); return $response->json($user); } }
4. Configure CQRS
Create or update config/cqrs.php
:
<?php return [ 'handler_paths' => [ app_path('Services'), ], ];
Advanced Configuration
The CQRS module includes several configuration options:
<?php return [ // Paths to scan for handlers 'handler_paths' => [ app_path('Services'), ], ];
Middleware
You can add custom middleware to each bus for tasks like logging, validation, or authentication.
Example: Command Bus Middleware
<?php namespace App\Middleware; use Ody\CQRS\Bus\Middleware\CommandBusMiddleware; class LoggingMiddleware extends CommandBusMiddleware { public function handle(object $command, callable $next): void { logger()->info('Handling command: ' . get_class($command)); // Execute the next middleware or the final handler $next($command); logger()->info('Command handled: ' . get_class($command)); } } // Register middleware $commandBus->addMiddleware(new LoggingMiddleware());
CQRS Middleware System
The CQRS middleware system allows you to intercept and modify the behavior of commands, queries, and events at various points in their lifecycle. This powerful feature enables cross-cutting concerns like logging, validation, authorization, and caching without modifying your core business logic.
Types of Middleware
There are four types of middleware:
- Before: Executes before the target method is called
- Around: Wraps the execution of the target method
- After: Executes after the target method returns successfully
- AfterThrowing: Executes when the target method throws an exception
Creating Middleware
Middleware classes are simple PHP classes with methods decorated with attribute annotations.
Example: Logging Middleware
namespace App\Middleware; use Ody\CQRS\Middleware\Before; use Ody\CQRS\Middleware\After; use Ody\CQRS\Middleware\AfterThrowing; class LoggingMiddleware { #[Before(pointcut: "Ody\\CQRS\\Bus\\CommandBus::executeHandler")] public function logBeforeCommand(object $command): void { logger()->info('Processing command: ' . get_class($command)); } #[After(pointcut: "Ody\\CQRS\\Bus\\QueryBus::executeHandler")] public function logAfterQuery(mixed $result, array $args): mixed { $query = $args[0] ?? null; if ($query) { logger()->info('Query processed: ' . get_class($query)); } return $result; } #[AfterThrowing(pointcut: "Ody\\CQRS\\Bus\\EventBus::executeHandlers")] public function logEventException(\Throwable $exception, array $args): void { $event = $args[0] ?? null; if ($event) { logger()->error('Error handling event: ' . get_class($event)); } } }
Example: Transactional Middleware
namespace App\Middleware; use Ody\CQRS\Middleware\Around; use Ody\CQRS\Middleware\MethodInvocation; class TransactionalMiddleware { public function __construct(private \PDO $connection) { } #[Around(pointcut: "Ody\\CQRS\\Bus\\CommandBus::executeHandler")] public function transactional(MethodInvocation $invocation): mixed { $this->connection->beginTransaction(); try { $result = $invocation->proceed(); $this->connection->commit(); return $result; } catch (\Throwable $exception) { $this->connection->rollBack(); throw $exception; } } }
Pointcut Expressions
Pointcut expressions determine which methods the middleware applies to. The syntax supports:
- Exact Class Match:
App\Services\UserService
- Namespace Wildcard:
App\Domain\*
- Method Match:
App\Services\UserService::createUser
- Any Method Wildcard:
App\Services\UserService::*
- Global Wildcard:
*
(matches everything) - Logical Operations:
App\Domain\* && !App\Domain\Internal\*
Examples
// Match any command in the Order namespace #[Before(pointcut: "App\\Commands\\Order\\*")] public function validateOrderCommand(object $command): void { } // Match a specific method in a specific class #[Around(pointcut: "App\\Services\\PaymentService::processPayment")] public function securePaymentProcessing(MethodInvocation $invocation): mixed { } // Match multiple patterns with logical OR #[After(pointcut: "App\\Domain\\User\\* || App\\Domain\\Account\\*")] public function auditUserChanges(mixed $result, array $args): mixed { }
Middleware Priority
You can control the order in which middleware executes by setting a priority. Lower values run first.
// Runs before other middleware #[Before(priority: 1, pointcut: "*")] public function highPriorityMiddleware(): void { } // Runs after middleware with lower priority values #[Before(priority: 100, pointcut: "*")] public function lowPriorityMiddleware(): void { }
Registering Middleware
Middleware is discovered and registered automatically from configured directories:
// config/cqrs.php return [ // ... 'middleware_paths' => [ app_path('Middleware'), ], // ... ];
Before Middleware
Before middleware runs before a method executes. It's useful for:
- Validation
- Authorization
- Parameter transformation
- Logging
#[Before(pointcut: "Ody\\CQRS\\Bus\\CommandBus::executeHandler")] public function validateCommand(object $command): void { // Validate the command $errors = $this->validator->validate($command); if (!empty($errors)) { throw new ValidationException($errors); } }
Around Middleware
Around middleware wraps a method execution. It's useful for:
- Transactions
- Timing measurements
- Caching
- Retry logic
#[Around(pointcut: "Ody\\CQRS\\Bus\\QueryBus::executeHandler")] public function cacheQueryResults(MethodInvocation $invocation): mixed { $args = $invocation->getArguments(); $query = $args[0]; $cacheKey = 'query:' . get_class($query) . ':' . md5(serialize($query)); if ($this->cache->has($cacheKey)) { return $this->cache->get($cacheKey); } $result = $invocation->proceed(); $this->cache->set($cacheKey, $result, 3600); return $result; }
After Middleware
After middleware runs after a method successfully returns. It's useful for:
- Result transformation
- Post-processing
- Logging
- Event publishing
#[After(pointcut: "Ody\\CQRS\\Bus\\QueryBus::executeHandler")] public function transformQueryResult(mixed $result, array $args): mixed { // Transform the result if (is_array($result)) { return array_map(function ($item) { return $this->transformer->transform($item); }, $result); } return $this->transformer->transform($result); }
AfterThrowing Middleware
AfterThrowing middleware runs when a method throws an exception. It's useful for:
- Exception handling
- Logging errors
- Fallback strategies
- Error notification
#[AfterThrowing(pointcut: "Ody\\CQRS\\Bus\\CommandBus::executeHandler")] public function handleCommandException(\Throwable $exception, array $args): void { $command = $args[0]; logger()->error('Command failed: ' . get_class($command), [ 'command' => $command, 'exception' => $exception->getMessage(), 'trace' => $exception->getTraceAsString() ]); // Notify monitoring system $this->alertService->sendAlert('Command failed: ' . get_class($command)); }
Performance Considerations
In the Swoole environment where ODY runs, middleware registration happens once at bootstrap time,
making the runtime overhead minimal. The middleware system is designed to be efficient:
- Cached Resolution: Pointcut expressions are evaluated once and cached
- Minimal Reflection: Heavy reflection work is done during bootstrap
- Optimized Invocation: Method invocation chains are built efficiently
Configuration Options
// config/cqrs.php return [ // ... // Paths to scan for middleware classes 'middleware_paths' => [ app_path('Middleware'), ], // Middleware configuration 'middleware' => [ // Global middleware applied to all buses 'global' => [ // Example: App\Middleware\LoggingMiddleware::class, ], // Command bus specific middleware 'command' => [ // Example: App\Middleware\TransactionalMiddleware::class, ], // Query bus specific middleware 'query' => [ // Example: App\Middleware\CachingMiddleware::class, ], // Event bus specific middleware 'event' => [ // Example: App\Middleware\AsyncEventMiddleware::class, ], ], // ... ];
Swoole Coroutines Integration
While the CQRS implementation itself is synchronous, you can still leverage Swoole's coroutines in your application when using this module:
- Multiple command and query handlers can still benefit from Swoole's coroutine scheduler
- I/O operations within handlers can take advantage of Swoole's non-blocking capabilities
- Your application remains responsive while handlers execute their logic
Best Practices
- Keep Commands and Queries Simple: They should be DTOs (Data Transfer Objects) without complex logic
- Single Responsibility: Each handler should handle one specific command or query
- Domain Events: Use events to notify about state changes, not to perform side effects
- Idempotency: Design command handlers to be idempotent (can be executed multiple times with the same result)
- Transactions: Use database transactions in command handlers to ensure atomicity
License
This project is licensed under the MIT License - see the LICENSE file for details.