bssd-ltd / laravel-aspect
Aspect Oriented Programming library for laravel framework, and lumen
Requires
- php: ^7.3|^8.0
- doctrine/annotations: ^1.11
- illuminate/bus: ^8.0
- illuminate/cache: ^8.0
- illuminate/config: ^8.0
- illuminate/console: ^8.0
- illuminate/contracts: ^8.0
- illuminate/database: ^8.0
- illuminate/events: ^8.0
- illuminate/filesystem: ^8.0
- illuminate/log: ^8.0
- illuminate/queue: ^8.0
- illuminate/support: ^8.0
- nikic/php-parser: ^4.0
- psr/log: ~1.0
- ray/aop: ^2.9
Requires (Dev)
- illuminate/encryption: ^8.0
- mockery/mockery: *
- pdepend/pdepend: ^2.2.4
- phploc/phploc: *
- phpmd/phpmd: @stable
- phpunit/phpunit: ^9.3.3
- roave/security-advisories: dev-latest
- satooshi/php-coveralls: *
- sebastian/phpcpd: *
- squizlabs/php_codesniffer: ~2.7
- symfony/console: ^5.2
- symfony/framework-bundle: ^5.2
- vlucas/phpdotenv: ^5.2
This package is auto-updated.
Last update: 2024-08-30 07:52:30 UTC
README
Origin source can be found here
added serviceProvider
'providers' => [ // added AspectServiceProvider \Bssd\LaravelAspect\AspectServiceProvider::class, // added Artisan Command \Bssd\LaravelAspect\ConsoleServiceProvider::class, ]
for Lumen
Add App\Providers\LumenAspectServiceProvider to your bootstrap/app.php file.
$app->register(\App\Providers\LumenAspectServiceProvider::class); $app->register(\Bssd\LaravelAspect\ConsoleServiceProvider::class);
publish aspect module class
$ php artisan ytake:aspect-module-publish
more command options [--help]
publish configure
- basic
$ php artisan vendor:publish
- use tag option
$ php artisan vendor:publish --tag=aspect
- use provider
$ php artisan vendor:publish --provider="Bssd\LaravelAspect\AspectServiceProvider"
register aspect module
config/ytake-laravel-aop.php
'modules' => [ // append modules // \App\Modules\CacheableModule::class, ],
use classes property
namespace App\Modules; use Bssd\LaravelAspect\Modules\CacheableModule as PackageCacheableModule; /** * Class CacheableModule */ class CacheableModule extends PackageCacheableModule { /** @var array */ protected $classes = [ \YourApplication\Services\SampleService::class ]; }
example
namespace YourApplication\Services; use Bssd\LaravelAspect\Annotation\Cacheable; class SampleService { /** * @Cacheable(cacheName="testing1",key={"#id"}) */ public function action($id) { return $this; } }
notice
- Must use a service container
- Classes must be non-final
- Methods must be public
for Lumen
override Bssd\LaravelAspect\AspectServiceProvider
use Bssd\LaravelAspect\AspectManager; use Bssd\LaravelAspect\AnnotationManager; use Bssd\LaravelAspect\AspectServiceProvider as AspectProvider; /** * Class AspectServiceProvider */ final class AspectServiceProvider extends AspectProvider { /** * {@inheritdoc} */ public function register() { $this->app->configure('ytake-laravel-aop'); $this->app->singleton('aspect.manager', function ($app) { $annotationConfiguration = new AnnotationConfiguration( $app['config']->get('ytake-laravel-aop.annotation') ); $annotationConfiguration->ignoredAnnotations(); // register annotation return new AspectManager($app); }); } }
bootstrap/app.php
$app->register(App\Providers\AspectServiceProvider::class); if ($app->runningInConsole()) { $app->register(Bssd\LaravelAspect\ConsoleServiceProvider::class); }
Cache Clear Command
$ php artisan ytake:aspect-clear-cache
PreCompile Command
$ php artisan ytake:aspect-compile
Annotations
@Transactional
for database transaction(illuminate/database)
you must use the TransactionalModule
- option
use Bssd\LaravelAspect\Annotation\Transactional; /** * @Transactional("master") */ public function save(array $params) { return $this->eloquent->save($params); }
Multiple Transaction
use Bssd\LaravelAspect\Annotation\Transactional; /** * @Transactional({"master", "second_master"}) */ public function save(array $params) { $this->eloquent->save($params); $this->query->save($params); }
Exception Rollback
use Bssd\LaravelAspect\Annotation\Transactional; /** * @Transactional(expect="\QueryException") */ public function save(array $params) { $this->eloquent->save($params); $this->query->save($params); }
Multiple Exception Rollback
use Bssd\LaravelAspect\Annotation\Transactional; /** * @Transactional(expect={"\QueryException", "\RuntimeException"}) */ public function save(array $params) { $this->eloquent->save($params); $this->query->save($params); }
@Cacheable
for cache(illuminate/cache)
you must use the CacheableModule
- option
use Bssd\LaravelAspect\Annotation\Cacheable; /** * @Cacheable(cacheName="testing1",key={"#id","#value"}) * @param $id * @param $value * @return mixed */ public function namedMultipleKey($id, $value) { return $id; }
@CacheEvict
for cache(illuminate/cache) / remove cache
you must use the CacheEvictModule
- option
use Bssd\LaravelAspect\Annotation\CacheEvict; /** * @CacheEvict(cacheName="testing",tags={"testing1"},allEntries=true) * @return null */ public function removeCache() { return null; }
@CachePut
for cache(illuminate/cache) / cache put
you must use the CachePutModule
- option
use Bssd\LaravelAspect\Annotation\CachePut; /** * @CachePut(cacheName={"testing1"},tags="testing1") */ public function throwExceptionCache() { return 'testing'; }
@Loggable / @LogExceptions
for logger(illuminate/log, monolog)
you must use the LoggableModule / LogExceptionsModule
- option
use Bssd\LaravelAspect\Annotation\Loggable; class AspectLoggable { /** * @Loggable * @param null $id * @return null */ public function normalLog($id = null) { return $id; } }
sample)
[2015-12-23 08:15:30] testing.INFO: Loggable:__Test\AspectLoggable.normalLog {"args":{"id":1},"result":1,"time":0.000259876251221}
About @LogExceptions
Also, take a look at @Loggable. This annotation does the same, but also logs non-exceptional situations.
use Bssd\LaravelAspect\Annotation\LogExceptions; class AspectLoggable { /** * @LogExceptions * @param null $id * @return null */ public function dispatchLogExceptions() { return $this->__toString(); } }
About @QueryLog
for database query logger(illuminate/log, monolog, illuminate/database)
use Bssd\LaravelAspect\Annotation\QueryLog; use Illuminate\Database\ConnectionResolverInterface; /** * Class AspectQueryLog */ class AspectQueryLog { /** @var ConnectionResolverInterface */ protected $db; /** * @param ConnectionResolverInterface $db */ public function __construct(ConnectionResolverInterface $db) { $this->db = $db; } /** * @QueryLog */ public function multipleDatabaseAppendRecord() { $this->db->connection()->statement('CREATE TABLE tests (test varchar(255) NOT NULL)'); $this->db->connection('testing_second')->statement('CREATE TABLE tests (test varchar(255) NOT NULL)'); $this->db->connection()->table("tests")->insert(['test' => 'testing']); $this->db->connection('testing_second')->table("tests")->insert(['test' => 'testing second']); } }
testing.INFO: QueryLog:AspectQueryLog.multipleDatabaseAppendRecord {"queries":[{"query":"CREATE TABLE tests (test varchar(255) NOT NULL)","bindings":[],"time":0.58,"connectionName":"testing"},{"query":"CREATE TABLE tests (test varchar(255) NOT NULL)","bindings":[],"time":0.31,"connectionName":"testing_second"} ...
@PostConstruct
The PostConstruct annotation is used on a method that needs to be executed after dependency injection is done to perform any initialization.
you must use the PostConstructModule
use Bssd\LaravelAspect\Annotation\PostConstruct; class Something { protected $abstract; protected $counter = 0; public function __construct(ExampleInterface $abstract) { $this->abstract = $abstract; } /** * @PostConstruct */ public function init() { $this->counter += 1; } /** * @return int */ public function returning() { return $this->counter; } }
The method MUST NOT have any parameters
@RetryOnFailure
Retry the method in case of exception.
you must use the RetryOnFailureModule.
- option
use Bssd\LaravelAspect\Annotation\RetryOnFailure; class ExampleRetryOnFailure { /** @var int */ public $counter = 0; /** * @RetryOnFailure( * types={ * LogicException::class, * }, * attempts=3, * ignore=Exception::class * ) */ public function ignoreException() { $this->counter += 1; throw new \Exception; } }
@MessageDriven
Annotation for a Message Queue(illuminate/queue. illuminate/bus).
you must use the MessageDrivenModule.
- option
use Bssd\LaravelAspect\Annotation\EagerQueue; use Bssd\LaravelAspect\Annotation\LazyQueue; use Bssd\LaravelAspect\Annotation\Loggable; use Bssd\LaravelAspect\Annotation\MessageDriven; /** * Class AspectMessageDriven */ class AspectMessageDriven { /** * @Loggable * @MessageDriven( * @LazyQueue(3), * onQueue="message" * ) * @return void */ public function exec($param) { echo $param; } /** * @MessageDriven( * @EagerQueue * ) * @param string $message */ public function eagerExec($message) { $this->logWith($message); } /** * @Loggable(name="Queued") * @param string $message * * @return string */ public function logWith($message) { return "Hello $message"; } }
LazyQueue
Handle Class Bssd\LaravelAspect\Queue\LazyMessage
EagerQueue
Handle Class Bssd\LaravelAspect\Queue\EagerMessage
Ignore Annotations
use config/ytake-laravel-aspect.php file
default: LaravelCollective/annotations
'annotation' => [ 'ignores' => [ // global Ignored Annotations 'Hears', 'Get', 'Post', 'Put', 'Patch', 'Options', 'Delete', 'Any', 'Middleware', 'Resource', 'Controller' ], ],
Append Custom Annotations
'annotation' => [ 'ignores' => [ // global Ignored Annotations 'Hears', 'Get', 'Post', 'Put', 'Patch', 'Options', 'Delete', 'Any', 'Middleware', 'Resource', 'Controller' ], 'custom' => [ \Acme\Annotations\Transactional::class // etc... ] ],
for testing
use none driver
<env name="ASPECT_DRIVER" value="none"/>