События
- Введение
- Регистрация событий и слушателей
- Определение событий
- Определение слушателей
- Слушатели события в очереди
- Размещение событий
- Подписчики событий
Введение
События в Laravel представлены реализацией паттерна Observer, что позволяет вам подписываться и прослушивать некие события вашего приложения. Как правило, классы событий находятся в директории app/Events
, а классы обработчиков, слушателей событий — в app/Listeners
. Не волнуйтесь, если не видите эти директории в своем приложении, так как они будут созданы для вас по мере того, как вы сгенерируете события и слушателей, используя artisan-команды.
События служат отличным способом отделить различные аспекты вашего приложения, поскольку одно событие может иметь несколько слушателей, которые не зависят друг от друга. Например, вы можете отправлять Slack-уведомление своему пользователю каждый раз, когда отправляется заказ. Вместо соединения своего кода обработки заказов со своим кодом Slack-уведомлений, вы можете просто отправить событие OrderShipped
, которое может получить слушатель и трансформировать в Slack-уведомление.
Регистрация событий и слушателей
Сервис-провайдер EventServiceProvider
, включённый в ваше Laravel приложение, предоставляет удобное место для регистрации всех слушателей событий. Свойство listen
содержит массив всех событий (ключей) и их слушателей (значения). Вы можете добавить столько событий в этот массив, сколько требуется вашему приложению. Например, давайте добавим событие OrderShipped
:
/**
* The event listener mappings for the application.
*
* @var array
*/
protected $listen = [
'App\Events\OrderShipped' => [
'App\Listeners\SendShipmentNotification',
],
];
Генерация событий и слушателей
Конечно, вручную создавать файлы для каждого события и слушателя - затруднительно. Вместо этого добавьте слушателей и события в ваш EventServiceProvider
и используйте команду event:generate
. Эта команда сгенерирует все события и слушателей, которые перечислены в вашем EventServiceProvider
. Уже существующие события и слушатели останутся нетронутыми:
php artisan event:generate
Регистрация событий вручную
Как правило, события должны регистрироваться через массив $listen
в EventServiceProvider
; Однако, также вы можете регистрировать события вручную с обработчиком событий, используя метод boot
вашего EventServiceProvider
:
/**
* Регистрация своих событий в приложении.
*
* @return void
*/
public function boot()
{
parent::boot();
Event::listen('event.name', function ($foo, $bar) {
//
});
}
Слушатели событий по маске
Вы даже можете регистрировать слушателей, используя символ *
, что позволит вам поймать несколько событий для одного слушателя. Такой метод вернёт весь массив данных событий одним параметром:
Event::listen('event.*', function ($eventName, array $data) {
//
});
Автоматическое обнаружение событий
Вместо регистрации событий и слушателей вручную в массиве $listen
EventServiceProvider
можно включить автоматическое обнаружение событий. Когда функция обнаружения событий включена, Laravel автоматически найдет и зарегистрирует ваши события и слушателей, сканируя каталог Listeners
вашего приложения. Все явно определенные события, перечисленные в EventServiceProvider
, будут по-прежнему регистрироваться.
Laravel находит слушателей событий, сканируя классы слушателей при помощи механизма PHP, известного как Reflection. Когда Laravel находит метод класса слушателя, который начинается с handle
, то регистрирует его как слушателя события, которое фигурирует в аргументе этого метода:
use App\Events\PodcastProcessed;
class SendPodcastProcessedNotification
{
/**
* Handle the given event.
*
* @param \App\Events\PodcastProcessed
* @return void
*/
public function handle(PodcastProcessed $event)
{
//
}
}
Автообнаружение по умолчанию отключено, чтобы включить его - переопределите метод shouldDiscoverEvents
в EventServiceProvider
:
/**
* Determine if events and listeners should be automatically discovered.
*
* @return bool
*/
public function shouldDiscoverEvents()
{
return true;
}
По умолчанию сканируется директория Listeners. Для изменения директории переопределите метод discoverEventsWithin
в EventServiceProvider
:
/**
* Get the listener directories that should be used to discover events.
*
* @return array
*/
protected function discoverEventsWithin()
{
return [
$this->app->path('Listeners'),
];
}
На сервере, в продакшне вы скорее всего не захотите, чтобы фреймворк сканировал всех ваших слушателей при каждом запросе. Поэтому, во время процесса установки приложения на сервер, вы должны запустить artisan-команду event:cache
, чтобы создать кэш-манифест всех событий и слушателей вашего приложения. Этот манифест будет использоваться фреймворком для ускорения процесса регистрации событий. Команда event:clear
удаляет этот кэш.
Чтобы увидеть все события и слушателей, зарегистрированных в вашем приложении, используйте artisan-команду
event:list
Определение событий
Класс события — это контейнер данных, содержащий информацию, которая относится к событию. Например, предположим, что наше сгенерированное событие OrderShipped
принимает объект Eloquent ORM:
<?php
namespace App\Events;
use App\Order;
use Illuminate\Queue\SerializesModels;
class OrderShipped
{
use SerializesModels;
public $order;
/**
* Создать новый экземпляр события.
*
* @param \App\Order $order
* @return void
*/
public function __construct(Order $order)
{
$this->order = $order;
}
}
Как видите, этот класс события не содержит логики. Это просто контейнер для объекта Order
, заказа, который был приобретён пользователем. Трейт SerializesModels
, необходим, чтобы корректно сериализовать (представлять в виде строки) Eloquent-модели.
Определение слушателей
Теперь давайте взглянем на класс слушателя. Слушатели событий принимают экземпляр события в свой метод handle
. Команда event:generate
автоматически импортирует класс события и указывает тип события в метод handle
. В методе handle
вы можете выполнять любую логику, необходимую для ответа на событие:
<?php
namespace App\Listeners;
use App\Events\OrderShipped;
class SendShipmentNotification
{
/**
* Создать слушатель события.
*
* @return void
*/
public function __construct()
{
//
}
/**
* Обработка события.
*
* @param \App\Events\OrderShipped $event
* @return void
*/
public function handle(OrderShipped $event)
{
// Заказ можно получить так: $event->order
}
}
Ваши слушатели событий могут также принимать в аргументах конструкторов любые нужные зависимости. Все слушатели события создаются через сервис-контейнер Laravel, поэтому зависимости будут внедрены в аргументы автоматически.
Остановка распространения события
Иногда, вам необходимо остановить распространение события для других слушателей. Вы можете сделать это, возвратив false
из метода handle
вашего слушателя.
Слушатели события в очереди
Добавление слушателей в очередь может быть полезным, если ваш слушатель собирается выполнить медленную задачу, такую как отправка e-mail или совершение HTTP-запроса. Перед началом работы со слушателями в очереди убедитесь, что настроили свою очередь и запустите обработчик очереди на своем сервере или в локальной среде разработки.
Чтобы указать, что слушатель должен быть помещен в очередь, добавьте интерфейс ShouldQueue
к классу слушателя. Слушателям, сгенерированным artisan-командой event:generate
, уже импортирован этот интерфейс в текущее пространство имен. Так что вы можете сразу использовать его:
<?php
namespace App\Listeners;
use App\Events\OrderShipped;
use Illuminate\Contracts\Queue\ShouldQueue;
class SendShipmentNotification implements ShouldQueue
{
//
}
Вот и всё! Теперь, когда этого слушателя вызывают для события, он будет автоматически поставлен в очередь диспетчером события, использующим систему очереди Laravel. Если никакие исключения не будут брошены, когда слушатель выполняется из очереди, то задача в очереди будет автоматически удалена после своего выполнения.
Настройка подключения очереди и названия очереди
Если вы бы хотели настроить подключение очереди и название очереди, используемые слушателем события, вы можете задать свойства $connection
, $queue
или $delay
в классе своего слушателя:
<?php
namespace App\Listeners;
use App\Events\OrderShipped;
use Illuminate\Contracts\Queue\ShouldQueue;
class SendShipmentNotification implements ShouldQueue
{
/**
* Имя соединения очереди, тип очереди
*
* @var string|null
*/
public $connection = 'sqs';
/**
* Имя очереди
*
* @var string|null
*/
public $queue = 'listeners';
/**
* Время в секундах, которое Laravel будет ждать прежде чем приступить к выполнению
*
* @var int
*/
public $delay = 60;
}
Выборочная отправка в очередь
Иногда вам может понадобиться определить непосредственно во время выполнения, следует ли ставить слушателя в очередь, или его можно выполнять синхронно. Для этого добавьте к слушателю метод shouldQueue
, в котором верните true
, если событие должно обработаться в очереди:
<?php
namespace App\Listeners;
use App\Events\OrderPlaced;
use Illuminate\Contracts\Queue\ShouldQueue;
class RewardGiftCard implements ShouldQueue
{
/**
* Reward a gift card to the customer.
*
* @param \App\Events\OrderPlaced $event
* @return void
*/
public function handle(OrderPlaced $event)
{
//
}
/**
* Determine whether the listener should be queued.
*
* @param \App\Events\OrderPlaced $event
* @return bool
*/
public function shouldQueue(OrderPlaced $event)
{
return $event->order->subtotal >= 5000;
}
}
Ручной доступ к очереди
Если вам необходимо получить доступ к базовым методам очереди delete
и release
вручную, вы можете сделать так. Трейт Illuminate\Queue\InteractsWithQueue
, который импортирован по умолчанию в сгенерированные слушатели, предоставляет вам доступ к этим методам:
<?php
namespace App\Listeners;
use App\Events\OrderShipped;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Queue\InteractsWithQueue;
class SendShipmentNotification implements ShouldQueue
{
use InteractsWithQueue;
/**
* Handle the event.
*
* @param \App\Events\OrderShipped $event
* @return void
*/
public function handle(OrderShipped $event)
{
if (true) {
$this->release(30);
}
}
}
Обработка проваленных задач
Иногда работа ваших слушателей в очереди может завершиться неудачей. Если слушатель в очередь преодолевает максимальное количество попыток, как указано в воркере очереди, будет вызван метод метод failed
на вашем слушателе. Этот метод получает экземпляр события и исключение, которое стало причиной провала задачи:
<?php
namespace App\Listeners;
use App\Events\OrderShipped;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Queue\InteractsWithQueue;
class SendShipmentNotification implements ShouldQueue
{
use InteractsWithQueue;
/**
* Handle the event.
*
* @param \App\Events\OrderShipped $event
* @return void
*/
public function handle(OrderShipped $event)
{
//
}
/**
* Handle a job failure.
*
* @param \App\Events\OrderShipped $event
* @param \Exception $exception
* @return void
*/
public function failed(OrderShipped $event, $exception)
{
//
}
}
Размещение событий
Чтобы разместить событие можно передать экземпляр события хелперу event
. Этот хелпер разместит событие всем его зарегистрированным слушателям. Так как хелпер event
доступен глобально, его можно вызвать из любого места в вашем приложении:
<?php
namespace App\Http\Controllers;
use App\Events\OrderShipped;
use App\Http\Controllers\Controller;
use App\Order;
class OrderController extends Controller
{
/**
* Отправить заданный заказ.
*
* @param int $orderId
* @return Response
*/
public function ship($orderId)
{
$order = Order::findOrFail($orderId);
// Order shipment logic...
event(new OrderShipped($order));
}
}
Во время тестирования может быть полезно утверждать (assert), что определенные события были размещены без фактического вызова их слушателей. Встроенные хелперы тестирования Laravel делают эту задачу чрезвычайно легкой.
Подписчики событий
Записывание подписчиков событий
Подписчики событий — это классы, которые могут подписаться на множество событий из самого класса, что позволяет вам определить несколько обработчиков событий в одном классе. Подписчики должны определить метод subscribe
, в который будет передан экземпляр диспетчера события:
<?php
namespace App\Listeners;
class UserEventSubscriber
{
/**
* Обработка событий входа пользователей в систему.
*/
public function handleUserLogin($event) {}
/**
* Обработка событий выхода пользователей из системы.
*/
public function handleUserLogout($event) {}
/**
* Register the listeners for the subscriber.
*
* @param \Illuminate\Events\Dispatcher $events
*/
public function subscribe($events)
{
$events->listen(
'Illuminate\Auth\Events\Login',
'App\Listeners\UserEventSubscriber@handleUserLogin'
);
$events->listen(
'Illuminate\Auth\Events\Logout',
'App\Listeners\UserEventSubscriber@handleUserLogout'
);
}
}
Регистрация подписчиков событий
Когда подписчик определён, его надо зарегистрировать в диспетчере события. Вы можете зарегистрировать подписчиков, используя свойство $subscribe
в EventServiceProvider
. Например, давайте добавим UserEventSubscriber
в список:
<?php
namespace App\Providers;
use Illuminate\Foundation\Support\Providers\EventServiceProvider as ServiceProvider;
class EventServiceProvider extends ServiceProvider
{
/**
* The event listener mappings for the application.
*
* @var array
*/
protected $listen = [
//
];
/**
* The subscriber classes to register.
*
* @var array
*/
protected $subscribe = [
'App\Listeners\UserEventSubscriber',
];
}