Access Log
概念阐述
ServiceComb 提供了基于 Vert.x 的 access log 和 request log 功能。当用户使用 REST over Vertx 通信方式时,可以通过简单的配置启用 access log 打印功能。当用户 client 端进行远程调用时,可以通过简单的配置启用 request log 打印功能
场景描述
-
用户在调试服务时可能需要开启 access log。在使用 REST over servlet 通信方式的情况下,可以使用 web容器 的 access log 功能;而在使用 REST over Vertx 通信方式的情况下,可以使用 ServiceComb 提供的一套 access log 功能。
-
用户想要跟踪,记录客户端远程调用信息, 可以开启 request log。request log 同时支持记录 rest 和 highway 远程调用方式。
配置说明
启用 Access Log & Request Log
用户需要在 microservice.yaml 文件中增加配置以启用 access log 和 request log,配置示例如下:
servicecomb:
accesslog:
## server 端 启用access log
enabled: true
## server 端 自定义 access log 日志格式
pattern: "%h - - %t %r %s %B %D"
request:
## client 端开启 request log
enabled: true
## client 端自定义 request log 日志格式
pattern: "%h %SCB-transport - - %t %r %s %D"
Access log & Request log 配置项说明
配置项 | 取值范围 | 默认值 | 说明 |
---|---|---|---|
servicecomb.accesslog.enabled | true/false | false | 如果为true则启用access log,否则不启用 |
servicecomb.accesslog.pattern | 表示打印格式的字符串 | "%h - - %t %r %s %B %D" | 配置项见_日志元素说明表_ |
servicecomb.accesslog.request.enabled | true/false | false | 如果为true则启用request log,否则不启用 |
servicecomb.accesslog.request.pattern | 表示打印格式的字符串 | "%h %SCB-transport - - %t %r %s %D" | 配置项见_日志元素说明表_ |
日志格式配置
目前可用的日志元素配置项见 日志元素说明表(Apache & W3C) 和 日志元素说明表(ServiceComb) 。
日志元素说明表 (Apache & W3C)
元素名称 | Apache日志格式 | W3C日志格式 | 说明 |
---|---|---|---|
HTTP method | %m | cs-method | - |
HTTP status | %s | sc-status | - |
Duration in second | %T | - | - |
Duration in millisecond | %D | - | - |
Remote hostname | %h | - | - |
Local hostname | %v | - | - |
Local port | %p | - | - |
Size of response | %B | - | 如果消息体长度为零则打印"0" |
Size of response | %b | - | 如果消息体长度为零则打印"-" |
First line of request | %r | - | 包含HTTP Method、Uri、Http版本三部分内容 |
URI path | %U | cs-uri-stem | - |
Query string | %q | cs-uri-query | - |
URI path and query string | - | cs-uri | - |
Request protocol | %H | - | - |
Datetime the request is received | %t | - | 按照默认设置打印时间戳,格式为"EEE, dd MMM yyyy HH:mm:ss zzz",语言为英文,时区为GMT |
Configurable datetime the request is received | %{PATTERN}t | - | 按照指定的格式打印时间戳,语言为英文,时区为GMT |
Configurable datetime the request is received | %{PATTERN|TIMEZONE|LOCALE}t | - | 按照指定的格式、语言、时区打印时间戳。允许省略其中的某部分配置(但两个分隔符号"|"不可省略)。 |
Request header | %{VARNAME}i | - | 如果没有找到指定的header,则打印"-" |
Response header | %{VARNAME}o | - | 如果没有找到指定的header,则打印"-" |
Cookie | %{VARNAME}C | - | 如果没有找到指定的cookie,则打印"-" |
日志元素说明表(ServiceComb)
Element | Placeholder | Comment |
---|---|---|
TraceId | %SCB-traceId | 打印ServiceComb生成的trace id,找不到则打印"-" |
Invocation Context | %{VARNAME}SCB-ctx | 打印key为VARNAME 的invocation context值,找不到则打印"-" |
Transport Method | %SCB-transport | 打印当前调用的 transport method 。 rest 或者 highway |
Access log 与 Request log 的日志元素对比
元素名称 | Apache&W3C日志格式 | access log | access log 说明 | request log | request log说明 |
---|---|---|---|---|---|
HTTP method | %m & cs-method | support | - | support | - |
HTTP status | %s & sc-status | support | - | support | - |
Duration in second | %T | support | - | support | - |
Duration in millisecond | %D | support | - | support | - |
Remote hostname | %h | support | - | support | - |
Local hostname | %v | support | - | support | - |
Local port | %p | support | - | support | - |
Size of response | %B | support | 如果消息体长度为零则打印"0" | unsupported | - |
Size of response | %b | support | 如果消息体长度为零则打印"-" | unsupported | - |
First line of request | %r | support | 包含HTTP Method、Uri、Http版本三部分内容 | support | 包含HTTP Method、Uri、Http版本三部分内容 |
URI path | %U & cs-uri-stem | support | - | support | - |
Query string | %q & cs-uri-query | support | - | support | - |
URI path and query string | cs-uri | support | - | support | - |
Request protocol | %H | support | - | support | - |
Datetime of the request | %t | support | 收到请求的时间。默认格式为"EEE, dd MMM yyyy HH:mm:ss zzz",语言为英文,时区为GMT | support | 发送请求的时间。默认格式为"EEE, dd MMM yyyy HH:mm:ss zzz",语言为英文,时区为GMT |
Configurable datetime the request of the request | %{PATTERN}t | support | 收到请求的时间。按照指定的格式打印时间戳,语言为英文,时区为GMT | support | 发送请求的时间。按照指定的格式打印时间戳,语言为英文,时区为GMT |
Configurable datetime the request of the request | %{PATTERN|TIMEZONE|LOCALE}t | support | 收到请求的时间。按照指定的格式、语言、时区打印时间戳。允许省略其中的某部分配置(但两个分隔符号"|"不可省略)。 | support | 发送请求的时间。按照指定的格式、语言、时区打印时间戳。允许省略其中的某部分配置(但两个分隔符号"|"不可省略)。 |
Request header | %{VARNAME}i | support | 如果没有找到指定的header,则打印"-" | support | 如果没有找到指定的header,则打印"-" |
Response header | %{VARNAME}o | support | 如果没有找到指定的header,则打印"-" | support | 如果没有找到指定的header,则打印"-" |
Cookie | %{VARNAME}C | support | 如果没有找到指定的cookie,则打印"-" | support | 如果没有找到指定的cookie,则打印"-" |
TraceId | %SCB-traceId | support | 打印ServiceComb生成的trace id,找不到则打印"-" | support | 打印ServiceComb生成的trace id,找不到则打印"-" |
Invocation Context | %{VARNAME}SCB-ctx | support | 打印key为VARNAME 的invocation context值,找不到则打印"-" |
support | 打印key为VARNAME 的invocation context值,找不到则打印"-" |
transport method | %SCB-transport | unsupported | 只支持 rest 形式调用 | support | 调用使用的transport method |
日志输出文件配置
Access log & Request log 的日志打印实现框架默认采用 Slf4j ,其中 logger 名称分别为 accesslog
和 requestlog
, 可以结合应用使用的日志框架将日志输出到不同的文件中。
自定义扩展 Access Log & Request Log
用户可以利用 ServiceComb 提供的 AccessLogItem 扩展机制,定制自己的 AccessLogItem, 我们把 Request Log 也当做一种 Access Log。
相关类说明
- AccessLogItem
public interface AccessLogItem<T> {
// 从Server端获取信息,打印 Access Log 日志
default void appendServerFormattedItem(ServerAccessLogEvent accessLogEvent, StringBuilder builder) {
}
// 从Client 端获取信息, 打印 Request Log
default void appendClientFormattedItem(InvocationFinishEvent clientLogEvent, StringBuilder builder) {
}
}
AccessLogItem 的定义如上所示
Server 端 每收到一个请求,会触发一次 Access Log 日志打印。Client 端 每次对外远程调用结束,会触发一次 Request Log 日志打印。
每次日志打印,SDK 都会遍历有效的
AccessLogItem
,调用对应的方法获取此 Item 生成的 Log片 段,并将全部片段拼接成一条 Log 打印到日志文件中。
- VertxRestAccessLogItemMeta
// pattern占位符前缀
protected String prefix;
// pattern占位符后缀
protected String suffix;
// 优先级序号
protected int order;
// AccessLogItem构造器
protected AccessLogItemCreator<RoutingContext> accessLogItemCreator;
VertxRestAccessLogItemMeta
包含如上属性,它定义了 ServiceComb 如何解析 pattern 字符串以获得特定的 AccessLogItem。
-
如果用户想要定义一个占位符为
%user-defined
的AccessLogItem
,则需要声明一个VertxRestAccessLogItemMeta
的子类,设置 prefix="%user-defined",suffix=null,当AccessLogPatternParser
解析到 "%user-defined" 时,从此 meta 类中取得AccessLogItemCreator
创建对应的AccessLogItem
。注意:由于 "%user-defined" 占位符中没有变量部分,因此调用AccessLogItemCreator
传入的配置参数为null。 -
如果用户想要定义一个占位符为
%{VARNAME}user-defined
的AccessLogItem
,则声明的VertxRestAccessLogItemMeta
子类中,设置prefix="%{",suffix="}user-defined",当AccessLogPatternParser
解析到 "%{VARNAME}user-defined"时,会截取出"VARNAME"作为配置参数传入AccessLogItemCreator
,创建一个AccessLogItem
。
VertxRestAccessLogItemMeta
有一个子类 CompositeVertxRestAccessLogItemMeta
,当用户需要定义多个 AccessLogItem 时,可以将多个 VertxRestAccessLogItemMeta
聚合到 CompositeVertxRestAccessLogItemMeta
中。Parser 加载到类型为 CompositeVertxRestAccessLogItemMeta
的AccessLogItemMeta时,会调用其 getAccessLogItemMetas()
方法获得一组 AccessLogItemMeta。VertxRestAccessLogItemMeta
使用SPI机制加载,而CompositeVertxRestAccessLogItemMeta
可以让用户只在SPI配置文件中配置一条记录就加载多条meta信息,给了用户更灵活的选择。
- AccessLogItemCreator
public interface AccessLogItemCreator<T> {
// 接收配置值,返回一个AccessLogItem。如果AccessLogItem的占位符没有可变的配置值部分,则此方法会接收到null。
AccessLogItem<T> createItem(String config);
}
用户通过设置在自定义的 VertxRestAccessLogItemMeta
中的 AccessLogItemCreator
实例化自己的 AccessLogItem
。由于这是一个函数式接口,当 AccessLogItem
的初始化方式较简单时,可以直接使用 Lambda表达式定义Creator,以简化开发。
AccessLogItemMeta 的匹配规则
AccessLogItemMeta 加载进 Parser 后,会进行一次排序。Parser 解析 pattern 串时会从前到后匹配 meta list,总的匹配规则如下:
-
优先匹配高优先级的meta。
-
优先匹配有后缀的meta,当匹配上多个有后缀meta时,取前后缀相距最小的一个。
-
优先匹配占位符长的meta,例如有两个 meta,"%abc"和"%a",如果匹配中了"%abc"则直接返回,不再匹配"%a"。
示例说明
- 扩展自定义 AccessLogItem
首先用户需要 AccessLogItem
接口实现自己的 item:
public class UserDefinedAccessLogItem implements AccessLogItem<RoutingContext> {
private String config;
public UserDefinedAccessLogItem(String config) {
this.config = config;
}
@Override
public void appendServerFormattedItem(ServerAccessLogEvent accessLogEvent, StringBuilder builder) {
builder.append("user-defined--server-")
.append(accessLogEvent.getRoutingContext().response().getStatusCode())
.append("-")
.append(config);
}
@Override
public void appendClientFormattedItem(InvocationFinishEvent clientLogEvent, StringBuilder builder) {
builder.append("user-server-defined-")
.append(clientLogEvent.getResponse().getStatus())
.append("-")
.append(config);
}
}
- 定义 AccessLogItem 的 meta 类
继承 VertxRestAccessLogItemMeta
或 CompositeVertxRestAccessLogItemMeta
类,定义AccessLogItem的前后缀等信息:
public class UserDefinedCompositeExtendedAccessLogItemMeta extends CompositeVertxRestAccessLogItemMeta {
private static final List<VertxRestAccessLogItemMeta> META_LIST = new ArrayList<>();
static {
META_LIST.add(new VertxRestAccessLogItemMeta("%{", "}user-defined", UserDefinedAccessLogItem::new));
}
@Override
public List<VertxRestAccessLogItemMeta> getAccessLogItemMetas() {
return META_LIST;
}
}
- 配置SPI加载文件
在 resources/META-INF/services/
目录下定义一个名为 "org.apache.servicecomb.common.accessLog.core.parser.VertxRestAccessLogItemMeta" 的文件,将上一步中定义的meta类完整类名填写到该文件中,供Parser加载meta类。
- 配置 Access Log 的 pattern
# 服务端配置
servicecomb:
accesslog:
enabled: true ## 应用作为服务端,开启 Access log
pattern: "%{param}user-defined" ## Access log 日志格式
request:
enabled: true ## 应用作为客户端,开启 Request log
pattern: "%{param}user-defined" ## Request log 日志格式
以服务端为例, 运行服务触发Access Log打印,假设请求返回状态码是 200,则可以看到Access Log打印内容为 "user-defined--server-200-param
"。