Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(common/es): log error request/response #681

Merged
merged 4 commits into from
Dec 20, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
22 changes: 14 additions & 8 deletions EMS/common-bundle/src/DataCollector/ElasticaDataCollector.php
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,10 @@

class ElasticaDataCollector extends DataCollector
{
public function __construct(private readonly ElasticaLogger $logger, private readonly ElasticaService $elasticaService)
{
public function __construct(
private readonly ElasticaLogger $logger,
private readonly ElasticaService $elasticaService
) {
}

public function collect(Request $request, Response $response, ?\Throwable $exception = null): void
Expand All @@ -34,18 +36,15 @@ public function getVersion(): string
return $this->data['version'];
}

/**
* @return mixed
*/
public function getQueryCount()
public function getQueryCount(): int
{
return $this->data['nb_queries'];
}

/**
* @return mixed
* @return array<mixed>
*/
public function getQueries()
public function getQueries(): array
{
return $this->data['queries'];
}
Expand All @@ -60,6 +59,13 @@ public function getTime(): int
return $time;
}

public function countErrors(): int
{
$queries = $this->data['queries'] ?? [];

return \count(\array_filter($queries, static fn (array $q) => isset($q['exception'])));
}

public function getExecutionTime(): float
{
$time = 0;
Expand Down
46 changes: 12 additions & 34 deletions EMS/common-bundle/src/Elasticsearch/Client.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,8 @@
namespace EMS\CommonBundle\Elasticsearch;

use Elastica\Client as BaseClient;
use Elastica\Connection;
use Elastica\Exception\ClientException;
use Elastica\Exception\ResponseException;
use Elastica\Request;
use Elastica\Response;
use Symfony\Component\Stopwatch\Stopwatch;
Expand All @@ -24,11 +24,14 @@ class Client extends BaseClient
*/
public function request($path, $method = Request::GET, $data = [], array $query = [], $contentType = Request::DEFAULT_CONTENT_TYPE): Response
{
if (null !== $this->stopwatch) {
$this->stopwatch->start('es_request', 'fos_elastica');
}
$this->stopwatch?->start('es_request', 'fos_elastica');

$response = parent::request($path, $method, $data, $query, $contentType);
try {
$response = parent::request($path, $method, $data, $query, $contentType);
} catch (ResponseException $e) {
$this->getLogger()?->logResponse($e->getResponse(), $e->getRequest(), $e);
throw $e;
}
$responseData = $response->getData();

$transportInfo = $response->getTransferInfo();
Expand All @@ -44,11 +47,8 @@ public function request($path, $method = Request::GET, $data = [], array $query
throw new ClientException($message);
}

$this->logQuery($response, $connection, $path, $method, $query, $data);

if ($this->stopwatch) {
$this->stopwatch->stop('es_request');
}
$this->getLogger()?->logResponse($response, $lastRequest);
$this->stopwatch?->stop('es_request');

return $response;
}
Expand All @@ -58,30 +58,8 @@ public function setStopwatch(?Stopwatch $stopwatch = null): void
$this->stopwatch = $stopwatch;
}

/**
* @param array<mixed>|string $data
* @param array<mixed> $query
*/
private function logQuery(Response $elasticaResponse, Connection $connection, string $path, string $method, array $query, array|string $data): void
public function getLogger(): ?ElasticaLogger
{
if (!$this->_logger instanceof ElasticaLogger || !$this->_logger->isEnabled()) {
return;
}

$connectionArray = [
'host' => $connection->getHost(),
'port' => $connection->getPort(),
'transport' => $connection->getTransport(),
'headers' => $connection->hasConfig('headers') ? $connection->getConfig('headers') : [],
];

$responseData = $elasticaResponse->getData();

$queryTime = $elasticaResponse->getQueryTime();
$engineMS = $responseData['took'] ?? 0;

$itemCount = $responseData['hits']['total']['value'] ?? $responseData['hits']['total'] ?? 0;

$this->_logger->logQuery($path, $method, $data, $queryTime, $connectionArray, $query, $engineMS, $itemCount);
return $this->_logger instanceof ElasticaLogger && $this->_logger->isEnabled() ? $this->_logger : null;
}
}
90 changes: 48 additions & 42 deletions EMS/common-bundle/src/Elasticsearch/ElasticaLogger.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,9 @@

