From 6f7cd4e26b9785e9de736920607291c4da29c4e4 Mon Sep 17 00:00:00 2001 From: jekuaitk Date: Fri, 13 Dec 2024 11:35:01 +0100 Subject: [PATCH 1/3] Added queue logic to enhance performance --- CHANGELOG.md | 1 + README.md | 25 ++++++ composer.json | 1 + ...queue.advancedqueue_queue.os2web_audit.yml | 10 +++ os2web_audit.services.yml | 2 +- .../AdvancedQueue/JobType/LogMessages.php | 77 +++++++++++++++++++ src/Service/Logger.php | 70 +++++++++++------ 7 files changed, 162 insertions(+), 24 deletions(-) create mode 100644 config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml create mode 100644 src/Plugin/AdvancedQueue/JobType/LogMessages.php diff --git a/CHANGELOG.md b/CHANGELOG.md index 5b30a23..874aba7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +- Added queue logic to enhance performance. - Add new module to track user accessing webform submissions. - Added remote ip to all log lines. diff --git a/README.md b/README.md index 95f75ea..4663132 100644 --- a/README.md +++ b/README.md @@ -12,6 +12,10 @@ These logging providers are designed using Drupal's plugin APIs. Consequently, it opens up possibilities for creating new AuditLogger plugins within other modules, thus enhancing the functionality of this audit logging. +For performance purposes we use a queue system. This avoids hindering +performance more than necessary as the actual logging is done async. Furthermore, +this allows for retries in case any audit log plugins should fail. + ## Installation Enable the module and go to the modules setting page at @@ -47,3 +51,24 @@ logger as shown below: $msg = sprintf('Fetch personal data from service with parameter: %s', $param); $this->auditLogger->info('Lookup', $msg); ``` + +### Queue + +The actual logging is handled by jobs in an [Advanced +Queue](https://www.drupal.org/project/advancedqueue) queue. + +The queue, OS2Web audit (`os2web_audit`), must be +processed by a server `cron` job, e.g. + +```sh +drush advancedqueue:queue:process os2web_audit +``` + +List the queue (and all other queues) with + +```sh +drush advancedqueue:queue:list +``` + +or go to `/admin/config/system/queues/jobs/os2web_audit` for a +graphical overview of jobs in the queue. diff --git a/composer.json b/composer.json index 236bd7f..7e53878 100644 --- a/composer.json +++ b/composer.json @@ -18,6 +18,7 @@ "require": { "ext-curl": "*", "php": "^8.1", + "drupal/advancedqueue": "^1.2", "drush/drush": "^11.5|^12.5" }, "require-dev": { diff --git a/config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml b/config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml new file mode 100644 index 0000000..005d834 --- /dev/null +++ b/config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml @@ -0,0 +1,10 @@ +status: true +dependencies: { } +id: os2web_audit +label: os2web_audit +backend: database +backend_configuration: + lease_time: 300 +processor: cron +processing_time: 90 +locked: false diff --git a/os2web_audit.services.yml b/os2web_audit.services.yml index 34ec888..739a6de 100644 --- a/os2web_audit.services.yml +++ b/os2web_audit.services.yml @@ -5,4 +5,4 @@ services: os2web_audit.logger: class: Drupal\os2web_audit\Service\Logger - arguments: ['@plugin.manager.os2web_audit_logger', '@config.factory', '@current_user', '@logger.factory', '@request_stack'] + arguments: ['@plugin.manager.os2web_audit_logger', '@config.factory', '@current_user', '@logger.factory', '@request_stack', '@entity_type.manager'] diff --git a/src/Plugin/AdvancedQueue/JobType/LogMessages.php b/src/Plugin/AdvancedQueue/JobType/LogMessages.php new file mode 100644 index 0000000..23e625c --- /dev/null +++ b/src/Plugin/AdvancedQueue/JobType/LogMessages.php @@ -0,0 +1,77 @@ + $configuration + */ + public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) { + return new static( + $configuration, + $plugin_id, + $plugin_definition, + $container->get('os2web_audit.logger'), + $container->get('logger.factory'), + ); + } + + /** + * {@inheritdoc} + * + * @phpstan-param array $configuration + */ + public function __construct( + array $configuration, + $plugin_id, + $plugin_definition, + private readonly Logger $logger, + private readonly LoggerChannelFactoryInterface $watchdog, + ) { + parent::__construct($configuration, $plugin_id, $plugin_definition); + } + + /** + * Processes the LogMessages job. + */ + public function process(Job $job): JobResult { + $payload = $job->getPayload(); + + $logger_context = [ + 'job_id' => $job->getId(), + 'operation' => 'response from queue', + ]; + + try { + $this->logger->log($payload['type'], $payload['timestamp'], $payload['line'], $payload['plugin_id'], $payload['metadata']); + $this->watchdog->get(Logger::OS2WEB_AUDIT_LOGGER_CHANNEL)->info('Successfully audit logged message.', $logger_context); + + return JobResult::success(); + } + catch (\Exception $e) { + $this->watchdog->get(Logger::OS2WEB_AUDIT_LOGGER_CHANNEL)->error(sprintf('Failed audit logging message: %s', $e->getMessage()), $logger_context); + + return JobResult::failure($e->getMessage()); + } + } + +} diff --git a/src/Service/Logger.php b/src/Service/Logger.php index 8d3b383..76e1331 100644 --- a/src/Service/Logger.php +++ b/src/Service/Logger.php @@ -2,14 +2,14 @@ namespace Drupal\os2web_audit\Service; -use Drupal\Component\Plugin\Exception\PluginException; use Drupal\Core\Config\ConfigFactoryInterface; +use Drupal\Core\Entity\EntityTypeManagerInterface; use Drupal\Core\Logger\LoggerChannelFactoryInterface; use Drupal\Core\Session\AccountProxyInterface; -use Drupal\os2web_audit\Exception\AuditException; -use Drupal\os2web_audit\Exception\ConnectionException; +use Drupal\advancedqueue\Job; use Drupal\os2web_audit\Form\PluginSettingsForm; use Drupal\os2web_audit\Form\SettingsForm; +use Drupal\os2web_audit\Plugin\AdvancedQueue\JobType\LogMessages; use Drupal\os2web_audit\Plugin\LoggerManager; use Symfony\Component\HttpFoundation\RequestStack; @@ -20,12 +20,16 @@ */ class Logger { + const string OS2WEB_AUDIT_QUEUE_ID = 'os2web_audit'; + const string OS2WEB_AUDIT_LOGGER_CHANNEL = 'os2web_audit_info'; + public function __construct( private readonly LoggerManager $loggerManager, private readonly ConfigFactoryInterface $configFactory, private readonly AccountProxyInterface $currentUser, private readonly LoggerChannelFactoryInterface $watchdog, private readonly RequestStack $requestStack, + private readonly EntityTypeManagerInterface $entityTypeManager, ) { } @@ -43,7 +47,7 @@ public function __construct( * Additional metadata for the log message. Default is an empty array. */ public function info(string $type, string $line, bool $logUser = TRUE, array $metadata = []): void { - $this->log($type, time(), $line, $logUser, $metadata + ['level' => 'info']); + $this->createLoggingJob($type, time(), $line, $logUser, $metadata + ['level' => 'info']); } /** @@ -60,11 +64,11 @@ public function info(string $type, string $line, bool $logUser = TRUE, array $me * Additional metadata for the log message. Default is an empty array. */ public function error(string $type, string $line, bool $logUser = TRUE, array $metadata = []): void { - $this->log($type, time(), $line, $logUser, $metadata + ['level' => 'error']); + $this->createLoggingJob($type, time(), $line, $logUser, $metadata + ['level' => 'error']); } /** - * Logs a message using a plugin-specific logger. + * Creates and enqueues logging job. * * @param string $type * The type of event to log (auth, lookup etc.) @@ -78,11 +82,9 @@ public function error(string $type, string $line, bool $logUser = TRUE, array $m * @param array $metadata * Additional metadata for the log message. Default is an empty array. */ - private function log(string $type, int $timestamp, string $line, bool $logUser = FALSE, array $metadata = []): void { - $config = $this->configFactory->get(SettingsForm::$configName); - $plugin_id = $config->get('provider') ?? SettingsForm::OS2WEB_AUDIT_DEFUALT_PROVIDER; - $configuration = $this->configFactory->get(PluginSettingsForm::getConfigName())->get($plugin_id); + private function createLoggingJob(string $type, int $timestamp, string $line, bool $logUser = FALSE, array $metadata = []): void { + // Enhance logging data with current user and current request information. if ($logUser) { // Add user id to the log message metadata. $metadata['userId'] = $this->currentUser->getEmail(); @@ -95,24 +97,46 @@ private function log(string $type, int $timestamp, string $line, bool $logUser = $line .= sprintf(' Remote ip: %s', $ip_address); } + $config = $this->configFactory->get(SettingsForm::$configName); + $plugin_id = $config->get('provider') ?? SettingsForm::OS2WEB_AUDIT_DEFUALT_PROVIDER; + + $queueStorage = $this->entityTypeManager->getStorage('advancedqueue_queue'); + /** @var \Drupal\advancedqueue\Entity\Queue $queue */ + $queue = $queueStorage->load(self::OS2WEB_AUDIT_QUEUE_ID); + $job = Job::create(LogMessages::class, [ + 'type' => $type, + 'timestamp' => $timestamp, + 'line' => $line, + 'plugin_id' => $plugin_id, + 'metadata' => $metadata, + ]); + + $queue->enqueueJob($job); + + $logger_context = [ + 'job_id' => $job->getId(), + 'operation' => 'log message queued', + ]; + + $this->watchdog->get(self::OS2WEB_AUDIT_LOGGER_CHANNEL)->notice('Added audit logging message to queue for processing.', $logger_context); + } + + /** + * Logs a message using a plugin-specific logger. + */ + public function log(string $type, int $timestamp, string $line, string $plugin_id, array $metadata = []): void { + + $configuration = $this->configFactory->get(PluginSettingsForm::getConfigName())->get($plugin_id); + try { /** @var \Drupal\os2web_audit\Plugin\AuditLogger\AuditLoggerInterface $logger */ $logger = $this->loggerManager->createInstance($plugin_id, $configuration ?? []); $logger->log($type, $timestamp, $line, $metadata); } - catch (PluginException $e) { - $this->watchdog->get('os2web_audit')->error($e->getMessage()); - } - catch (AuditException | ConnectionException $e) { - // Change metadata into string. - $data = implode(', ', array_map(function ($key, $value) { - return $key . " => " . $value; - }, array_keys($metadata), $metadata)); - - // Fallback to send log message info watchdog. - $msg = sprintf("Plugin: %s, Type: %s, Msg: %s, Metadata: %s", $e->getPluginName(), $type, $line, $data); - $this->watchdog->get('os2web_audit')->info($msg); - $this->watchdog->get('os2web_audit_error')->error($e->getMessage()); + catch (\Exception $e) { + // Log (not audit log) an error and throw error in order to retry. + $this->watchdog->get(self::OS2WEB_AUDIT_LOGGER_CHANNEL)->error($e->getMessage()); + throw $e; } } From 8d339b9c6fd7789cbffb876de95fa9804f1a4ad5 Mon Sep 17 00:00:00 2001 From: jekuaitk Date: Fri, 13 Dec 2024 11:43:47 +0100 Subject: [PATCH 2/3] Upgrade php version in actions --- .github/workflows/pr.yml | 6 +++--- src/Service/Logger.php | 13 +++++++++++++ 2 files changed, 16 insertions(+), 3 deletions(-) diff --git a/.github/workflows/pr.yml b/.github/workflows/pr.yml index 42565ef..2003cf9 100644 --- a/.github/workflows/pr.yml +++ b/.github/workflows/pr.yml @@ -23,7 +23,7 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - php-versions: [ '8.1' ] + php-versions: [ '8.3' ] dependency-version: [ prefer-lowest, prefer-stable ] steps: - uses: actions/checkout@master @@ -55,7 +55,7 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - php-versions: [ '8.1' ] + php-versions: [ '8.3' ] steps: - uses: actions/checkout@master - name: Setup PHP, with composer and extensions @@ -87,7 +87,7 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - php-versions: [ '8.1' ] + php-versions: [ '8.3' ] steps: - uses: actions/checkout@master - name: Setup PHP, with composer and extensions diff --git a/src/Service/Logger.php b/src/Service/Logger.php index 76e1331..b7e6b9d 100644 --- a/src/Service/Logger.php +++ b/src/Service/Logger.php @@ -123,6 +123,19 @@ private function createLoggingJob(string $type, int $timestamp, string $line, bo /** * Logs a message using a plugin-specific logger. + * + * @param string $type + * The type of event to log (auth, lookup etc.) + * @param int $timestamp + * The timestamp for the log message. + * @param string $line + * The log message. + * @param string $plugin_id + * The logging plugin id. + * @param array $metadata + * Additional metadata for the log message. Default is an empty array. + * + * @throws \Exception. */ public function log(string $type, int $timestamp, string $line, string $plugin_id, array $metadata = []): void { From 1bf3fcb8f9b17c5da64336d3d35492cec50db2ed Mon Sep 17 00:00:00 2001 From: jekuaitk Date: Fri, 13 Dec 2024 12:14:53 +0100 Subject: [PATCH 3/3] Reduced logging significantly --- ...queue.advancedqueue_queue.os2web_audit.yml | 2 +- .../AdvancedQueue/JobType/LogMessages.php | 16 ++------ src/Service/Logger.php | 41 +++++++++---------- 3 files changed, 24 insertions(+), 35 deletions(-) diff --git a/config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml b/config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml index 005d834..457c518 100644 --- a/config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml +++ b/config/install/advancedqueue.advancedqueue_queue.os2web_audit.yml @@ -5,6 +5,6 @@ label: os2web_audit backend: database backend_configuration: lease_time: 300 -processor: cron +processor: daemon processing_time: 90 locked: false diff --git a/src/Plugin/AdvancedQueue/JobType/LogMessages.php b/src/Plugin/AdvancedQueue/JobType/LogMessages.php index 23e625c..b6a569a 100644 --- a/src/Plugin/AdvancedQueue/JobType/LogMessages.php +++ b/src/Plugin/AdvancedQueue/JobType/LogMessages.php @@ -2,11 +2,13 @@ namespace Drupal\os2web_audit\Plugin\AdvancedQueue\JobType; -use Drupal\Core\Logger\LoggerChannelFactoryInterface; +use Drupal\Component\Plugin\Exception\PluginException; use Drupal\Core\Plugin\ContainerFactoryPluginInterface; use Drupal\advancedqueue\Job; use Drupal\advancedqueue\JobResult; use Drupal\advancedqueue\Plugin\AdvancedQueue\JobType\JobTypeBase; +use Drupal\os2web_audit\Exception\AuditException; +use Drupal\os2web_audit\Exception\ConnectionException; use Drupal\os2web_audit\Service\Logger; use Symfony\Component\DependencyInjection\ContainerInterface; @@ -31,7 +33,6 @@ public static function create(ContainerInterface $container, array $configuratio $plugin_id, $plugin_definition, $container->get('os2web_audit.logger'), - $container->get('logger.factory'), ); } @@ -45,7 +46,6 @@ public function __construct( $plugin_id, $plugin_definition, private readonly Logger $logger, - private readonly LoggerChannelFactoryInterface $watchdog, ) { parent::__construct($configuration, $plugin_id, $plugin_definition); } @@ -56,20 +56,12 @@ public function __construct( public function process(Job $job): JobResult { $payload = $job->getPayload(); - $logger_context = [ - 'job_id' => $job->getId(), - 'operation' => 'response from queue', - ]; - try { $this->logger->log($payload['type'], $payload['timestamp'], $payload['line'], $payload['plugin_id'], $payload['metadata']); - $this->watchdog->get(Logger::OS2WEB_AUDIT_LOGGER_CHANNEL)->info('Successfully audit logged message.', $logger_context); return JobResult::success(); } - catch (\Exception $e) { - $this->watchdog->get(Logger::OS2WEB_AUDIT_LOGGER_CHANNEL)->error(sprintf('Failed audit logging message: %s', $e->getMessage()), $logger_context); - + catch (PluginException | ConnectionException | AuditException $e) { return JobResult::failure($e->getMessage()); } } diff --git a/src/Service/Logger.php b/src/Service/Logger.php index b7e6b9d..1882573 100644 --- a/src/Service/Logger.php +++ b/src/Service/Logger.php @@ -100,25 +100,27 @@ private function createLoggingJob(string $type, int $timestamp, string $line, bo $config = $this->configFactory->get(SettingsForm::$configName); $plugin_id = $config->get('provider') ?? SettingsForm::OS2WEB_AUDIT_DEFUALT_PROVIDER; - $queueStorage = $this->entityTypeManager->getStorage('advancedqueue_queue'); - /** @var \Drupal\advancedqueue\Entity\Queue $queue */ - $queue = $queueStorage->load(self::OS2WEB_AUDIT_QUEUE_ID); - $job = Job::create(LogMessages::class, [ + $payload = [ 'type' => $type, 'timestamp' => $timestamp, 'line' => $line, 'plugin_id' => $plugin_id, 'metadata' => $metadata, - ]); + ]; - $queue->enqueueJob($job); + try { + $queueStorage = $this->entityTypeManager->getStorage('advancedqueue_queue'); + /** @var \Drupal\advancedqueue\Entity\Queue $queue */ + $queue = $queueStorage->load(self::OS2WEB_AUDIT_QUEUE_ID); - $logger_context = [ - 'job_id' => $job->getId(), - 'operation' => 'log message queued', - ]; + $job = Job::create(LogMessages::class, $payload); + + $queue->enqueueJob($job); + } + catch (\Exception $exception) { + $this->watchdog->get(self::OS2WEB_AUDIT_LOGGER_CHANNEL)->error(sprintf('Failed creating job: %s', $exception->getMessage()), $payload); + } - $this->watchdog->get(self::OS2WEB_AUDIT_LOGGER_CHANNEL)->notice('Added audit logging message to queue for processing.', $logger_context); } /** @@ -135,22 +137,17 @@ private function createLoggingJob(string $type, int $timestamp, string $line, bo * @param array $metadata * Additional metadata for the log message. Default is an empty array. * - * @throws \Exception. + * @throws \Drupal\Component\Plugin\Exception\PluginException + * @throws \Drupal\os2web_audit\Exception\ConnectionException + * @throws \Drupal\os2web_audit\Exception\AuditException */ public function log(string $type, int $timestamp, string $line, string $plugin_id, array $metadata = []): void { $configuration = $this->configFactory->get(PluginSettingsForm::getConfigName())->get($plugin_id); - try { - /** @var \Drupal\os2web_audit\Plugin\AuditLogger\AuditLoggerInterface $logger */ - $logger = $this->loggerManager->createInstance($plugin_id, $configuration ?? []); - $logger->log($type, $timestamp, $line, $metadata); - } - catch (\Exception $e) { - // Log (not audit log) an error and throw error in order to retry. - $this->watchdog->get(self::OS2WEB_AUDIT_LOGGER_CHANNEL)->error($e->getMessage()); - throw $e; - } + /** @var \Drupal\os2web_audit\Plugin\AuditLogger\AuditLoggerInterface $logger */ + $logger = $this->loggerManager->createInstance($plugin_id, $configuration ?? []); + $logger->log($type, $timestamp, $line, $metadata); } }