Привет. Недавно получил в руки интересный проект, который, несмотря на свою простоту требовал не использовать какой-либо фреймворк. О пакетах речи не шло, поэтому было принято решение использовать привычные компоненты Laravel. Если есть необходимость в использовании очередей, Eloquent или контейнера — добро пожаловать под кат.
О простоте деления фреймворка на компоненты
Начиная с Laravel 4, все компоненты — отдельные пакеты, которые в теории можно заставить работать в любом PHP проекте. Однако, некоторые компоненты требуют дополнительной настройки, которая в основном фреймворке спрятана под капот.
Компоненты
Контейнер
Из всех рассматриваемых компонентов illuminate/container
— самый простой в установке и использовании.
<?php
use IlluminateContainerContainer;
$container = Container::getInstance();
return $container->make(Foo::class);
Впрочем, вызов статичного метода на классе Container
при каждом использовании контейнера — не самая лучшая идея. В фреймворке доступен хелпер app()
, который вернёт нам инстанс глобального контейнера, однако, нам такой нужно создать вручную.
Создадим файл helpers.php
для таких хелперов, и добавим его в автозагрузку.
"autoload": {
"files": [
"src/helpers.php"
],
"psr-4": {
"YourApp\": "src/"
}
}
Добавляем хелпер app()
в файл.
<?php
use IlluminateContainerContainer;
if (! function_exists('app')) {
/**
* Get the available container instance.
*
* @param string|null $abstract
* @param array $parameters
* @return mixed|IlluminateContainerContainer
*/
function app($abstract = null, array $parameters = [])
{
if (is_null($abstract)) {
return Container::getInstance();
}
return Container::getInstance()->make($abstract, $parameters);
}
}
Готово. Можно пробовать использовать хелпер.
<?php
return app(Foo::class);
Query Builder и Eloquent
Для использования компонента illuminate/database
мы будем использовать CapsuleManager
— класс, предназначенный для работы с построителем запросов вне Laravel.
Начнём с настройки подключения к БД. Желательно эту настройку проводить на этапе запуска вашего приложения, сразу после подключения autoload.php
.
<?php
require_once __DIR__.'/../vendor/autoload.php';
use IlluminateDatabaseCapsuleManager;
$manager = new Manager;
$manager->addConnection([
'driver' => 'mysql',
'host' => 'localhost',
'database' => 'database',
'username' => 'root',
'password' => 'password',
'charset' => 'utf8',
'collation' => 'utf8_unicode_ci',
'prefix' => '',
]);
// Позволяет использовать статичные вызовы при работе с Capsule.
$manager->setAsGlobal();
Можно начинать работу с построителем запросов.
<?php
use IlluminateDatabaseCapsuleManager;
return Manager::table('orders')->get();
Что насчёт Eloquent?
<?php
namespace YourAppModels;
use IlluminateDatabaseEloquentModel;
class Order extends Model {
protected $fillable = [
'name',
];
}
<?php
use YourAppModelsOrder;
Order::first();
С миграциями ситуация сложнее — с одной стороны, в комплекте есть Schema Builder. С другой — автоматический механизм запуска миграций отсутствует.
<?php
use IlluminateDatabaseCapsuleManager;
Manager::schema()->create('orders', function ($table) {
$table->bigIncrements('id');
$table->string('name');
$table->timestamps();
});
Для запуска достаточно выполнить файл с миграцией: php migration.php
Очереди
У очередей тоже есть свой Capsule
, однако аналоги queue:work
и queue:listen
необходимо писать вручную.
Начнём с класса Application
. В Laravel этот класс используется как глобальный инстанс контейнера, который помимо методов IlluminateContainerContainer
содержит вспомогательные методы для работы с фреймворком (текущая версия, пути к хранилищу, ресурсам). Однако, наш класс будет содержать лишь один метод — isDownForMaintenance
. Он необходим для работы воркера, так как определяет состояние работы приложения в данный момент.
<?php
namespace YourApp;
use IlluminateContainerContainer;
class Application extends Container {
public function isDownForMaintenance()
{
return false;
}
}
Далее, нам необходимо зарегистрировать провайдер illuminate/events
, и забиндить контракт IlluminateContractsEventsDispatcher
к алиасу events
.
<?php
use YourAppApplication;
use IlluminateContractsEventsDispatcher;
$application = Application::getInstance();
$application->bind(Dispatcher::class, 'events');
Теперь необходимо создать инстанс Capsule
, и добавить туда конфигурации соединений.
Пример конфигурации для БД
<?php
use YourAppApplication;
use IlluminateQueueCapsuleManager;
use IlluminateDatabaseCapsuleManager as DB;
use IlluminateDatabaseConnectionResolver;
$container = Application::getInstance();
$queue = new Manager($container);
$queue->addConnection([
'driver' => 'database',
'table' => 'jobs',
'connection' => 'default',
'queue' => 'default',
], 'default');
// Также, как и с IlluminateDatabaseCapsuleManager позволяет использовать статичные вызовы для очередей
$queue->setAsGlobal();
$connection = Capsule::schema()->getConnection();
$resolver = new ConnectionResolver(['default' => $connection]);
$queue->getQueueManager()->addConnector('database', function () use ($resolver) {
return new DatabaseConnector($resolver);
});
Пример конфигурации для Redis (требует установленного illuminate/redis
)
<?php
use IlluminateRedisRedisManager;
use IlluminateQueueCapsuleManager;
$container->bind('redis', function () use ($container) {
return new RedisManager($container, 'predis', [
'default' => [
'host' => '127.0.0.1',
'password' => null,
'port' => 6379,
'database' => 0,
]
]);
});
$queue = new Manager($container);
$queue->addConnection([
'driver' => 'redis',
'connection' => 'default',
'queue' => 'default',
], 'default');
$queue->setAsGlobal();
Последний этап в конфигурации — добавление аналога Exception Handler.
<?php
use IlluminateContractsDebugExceptionHandler;
class Handler implements ExceptionHandler {
public function report(Exception $e)
{
// Действие, если Exception подпадает под критерии об уведомлении.
// Пример: отправка в Sentry, отправка сообщения на почту.
}
public function render($request, Exception $e)
{
// Отображение ошибки
}
public function renderForConsole($output, Exception $e)
{
// Отображение ошибки, если среда запуска приложения - терминал
}
public function shouldReport(Exception $e)
{
// Необходимо ли уведомлять об ошибке
}
}
app()->bind('exception.handler', function () {
return new Handler;
});
Конфигурация очередей завершена. Теперь можно приступать к написанию queue:work
.
<?php
require_once __DIR__.'/bootstrap/bootstrap.php';
use IlluminateQueueWorker;
use IlluminateQueueCapsuleManager;
use IlluminateQueueWorkerOptions;
$queueManager = Manager::getQueueManager();
$worker = new Worker($queueManager, app('events'), app('exception.handler'));
$worker->daemon('default', 'default', new WorkerOptions);
Теперь для запуска воркера очередей достаточно написать php worker.php
.
Если же есть необходимость в использовании queue:listen
, то нужно создавать два отдельных файла. Один — демон, который слушает очередь, и запускает второй файл на каждый новый job. Второй файл, в свою очередь, выступает в роли исполнителя задачи.
worker.php
<?php
require_once __DIR__.'/bootstrap/bootstrap.php';
use IlluminateQueueListener;
use IlluminateQueueListenerOptions;
// По умолчанию, в качестве "второго файла", в Laravel выступает artisan, однако в нашем случае это будет файл work.php.
// https://github.com/laravel/framework/blob/6.x/src/Illuminate/Queue/Listener.php#L72
define('ARTISAN_BINARY', 'work.php');
$worker = app(Listener::class, ['commandPath' => __DIR__]);
$worker->setOutputHandler(function ($type, $line) {
echo $line;
});
$worker->listen('default', 'default', new ListenerOptions);
work.php
<?php
require_once 'bootstrap/bootstrap.php';
use IlluminateQueueWorker;
use IlluminateQueueWorkerOptions;
use IlluminateQueueCapsuleManager;
$queueManager = Manager::getQueueManager();
$worker = new Worker($queueManager, app('events'), app('exception.handler'));
$worker->runNextJob('default', 'default', new WorkerOptions);
Переходим к использованию. Все методы можно просмотреть в API
<?php
use IlluminateQueueCapsuleManager;
Manager::push(SomeJob::class);
Автор: ivan77011