namespace EMS\CommonBundle\Elasticsearch;

use Elastica\Exception\ResponseException;
use Elastica\Request;
use Elastica\Response;
use EMS\CommonBundle\Contracts\Elasticsearch\QueryLoggerInterface;
use EMS\Helpers\Standard\Json;
use Psr\Log\AbstractLogger;
Expand All @@ -15,8 +18,10 @@ class ElasticaLogger extends AbstractLogger implements QueryLoggerInterface
private array $queries = [];
private bool $enabled = true;

public function __construct(private readonly ?LoggerInterface $logger = null, private readonly bool $debug = false)
{
public function __construct(
private readonly ?LoggerInterface $logger = null,
private readonly bool $debug = false
) {
}

public function isEnabled(): bool
Expand All @@ -34,70 +39,71 @@ public function enable(): void
$this->enabled = true;
}

public function getNbQueries(): int
{
return \count($this->queries);
}

/**
* @param array<mixed>|string $data Arguments
* @param array<mixed> $connection Host, port, transport, and headers of the query
* @param array<mixed> $query Arguments
* @return array<mixed>
*/
public function logQuery(string $path, string $method, array|string $data, float $queryTime, array $connection = [], array $query = [], int $engineTime = 0, int $itemCount = 0): void
public function getQueries(): array
{
return $this->queries;
}

/**
* @param array<mixed> $context
*/
public function log($level, $message, array $context = []): void
{
if (null !== $this->logger && $this->isEnabled()) {
$this->logger->log($level, $message, $context);
}
}

public function logResponse(Response $response, Request $request, ?ResponseException $responseException = null): void
{
$responseData = $response->getData();
$queryTime = $response->getQueryTime();
$connection = $request->getConnection();
$data = $request->getData();

$executionMS = $queryTime * 1000;

if ($this->debug) {
if (\is_string($data)) {
$jsonStrings = \explode("\n", $data);
$data = [];
foreach ($jsonStrings as $json) {
if ('' != $json) {
$data[] = Json::decode($json);
}
}
$data = \array_filter(\array_map(static fn ($v) => Json::isJson($v) ? Json::decode($v) : null, $jsonStrings));
} else {
$data = [$data];
}

$this->queries[] = [
'path' => $path,
'method' => $method,
'path' => $request->getPath(),
'method' => $request->getMethod(),
'data' => $data,
'executionMS' => $executionMS,
'engineMS' => $engineTime,
'connection' => $connection,
'queryString' => $query,
'itemCount' => $itemCount,
'engineMS' => $responseData['took'] ?? 0,
'exception' => $responseException,
'connection' => [
'host' => $connection->getHost(),
'port' => $connection->getPort(),
'transport' => $connection->getTransport(),
'headers' => $connection->hasConfig('headers') ? $connection->getConfig('headers') : [],
],
'queryString' => $request->getQuery(),
'itemCount' => $responseData['hits']['total']['value'] ?? $responseData['hits']['total'] ?? 0,
'backtrace' => (new \Exception())->getTraceAsString(),
];
}

if (null !== $this->logger) {
$message = \sprintf('%s (%s) %0.2f ms', $path, $method, $executionMS);
$message = \sprintf('%s (%s) %0.2f ms', $request->getPath(), $request->getMethod(), $executionMS);
$this->logger->info($message, (array) $data);
}
}

public function getNbQueries(): int
{
return \count($this->queries);
}

/**
* @return array<mixed>
*/
public function getQueries(): array
{
return $this->queries;
}

/**
* @param array<mixed> $context
*/
public function log($level, $message, array $context = []): void
{
if (null !== $this->logger && $this->isEnabled()) {
$this->logger->log($level, $message, $context);
}
}

