日志记录
Overview
在本指南中,您可以了解如何使用驱动程序为应用程序配置日志记录。 日志记录的目的是记录驱动程序事件。
记录器按照您可以指定的严重性或详细程度记录消息。 通过在应用程序中启用记录器,您可以接收有关应用程序活动的高级别、详细级别或介于两者之间的信息。
提示
要学习;了解有关日志记录严重性级别的更多信息,请参阅有关消息日志记录的 系统日志标准的 维基百科条目。
启用日志记录
要在 Client
实例上配置记录器,请在创建ClientOptions
对象时调用SetLoggerOptions()
方法。 SetLoggerOptions()
方法采用LoggerOptions
类型作为参数。 设置此LoggerOptions
类型以配置应用程序的记录器。
以下代码展示了如何创建启用日志记录的客户端:
loggerOptions := options. Logger(). SetComponentLevel(options.LogComponentCommand, options.LogLevelDebug) clientOptions := options. Client(). ApplyURI(uri). SetLoggerOptions(loggerOptions) client, err := mongo.Connect(clientOptions)
配置记录器
要创建LoggerOptions
对象,请调用options.Logger()
方法。 下表描述了如何设置LoggerOptions
类型的属性以配置记录器。 第一列列出LoggerOptions
属性,第二列描述属性,第三列列出每个属性相应的 setter 方法和参数:
属性 | 说明 | Setter 方法 |
---|---|---|
ComponentLevels Type: map[LogComponent]LogLevel | A mapping of components to log severity levels. The driver uses the
LogLevel for each LogComponent to determine if the log
message is generated.To learn more about the LogComponent and LogLevel types, see
the Log Components and Severity Levels
section of this guide. | SetComponentLevel() Parameters: LogComponent , LogLevel |
Sink Type: LogSink | The logging interface that the driver uses to log messages.
The LogSink type is an interface that you can implement to
provide a custom sink or integrate a third-party
logger for the driver's logs. If you don't set this
property, the driver uses the standard logging library.To learn more, see the Use a Custom Logger and Integrate Third-Party
Loggers sections of this guide. | SetSink() Parameter: LogSink |
MaxDocumentLength Type: uint Default: 1000 | The maximum length in bytes of each log message that the driver emits. If the
message is larger than this value, the driver
truncates it and appends ellipses to the partial log message. | SetMaxDocumentLength() Parameter: uint |
提示
将日志写入特定文件
默认情况下,标准记录器会将消息记录到控制台 ( stderr
)。 您可以通过将MONGODB_LOG_PATH
环境变量设置为stdout
或文件路径来指定日志记录目标。
日志组件和严重性级别
要指定驱动程序记录日志的组件,请设置LogComponent
类型。 下表描述了LogComponent
的内置规范:
设置 | 说明 | 枚举值 |
---|---|---|
| 为所有组件启用日志记录 |
|
| 启用命令监控器日志记录 |
|
| 启用拓扑结构日志记录 |
|
| 启用服务器选择日志记录 |
|
| 启用连接服务日志记录 |
|
您可以使用设置名称或其枚举值来指定日志组件。 以下代码显示了启用命令监控的等效方法:
// Using named value comp := options.LogComponentCommand // Using enumeration comp := options.LogComponent(1)
要指定日志严重性级别,请设置 LogLevel
类型。以下代码展示了如何启用 LevelDebug
级别的日志记录:
lvl := options.LogLevelDebug
重要
Go 驱动程序目前仅发出LevelDebug
级别的消息,但它支持LogLevel
的其他规范。 要了解详情,请参阅 LogLevel API 文档。
例子
此示例展示了如何配置具有以下规范的标准记录器:
最大文档长度为
25
字节。日志组件是
LogComponentCommand
。日志记录严重性级别为
LevelDebug
。
loggerOptions := options. Logger(). SetMaxDocumentLength(25). SetComponentLevel(options.LogComponentCommand, options.LogLevelDebug) // Creates options that include the logger specification clientOptions := options. Client(). ApplyURI(uri). SetLoggerOptions(loggerOptions)
以下代码执行插入操作,该操作会生成日志消息:
type Item struct { Name string } coll := client.Database("db").Collection("testColl") _, err = coll.InsertOne(context.TODO(), Item{Name: "grapefruit"})
{"command":"{\"insert\": \"testColl\",\"or...","commandName":"insert","databaseName":"db","driverConnectionId":1,"message":"Command started","operationId":0,"requestId":13,"serverConnectionId":97377,"serverHost":"...","serverPort":27017,"timestamp":...} {"commandName":"insert","driverConnectionId":1,"durationMS":19,"message":"Command succeeded","operationId":0,"reply":"{\"n\": {\"$numberInt\":\"1\"},...","requestId":13,"serverConnectionId":97377,"serverHost":"...","serverPort":27017,"timestamp":...}
使用自定义记录器
如果标准日志记录库不能满足您的需求,您可以实现自定义记录器。 通过自定义日志记录配置,您可以更好地控制日志消息的内容、格式和频率。
要使用自定义记录器,请定义记录器结构体,并实现该结构体的 Info()
和 Error()
方法。然后,在您的 LoggerOptions
实例上调用 SetSink()
方法,将记录器设置为您的 Client
的 LogSink
。
例子
此示例演示了如何定义和实现自定义记录器。
使用自定义日志消息格式实施 和Info()
Error()
方法。
func (logger *CustomLogger) Info(level int, msg string, _ ...interface{}) { logger.mu.Lock() defer logger.mu.Unlock() if options.LogLevel(level+1) == options.LogLevelDebug { fmt.Fprintf(logger, "level: %d DEBUG, message: %s\n", level, msg) } else { fmt.Fprintf(logger, "level: %d INFO, message: %s\n", level, msg) } } func (logger *CustomLogger) Error(err error, msg string, _ ...interface{}) { logger.mu.Lock() defer logger.mu.Unlock() fmt.Fprintf(logger, "error: %v, message: %s\n", err, msg) }
Writer
为您的记录器分配一个 ,并将其设立为Sink
的Client
。
在此示例中,记录器在LevelDebug
级别记录命令和连接事件:
buf := bytes.NewBuffer(nil) sink := &CustomLogger{Writer: buf} loggerOptions := options. Logger(). SetSink(sink). SetComponentLevel(options.LogComponentCommand, options.LogLevelDebug). SetComponentLevel(options.LogComponentConnection, options.LogLevelDebug) // Creates options that include the logger specification clientOptions := options. Client(). ApplyURI(uri). SetLoggerOptions(loggerOptions)
执行操作。
以下代码执行插入操作,该操作会生成日志消息:
type Item struct { Name string } coll := client.Database("db").Collection("testColl") _, err = coll.InsertOne(context.TODO(), Item{Name: "grapefruit"})
level: 1 DEBUG, message: Connection pool created level: 1 DEBUG, message: Connection pool ready level: 1 DEBUG, message: Connection pool created level: 1 DEBUG, message: Connection pool ready level: 1 DEBUG, message: Connection pool created level: 1 DEBUG, message: Connection pool ready level: 1 DEBUG, message: Connection checkout started level: 1 DEBUG, message: Connection created level: 1 DEBUG, message: Connection ready level: 1 DEBUG, message: Connection checked out level: 1 DEBUG, message: Command started level: 1 DEBUG, message: Command succeeded level: 1 DEBUG, message: Connection checked in
集成第三方记录器
Go 中有许多可用的第三方日志记录包。 要在应用程序中使用第三方记录器,请创建一个记录器并将其指定为LoggerOptions
实例中的接收器。
例子
此示例演示如何将第三方日志记录包logrus
集成到您的应用程序中。
将记录器设置为Sink
Client
的 。
在以下代码示例中,记录器配置为在LevelDebug
级别记录命令。
sink := logrusr.New(myLogger).GetSink() // Sets options when configuring the logrus logger loggerOptions := options. Logger(). SetSink(sink). SetComponentLevel(options.LogComponentCommand, options.LogLevelDebug) // Creates options that include the logger specification clientOptions := options. Client(). ApplyURI(uri). SetLoggerOptions(loggerOptions)
执行操作。
以下代码执行一些 CRUD 操作,从而生成日志消息:
type Item struct { Name string } coll := client.Database("db").Collection("testColl") docs := []interface{}{ Item{Name: "starfruit"}, Item{Name: "kiwi"}, Item{Name: "cantaloupe"}, } _, err = coll.InsertMany(context.TODO(), docs) if err != nil { panic(err) } _, err = coll.DeleteOne(context.TODO(), Item{Name: "kiwi"}) if err != nil { panic(err) }
{ "command": "{\"insert\": \"testColl\", ...}", "commandName": "insert", "databaseName": "db", ... "level": "debug", "message": "Command started", "msg": "Command started", ... "time": "2023-07-06 10:23:42" } { "commandName": "insert", ... "level": "debug", "message": "Command succeeded", "msg": "Command succeeded", ... "time": "2023-07-06 10:23:42" } { "command": "{\"delete\": \"testColl\", ...}", "commandName": "delete", "databaseName": "db", ... "level": "debug", "message": "Command started", "msg": "Command started", ... "time": "2023-07-06 10:23:42" } { "commandName": "delete", ... "level": "debug", "message": "Command succeeded", "msg": "Command succeeded", ... "time": "2023-07-06 10:23:42" }
提示
日志记录包
更多信息
有关设置客户端选项的更多信息,请参阅《连接指南》。
API 文档
要进一步了解本指南所讨论的任何类型或方法,请参阅以下 API 文档: