Адаптер для promphp/prometheus_client_php
Добавьте пакет в приложение
composer require ensi/laravel-prometheus
Скопируйте конфигурацию для дальнейшей настройки
php artisan vendor:publish --tag=prometheus-config
Перед тем как накручивать счётчики метрик, их надо зарегистрировать. Лучше всего это делать в методе boot() в AppServiceProvider.
# app/Providers/AppServiceProvider.php
public function boot() {
Prometheus::counter('http_requests_count')->labels(['endpoint', 'code']);
Prometheus::summary('http_requests_duration_seconds', 60, [0.5, 0.95, 0.99]);
}
Обновить значение счётчика так же просто
# app/Http/Middleware/Telemetry.php
public function handle($request, Closure $next)
{
$startTime = microtime(true);
$response = $next($request);
$endTime = microtime(true);
Prometheus::update('http_requests_count', 1, [Route::current()?->uri, $response->status()]);
Prometheus::update('http_requests_duration_seconds', $endTime - $startTime);
return $response;
}
Структура файла конфигурации
# config/prometheus.php
return [
'default_bag' => '<bag-name>',
'enabled' => env('PROMETHEUS_ENABLED', true),
'app_name' => env('PROMETHEUS_APP_NAME', env('APP_NAME')),
'bags' => [
'<bag-name>' => [
'namespace' => '<prometheus-namespace>',
'route' => '<path-of-scrape-endpoint>',
'basic_auth' => [
'login' => env('PROMETHEUS_AUTH_LOGIN'),
'password' => env('PROMETHEUS_AUTH_PASSWORD'),
],
'<storage-type>' => [
'<connection-parameters>'
],
'label_middlewares' => [
'<middleware-class>'
],
'on_demand_metrics' => [
'<on-demand-metric-class>'
]
],
],
];
Bag
Вы можете захотеть иметь несколько наборов метрик, например один набор с техническими метриками, вроде количества http запросов или непойманных исключений, и второй для бизнес-значений вроде количества заказов или показов определённой страницы. Для этого вводится понятие bag. Вы можете настроить несколько бэгов, указав для каждого своё хранилище данных, отдельный эндпоинт для сбора метрик и т.д.
Storage type
Вы можете использовать все хранилища (Adapters) из пакета promphp/prometheus_client_php. Кроме того вы можете указать имя
redis connection'a из файла config/databases.php
.
Варианты настройки хранилища.
Хранить метрики в памяти процесса.
'memory' => true
Использовать APCu
'apcu' => [
'prefix' => 'metrics'
]
или альтернативный адаптер APCuNG
'apcu-ng' => [
'prefix' => 'metrics'
]
Redis адаптер, который сам создаст phpredis соединение
'redis' => [
'host' => '127.0.0.1',
'port' => 6379,
'timeout' => 0.1,
'read_timeout' => '10',
'persistent_connections' => false,
'password' => null,
'prefix' => 'my-app',
'bag' => 'my-metrics-bag'
]
Laravel Redis соединение из config/databases.php
. Под капотом будет создан тот же Redis адаптер,
но он возьмёт нативный объект соединения phpredis из RedisManager'a ларавеля.
'connection' => [
'connection' => 'default',
'bag' => 'default',
]
Выбрать другой bag для создания и обновления в нём метрик можно через метод bag($bagName)
.
# app/Providers/AppServiceProvider.php
public function boot() {
// создаём метрики в конкретном bag
Prometheus::bag('business')->counter('orders_count')->labels(['delivery_type', 'payment_method'])
}
# app/Actions/CreateOrder.php
public function execute(Order $order) {
// ...
Prometheus::bag('business')->update('orders_count', 1, [$order->delivery_type, $order->payment_method]);
}
Вы можете добавить лейбл ко всем метрикам bag'a указав в его конфигурации т.н. Label middleware. Label middleware срабатывает в момент определения метрики и в момент обновления её счётчика, добавляя в первом случае на название лейбла, а во втором значение.
Например у намс есть TenantLabelProvider
class TenantLabelMiddleware implements LabelMiddleware
{
public function labels(): array
{
return ['tenant'];
}
public function values(): array
{
return [Tenant::curent()->id];
}
}
Регистрируем его в конфигурации bag'a.
# config/prometheus.php
return [
// ...
'bags' => [
'default' => [
// ...
'label_middlewares' => [
\App\System\TenantLabelMiddleware::class,
]
],
],
];
Далее как обычно работаем с метриками.
Prometheus::counter('http_requests_count')->labels(['endpoint', 'code']);
// ...
Prometheus::update('http_requests_count', 1, [Route::current()?->uri, $response->status()]);
В результате метрика будет иметь не два, а три лейбла
app_http_requests_count{endpoint="catalog/products",code="200",tenant="JBZ-987-H6"} 987
Иногда метрики не привязаны к событиям приложения. Обычно это метрики типа gauge, которые нет смысла обновлять на каждом входящем запросе, т.к. прометеус всё-равно заберёт только последнее установленное значение. Такие метрики можно рассчитывать в момент сбора метрик прометеусом. Для этого вам нужно создать т.н. on demand метрику. Это класс, в котором вы регистрируете метрики и устанавливаете в них значения.
class QueueLengthOnDemandMetric extends OnDemandMetric {
public function register(MetricsBag $metricsBag): void
{
$metricsBag->gauge('queue_length');
}
public function update(MetricsBag $metricsBag): void
{
$metricsBag->update('queue_length', Queue::size());
}
}
Обновление таких метрик происходит в момент обращения прометеуса к эндпоинту получения метрик.
Laravel Prometheus is open-sourced software licensed under the MIT license.