public function reset(): void
{
$this->queries = [];
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,25 +10,16 @@
{% endblock %}

{% block toolbar %}
{% set profiler_markup_version = profiler_markup_version|default(1) %}

{% set status_color = collector.countErrors > 0 ? 'red' %}
{% set icon %}
{% if profiler_markup_version == 1 %}
<img alt="elastica" src="" />
<span class="sf-toolbar-status">{{ collector.querycount }}</span>
{% if collector.querycount > 0 %}
<span class="sf-toolbar-info-piece-additional-detail">in {{ '%0.2f'|format(collector.time) }} ms</span>
{% endif %}
{% else %}
{{ include('@EMSCommon/DataCollector/icon.svg') }}
<span class="sf-toolbar-value">{{ collector.querycount }}</span>
{% if collector.querycount > 0 %}
<span class="sf-toolbar-info-piece-additional-detail">
<span class="sf-toolbar-label">in</span>
<span class="sf-toolbar-value">{{ '%0.2f'|format(collector.time) }}</span>
<span class="sf-toolbar-label">ms</span>
</span>
{% endif %}
{{ include('@EMSCommon/DataCollector/icon.svg') }}
<span class="sf-toolbar-value">{{ collector.queryCount }}</span>
{% if collector.queryCount > 0 %}
<span class="sf-toolbar-info-piece-additional-detail">
<span class="sf-toolbar-label">in</span>
<span class="sf-toolbar-value">{{ '%0.2f'|format(collector.time) }}</span>
<span class="sf-toolbar-label">ms</span>
</span>
{% endif %}
{% endset %}
{% set text %}
Expand All @@ -42,7 +33,7 @@
</div>
<div class="sf-toolbar-info-piece">
<b>Queries</b>
<span class="sf-toolbar-status">{{ collector.querycount }}</span>
<span class="sf-toolbar-status">{{ collector.queryCount }}</span>
</div>
<div class="sf-toolbar-info-piece">
<b>Query Time</b>
Expand All @@ -53,19 +44,18 @@
<span>{{ '%0.2f'|format(collector.executionTime) }} ms</span>
</div>
{% endset %}
{% include '@WebProfiler/Profiler/toolbar_item.html.twig' with { 'link': profiler_url } %}
{% include '@WebProfiler/Profiler/toolbar_item.html.twig' with { 'link': profiler_url, status: status_color } %}
{% endblock %}

{% block menu %}
{% set profiler_markup_version = profiler_markup_version|default(1) %}

<span class="label {{ collector.querycount ? '' : 'disabled' }}">
{% if profiler_markup_version == 1 %}
<span class="icon"><img src="" alt="" /></span>
{% else %}
<span class="icon">{{ include('@EMSCommon/DataCollector/icon.svg') }}</span>
{% endif %}
<span class="label label-status-{{ collector.countErrors > 0 ? 'error' }} {{ collector.queryCount ? '' : 'disabled' }}">
<span class="icon">{{ include('@EMSCommon/DataCollector/icon.svg') }}</span>
<strong>Elastica</strong>
{% if collector.countErrors > 0 %}
<span class="count">
<span>{{ collector.countErrors }}</span>
</span>
{% endif %}
</span>
{% endblock %}

Expand All @@ -82,7 +72,7 @@
<span class="label">Cluster health</span>
</div>
<div class="metric">
<span class="value">{{ collector.querycount }}</span>
<span class="value">{{ collector.queryCount }}</span>
<span class="label">Database Queries</span>
</div>
<div class="metric">
Expand All @@ -97,7 +87,7 @@

<h2>Queries</h2>

{% if not collector.querycount %}
{% if not collector.queryCount %}
<div class="empty">
<p>No queries were performed.</p>
</div>
Expand All @@ -114,11 +104,20 @@
</thead>
<tbody id="queries">
{% for key, query in collector.queries %}
<tr id="queryNo-{{ key }}">
<tr id="queryNo-{{ key }}" class="{{ html_classes({
'log-status-error': query.exception is not null,
'status-error': query.exception is not null,
}) }}">
<td class="nowrap">{{ loop.index }}</td>
<td class="nowrap">{{ '%0.2f'|format(query.executionMS) }}&nbsp;ms</td>
<td class="nowrap">{{ '%0.2f'|format(query.engineMS) }}&nbsp;ms</td>
<td>
{% if query.exception is not null %}
<div>
{{ query.exception.message }}
<hr />
</div>
{% endif %}
<div>
{% for data in query.data %}
<pre>{{ data|json_encode }}</pre>
Expand Down
Loading