Всем привет! Меня зовут Иван Шишкин и я руковожу разработкой в агентстве Intensa.
В этой статье хотел бы поделиться методом автоматического подключения каналов логирования в Laravel через механизм сервис контейнеров(DI).
Логирование в Laravel
Фреймворк имеет на борту мощный функционал логирования, реализованный на базе библиотеки Monolog и для удобства клиентского кода скрыт за фасадом Log.
В этой статье я не буду углубляться в устройство и возможности функционала логирования, для этого есть прекрасная документация.
Определение проблемы
Обычно для сбора логов в классах мы используем следующую конструкцию:
<?php
namespace App\Http\Controllers;
class TaskController extends Controller
{
protected LoggerInterface $logger;
public function __construct()
{
$this->logger = Log::channel('task');
}
}
Сохраняем экземпляр LoggerInterface конкретного канала логирования в свойство класса.
Такой подход имеет место быть, если:
— у вас небольшой проект,
— логирование введено в класс временно,
— вы не используете DI в проекте.
В остальных случаях такая реализация даст сильную связанность ваших классов с конкретным каналом логирования. И, когда нам понадобиться изменить поведение узла логирования, например, поменять канал логирования, придётся прыгать по классам и переписывать код. При большом количестве классов в проекте это может вызвать у вас боль и страдания.
Автоматическое подключение
Перед тем как начать описание, еще раз подчеркну — в нашем мире мало магии и PHP не исключение. Для того, чтобы автоматическое подключение сработало, классы вашего проекта должны быть получены через DI.
Первым делом нужно создать интерфейс-маркер, к которому мы привяжем реализацию. Интерфейс должен наследовать от LoggerAwareInterface.
<?php
namespace App\Support\Logging;
use Psr\Log\LoggerAwareInterface;
interface DefaultLoggerAwareInterface extends LoggerAwareInterface
{
}
Теперь в провайдере свяжем данный интерфейс с нужным каналом логирования при помощи метода afterResolving
.
<?php
namespace App\Providers;
class AppServiceProvider extends ServiceProvider
{
public function register(): void
{
$this->app->afterResolving(
DefaultLoggerAwareInterface::class,
function (DefaultLoggerAwareInterface $aware) {
// Привязываем канал "common".
// Log::channel вернет экземпляр \Psr\Log\LoggerInterface
$aware->setLogger(Log::channel('common'));
}
);
}
}
Переходим к подключению нашего логгера к классу. Для демонстрации возьму контроллер из предыдущего блока.
<?php
namespace App\Http\Controllers;
use App\Support\Logging\DefaultLoggerAwareInterface;
class TaskController extends Controller implements DefaultLoggerAwareInterface
{
use LoggerAwareTrait;
}
Имплементируем интерфейс DefaultLoggerAwareInterface и подключаем трейт Psr\Log\LoggerAwareTrait, чтобы наш класс получил свойство $this->logger
.
Логируем нужные данные в нашем классе:
<?php
namespace App\Http\Controllers;
use App\Support\Logging\DefaultLoggerAwareInterface;
use Illuminate\Http\Request;
class TaskController extends Controller implements DefaultLoggerAwareInterface
{
use LoggerAwareTrait;
public function __invoke(Request $request)
{
//...
$this->logger->info('Данные запроса', $request->toArray());
}
}
Для безопасного вызова логгера я бы рекомендовал использовать Null-safe оператор перед вызовом метода логирования. Это исключит фатальные ошибки в вашем коде, в случае, если по какой-то причине логгер не был доставлен в ваш класс.
$this->logger?->info('Данные запроса', $request->toArray());
Без магии
Также есть вариант пробрасывать логгеры в ваши классы через автоматическое связывание(Autowiring).
Передаем аргумент LoggerInterface в метод вашего класса
<?php
namespace App\Http\Controllers;
use App\Support\Logging\DefaultLoggerAwareInterface;
use Illuminate\Http\Request;
class TaskController extends Controller
{
public function __invoke(Request $request, LoggerInterface $logger)
{
//...
$this->logger->info('Данные запроса', $request->toArray());
}
}
В AppServiceProvider определим правило для сервис контейнера. Через данную запись мы явно определяем, что если, класс TaskController запросил у DI экземпляр LoggerInterface, то нужно вернуть канал логирования с кодом ‘common’
<?php
namespace App\Providers;
class AppServiceProvider extends ServiceProvider
{
public function register(): void
{
$this->app->when(TaskController::class)
->needs(LoggerInterface::class)
->give(fn() => Log::channel('common'));
}
}
Эпилог
В завершение хочу отметить, что описанное в статье не является панацеей и правилом из категории “Right Way”. Я бы назвал это трюком, который поможет сделать ваш код менее связанным и поддерживаемым в перспективе.