ウォッチドッグが何かをログに記録するたびに追加のアクションを実行する方法はありますか?たとえば、他の場所で追加情報を使用してログを記録、解析、およびコンパイルしたいのですが、それでもウォッチドッグにデフォルトの機能を実行させます。
使用できるフックはありますか?
Drupal 7では、 hook_watchdog()
を使用してタスクを実行します。これは、dblog.moduleによって実装されたフックであり、 admin/reports/dblogで、watchdog()
で追加されたメッセージ。
Drupal 8では、次のコードのように、メッセージがログに記録されるたびに呼び出されるサービスを実装する必要があります。
services:
logger.mylog:
class: Drupal\mylog\Logger\MyLog
tags:
- { name: logger }
namespace Drupal\mylog\Logger;
use Drupal\Core\Logger\RfcLoggerTrait;
use Psr\Log\LoggerInterface;
class MyLog implements LoggerInterface {
use RfcLoggerTrait;
/**
* {@inheritdoc}
*/
public function log($level, $message, array $context = array()) {
// Log the message, for example writing it in a file.
}
}
重要なのは、サービスのタグ付け方法です。 tags: [{ name: logger }]
がない場合、サービスはメッセージのログ記録に使用されません。
コード例は hook_watchdog()およびwatchdog()が削除された から取得されます。 Drupalコアから実装されたコードの例については、Syslogサーバーにメッセージを記録する SysLog.php を参照してください。
class SysLog implements LoggerInterface {
use RfcLoggerTrait;
/**
* A configuration object containing syslog settings.
*
* @var \Drupal\Core\Config\Config
*/
protected $config;
/**
* The message's placeholders parser.
*
* @var \Drupal\Core\Logger\LogMessageParserInterface
*/
protected $parser;
/**
* Stores whether there is a system logger connection opened or not.
*
* @var bool
*/
protected $connectionOpened = FALSE;
/**
* Constructs a SysLog object.
*
* @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory
* The configuration factory object.
* @param \Drupal\Core\Logger\LogMessageParserInterface $parser
* The parser to use when extracting message variables.
*/
public function __construct(ConfigFactoryInterface $config_factory, LogMessageParserInterface $parser) {
$this->config = $config_factory
->get('syslog.settings');
$this->parser = $parser;
}
/**
* Opens a connection to the system logger.
*/
protected function openConnection() {
if (!$this->connectionOpened) {
$facility = $this->config
->get('facility');
$this->connectionOpened = openlog($this->config
->get('identity'), LOG_NDELAY, $facility);
}
}
/**
* {@inheritdoc}
*/
public function log($level, $message, array $context = []) {
global $base_url;
// Ensure we have a connection available.
$this
->openConnection();
// Populate the message placeholders and then replace them in the message.
$message_placeholders = $this->parser
->parseMessagePlaceholders($message, $context);
$message = empty($message_placeholders) ? $message : strtr($message, $message_placeholders);
$entry = strtr($this->config
->get('format'), [
'!base_url' => $base_url,
'!timestamp' => $context['timestamp'],
'!type' => $context['channel'],
'!ip' => $context['ip'],
'!request_uri' => $context['request_uri'],
'!referer' => $context['referer'],
'!severity' => $level,
'!uid' => $context['uid'],
'!link' => strip_tags($context['link']),
'!message' => strip_tags($message),
]);
$this
->syslogWrapper($level, $entry);
}
/**
* A syslog wrapper to make syslog functionality testable.
*
* @param int $level
* The syslog priority.
* @param string $entry
* The message to send to syslog function.
*/
protected function syslogWrapper($level, $entry) {
syslog($level, $entry);
}
}
Drupal 8では、Drupal 7のように、同時に異なるロガーをアクティブにすることができるため、ログメッセージを同時に異なる方法。
Drupal 8はPSR3互換のロギングAPIを使用します。追加のロガー動作を実装するために、カスタムロガーサービスを作成して登録できます。
https://www.drupal.org/docs/8/api/logging-api/overview
drush generate logger
コマンドを使用して、ロガーサービスを作成できます。
現在のところ、記録されたメッセージをDrupal 8.に変更する簡単な方法はありません。 https://drupal.stackexchange.com/a/222123/4
Drupal 8:
// Logs a notice
\Drupal::logger('my_module')->notice($message);
// Logs an error
\Drupal::logger('my_module')->error($message);
Drupal 7の場合:
// Logs a notice
watchdog('my_module', $message, array());
// Logs an error
watchdog('my_module', $message, array(), WATCHDOG_ERROR);