Implements Clean Architecture as described by Robert C. Martin (Uncle Bob) here:
Clean Architecture
This is an opinionated package that defines levels of the architecture and the dependencies between them.
And uses deptrac to check if the levels are respected.
You can install the package via composer:
composer require giacomomasseron/php-clean-architectureAfter the installation, you must run the install command to publish the deptrac.yaml, php-clean-architecture.yaml and rector.php config files to your root folder:
vendor/bin/php-clean-architecture installTo check the architecture levels in your project, use the following command:
vendor/bin/php-clean-architecture checkAll make commands read the php-clean-architecture.yaml config file to know where to put the files created and which namespace they belong to.
The install command publishes needed files your root folder.
If you have already rector installed, you must add these lines of code in your rector.php file:
return RectorConfig::configure() // This line is already present in your rector.php file
->withConfiguredRule(
\GiacomoMasseroni\PHPCleanArchitecture\Rector\Rules\AddPHPCleanArchitectureInterfaceControllerToClassesRector::class,
[
'targetNamespaces' => [
'App\Http\Controllers',
]
]
)
->withConfiguredRule(
\GiacomoMasseroni\PHPCleanArchitecture\Rector\Rules\AddPHPCleanArchitectureInterfaceEntityToClassesRector::class,
[
'targetNamespaces' => [
'App\Entities',
]
]
)
->withConfiguredRule(
\GiacomoMasseroni\PHPCleanArchitecture\Rector\Rules\AddPHPCleanArchitectureInterfaceRepositoryToClassesRector::class,
[
'targetNamespaces' => [
'App\Repositories',
]
]
)
->withConfiguredRule(
\GiacomoMasseroni\PHPCleanArchitecture\Rector\Rules\AddPHPCleanArchitectureInterfaceServiceToClassesRector::class,
[
'targetNamespaces' => [
'App\Services',
]
]
)
->withConfiguredRule(
\GiacomoMasseroni\PHPCleanArchitecture\Rector\Rules\AddPHPCleanArchitectureInterfaceUseCaseToClassesRector::class,
[
'targetNamespaces' => [
'App\UseCases',
]
]
)The check command checks the architecture levels in your project.
You can use it in your CI/CD pipeline to be sure that the architecture levels are respected.
The rector command adds the needed interfaces to your classes based on namespaces defined in your config files.
To create an Entity, you can use the following command:
vendor/bin/php-clean-architecture make:entity EntityInYourProjectTo create a Repository, you can use the following command:
vendor/bin/php-clean-architecture make:repository UserRepositoryTo create a UseCase, you can use the following command:
vendor/bin/php-clean-architecture make:usecase DoSomethingUseCaseTo create a Controller, you can use the following command:
vendor/bin/php-clean-architecture make:controller UserControllerTo create a Service, you can use the following command:
vendor/bin/php-clean-architecture make:service ThirdyPartyService
vendor/bin/php-clean-architecture make:service ThirdyPartyIn both cases, the created class will be named ThirdyPartyService.
Why not?
It is a well-known, well-structured architecture system.
In the Clean Architecture, a level is a layer of the architecture with a specific function, only connected to the upper level.
The rule of thumb of the Clean Architecture is:
An inner circle must never know anything about the circles around it.
UseCase is a concept of Use Cases level.
A UseCase is every action your project performs.
Good examples of use cases are:
- Login
- Register
- CompleteOrder
- UpdateProfile
A use case should be a single, very specific action. It shouldn’t do anything more than its name suggests.
The package uses deptrac to define the levels and to check the dependencies between them.
These are the levels defined:
- Entity
- Repository
- UseCase
- Controller
- Service
These are the dependencies between the levels:
graph TD;
Controller-->UseCase-->Repository; UseCase-->Service; Repository-->Service; Repository-->Entity;
The Entity level must not depend on any other level.
The Repository level can only depend on Entity or Service levels.
The UseCase level can only depend on Repository or Service levels.
The Controller level can only depend on UseCase levels.
What is the Service level?
The Service level can be used for third-party tools or libraries.
The package comes with these interfaces:
- EntityInterface: implement this interface if the class belongs to the Entity level.
- RepositoryInterface: implement this interface if the class belongs to the Repository level.
- UseCaseInterface: implement this interface if the class belongs to the UseCase level.
- ControllerInterface: implement this interface if the class belongs to the Controller level.
- ServiceInterface: implement this interface if the class belongs to the Service level.
If you want your controller to be part of the Controller level, you need to implement the ControllerInterface.
For example:
use GiacomoMasseroni\PHPCleanArchitecture\Contracts\ControllerInterface;
public class YourController implements ControllerInterface When you create a UseCase, the class needs to extend the BaseUseCase class, and you need to implement the UseCaseInterface.
For example:
use GiacomoMasseroni\PHPCleanArchitecture\BaseUseCase;
use GiacomoMasseroni\PHPCleanArchitecture\Contracts\UseCaseInterface;
public class DoSomething extends BaseUseCase implements UseCaseInterface
{
public function handle(...$arguments): mixed
{
//
}
} To execute the UseCase, you need to call the run method defined in the BaseUseCase class:
DoSomething::run($arg1, $arg2); The package defines an abstract class for use cases: BaseUseCase.
This class defines variable for the user executing the use case:
UseCaseExecutorInterface $executorYou can set the executor using the following example:
DoSomething::actingAs($user)->run($arg1, $arg2); If you need to rollback the use case, you can override the rollback method:
public function rollback(): void
{
} The package dispatches two events, one when the use case starts, and one when the use case ends.
It uses the Symfony Event Dispatcher Component.
The events are:
- UseCaseStartedEvent
- UseCaseCompletedEvent
If you want to check the architecture levels in your CI/CD pipeline, you can use the following command:
vendor/bin/php-clean-architecture checkThis command will stop your pipeline if there are architecture violations, based on the deptrac configuration file.
composer testI wrote an article on how to use this package with Laravel
Laravel pulse is a package that helps you to monitor the health of your Laravel application.
If you want to monitor use cases execution time, you can:
- Create two laravel events,
UseCaseStartedandUseCaseCompleted - Create a Pulse card that listens to these events and measures the execution time.
- Add this code to
registerfunction inAppServiceProvider.phpfile:\GiacomoMasseroni\PHPCleanArchitecture\Dispatcher::getInstance()->addListener(\GiacomoMasseroni\PHPCleanArchitecture\Events\UseCaseStartedEvent::class, function (\GiacomoMasseroni\PHPCleanArchitecture\Events\UseCaseStartedEvent $event): void { // Just propagate the event to Laravel event system \App\Events\UseCaseStarted::dispatch($event->useCase); }); \GiacomoMasseroni\PHPCleanArchitecture\Dispatcher::getInstance()->addListener(\GiacomoMasseroni\PHPCleanArchitecture\Events\UseCaseCompletedEvent::class, function (\GiacomoMasseroni\PHPCleanArchitecture\Events\UseCaseCompletedEvent $event): void { // Just propagate the event to Laravel event system \App\Events\UseCaseCompleted::dispatch($event->useCase); });
Please see CHANGELOG for more information on what has changed recently.
Please see CONTRIBUTING for details.
Please review our security policy on how to report security vulnerabilities.
The MIT License (MIT). Please see License File for more information.