当前位置: 首页 > 工具软件 > Monolog > 使用案例 >

php日志,monolog - PHP 日志神器

狄法
2023-12-01

monolog - PHP 日志神器

monolog - PHP 日志神器

Monolog 发送你的日志到文件、到sockets、到邮箱、到数据库或(和)者其他网路存储服务(云)。这里用了或与和,因为Monolog的确可以做到同时保存到一个或多个存储介质。

#### 安装

$ composer require monolog/monolog

#### 基本用法 (初步印象)

use Monolog\Logger;

use Monolog\Handler\StreamHandler;

// create a log channel

$log = new Logger('name');

$log->pushHandler(new StreamHandler('path/to/your.log', Logger::WARNING));

// add records to the log

$log->warning('Foo');$log->error('Bar');

#### 核心概念

Every Logger instance has a channel (name) and a stack of handlers. Whenever you add a record to the logger, it traverses the handler stack. Each handler decides whether it fully handled the record, and if so, the propagation of the record ends there.

每一个Logger实例都有一个通道(也就是一个唯一的名称)和一个有由一个或多个处理程序组成的栈。当我们添加一个记录到Logger的时候,它会遍历这个处理程序栈。每一个处理程序决定是否去充分处理这个记录,如果是,则处理到此为止(停止冒泡)。这里的充分指的是我们想不想了,想的话就继续,不想就停止。

这就允许我们灵活的设置日志了。比如我们有一个StreamHandler,它在栈的最底部,它会把记录都保存到硬盘上,在它上面有一个MailHandler,它会在错误消息被记录的时候发送邮件。Handlers 都有一个$bubble属性,用来定义当某个处理程序在处理记录的时候是否阻塞处理(阻塞的话,就是这个记录到我这里就算处理完毕了,不要冒泡处理了,听话)。在这个例子中,我们设置MailHandler的$bubble为false,意思就是说记录都会被MailHandler处理,不会冒泡到StreamHandler了。

我必须补充一下:这里提到了栈,也提到了冒泡,乍一看有点晕,因为我们理解冒泡是自下而上的过程,栈就是一个类似杯子的容器,然后上面又说底部是StreamHandler,上面是MailHandler,结果是MailHandler处理了,停止冒泡到StreamHandler了,给人的感觉是这个泡是从上往下冒的,666,这能叫冒泡么?英雄时刻:堆呀栈呀啥的,我也看过N次,但也总是忘(原谅我野生的),这里再次谨记,堆是先进先出(First-In/First-Out),想想[自来]水管;栈就是先进后出(First-In/Last-Out),想想一个有N层颜色的冰淇淋装在一个杯子里,下面是黄色的,...,最上面是粉红的,所以,你先吃得是粉红色的(MailHandler),后吃的是黄色的(StreamHandler),实际上,这个泡冒的没错,确切的说,这个泡冒在了一个倒立的杯子中,当然杯口没有被封住。

继续...

我们可以创建很多Logger,每个Logger定义一个通道(e.g.:db,request,router,...),每个通道可结合多个Handler,Handler可以被写成可通用的或者不可通用的。通道,同日志中日期时间一样,它是一个名称,在日志中就是一个字符串被记录下来,大概是这样 2016-04-25 12:33:00 通道名称 记录内容,具体格式看设置了,可以用来识别或者过滤。

每一个Handler都有一个Formatter,用来格式化日志了。不详细介绍了。

自定义日志等级在monolog中不可用,只有8种RFC 5424 等级,即 debug, info, notice, warning, error, critical, alert, emergency。但是如果我们真的有特殊需求的话,比如归类等,我们可以添加Processors到 Logger,当然是在日志消息被处理之前。我这估计这辈子都不会添加Processors。

#### 日志等级

- **DEBUG** (100): Detailed debug information.详细的Debug信息

- **INFO** (200): Interesting events. Examples: User logs in, SQL logs.感兴趣的事件或信息,如用户登录信息,SQL日志信息

- **NOTICE** (250): Normal but significant events.普通但重要的事件信息

- **WARNING **(300): Exceptional occurrences that are not errors. Examples: Use of deprecated APIs, poor use of an API, undesirable things that are not necessarily wrong.

- **ERROR** (400): Runtime errors that do not require immediate action but should typically be logged and monitored.

- **CRITICAL** (500): Critical conditions. Example: Application component unavailable, unexpected exception.

- **ALERT** (550): Action must be taken immediately. Example: Entire website down, database unavailable, etc. This should trigger the SMS alerts and wake you up.

- **EMERGENCY** (600): Emergency: system is unusable.

#### 配置一个Logger

Here is a basic setup to log to a file and to firephp on the DEBUG level:

use Monolog\Logger;

use Monolog\Handler\StreamHandler;

use Monolog\Handler\FirePHPHandler;

// Create the logger

$logger = new Logger('my_logger');

// Now add some handlers

$logger->pushHandler(new StreamHandler(__DIR__.'/my_app.log', Logger::DEBUG));

$logger->pushHandler(new FirePHPHandler());

// You can now use your logger

$logger->addInfo('My logger is now ready');

我们来分析一下这个配置。

The first step is to create the logger instance which will be used in your code. The argument is a channel name, which is useful when you use several loggers (see below for more details about it).

第一步,创建Logger实例,参数即通道名字。

The logger itself does not know how to handle a record. It delegates it to some handlers. The code above registers two handlers in the stack to allow handling records in two different ways.

Logger本身不知道如何处理记录,它将处理委托给Handler[s],上面的代码注册了两个Handlers,这样就可以用两种方法来处理记录。

Note that the FirePHPHandler is called first as it is added on top of the stack. This allows you to temporarily add a logger with bubbling disabled if you want to override other configured loggers.

提示:FirePHPHandler最先被调用,因为它被添加在栈的顶部。这就允许你临时添加一个阻塞的Logger,如果你想覆盖其他Logger[s]的话。

#### 添加额外的数据到记录

Monolog 提供两种方法来添加额外的信息到简单的文本信息(along the simple textual message)。

#### 使用日志上下文

第一种,即当前日志上下文,允许传递一个数组作为第二个参数,这个数组的数据是额外的信息:

$logger->addInfo('Adding a new user', array('username' => 'Seldaek'));

简单的Handler(SteamHandler)会简单的将数组格式化为字符串,功能丰富点的Handler(FirePHP)可以搞得更好看。

#### 使用 processors

Processors 可以是任何可调用的方法(回调)。它们接受$record作为参数,然后返回它($record),返回之前,即是我们添加额外信息的操作,在这里,这个操作是改变$record的extrakey的值。像这样:

$logger->pushProcessor(function ($record) {

$record['extra']['dummy'] = 'Hello world!';

return $record;

});

Monolog 提供了一些内置的 processors。看dedicated chapter

收回我说的话,我可能很快就会用到 Processors的。

#### 使用通道

通道是识别record记录的是程序哪部分的好方法(当然,关键词匹配啊),这在大型应用中很有用,如 MonologBundle in Symfony2。

想象一下,两个Logger共用一个Handler,通过这个Handler将记录写入一个文件。这时使用通道能够让我们识别出是哪一个Logger处理的。我们可简单的在这个文件中过滤这个或者那个通道。

use Monolog\Logger;

use Monolog\Handler\StreamHandler;

use Monolog\Handler\FirePHPHandler;

// Create some handlers

$stream = new StreamHandler(__DIR__ . '/my_app.log', Logger::DEBUG);

$firephp = new FirePHPHandler();

// Create the main logger of the app

$logger = new Logger('my_logger');

$logger->pushHandler($stream);

$logger->pushHandler($firephp);

// Create a logger for the security-related stuff with a different channel

$securityLogger = new Logger('security');

$securityLogger->pushHandler($stream);

$securityLogger->pushHandler($firephp);

// Or clone the first one to only change the channel

$securityLogger = $logger->withName('security');

#### 自定义日志格式

在 Monolog 中个性化日志是很easy的。大部分 Handler 使用

$record['formatted']

的值。这个值依赖于 formatter 的设置。我们可以选择预定义的 formatter 类或者编写自己的。

配置一个预定义的 formatter 类,只需要将其设置成 Handler 的字段(属性)即可:

// the default format is "Y-m-d H:i:s"

$dateFormat = "Y n j, g:i a";

// the default output format is [%datetime%] %channel%.%level_name%: %message% %context% %extra%\n"

$output = "%datetime% > %level_name% > %message% %context% %extra%\n";

$formatter = new LineFormatter($output, $dateFormat);

// Create a handler

$stream = new StreamHandler(__DIR__ . 'my_app.log', Logger:DEBUG);

$stream->setFormatter($formatter);

// bind it to a logger object

$securityLogger = new Logger('security');

$securityLogger->pushHandler($stream);

formatter 是可以在N个 Handler 之间复用的,并且可在N个 Logger 之间共享 Handler。

 类似资料: