SpringCloud升级之路2020.0.x版-14.UnderTow AccessLog 配置介绍

本文主要是介绍SpringCloud升级之路2020.0.x版-14.UnderTow AccessLog 配置介绍,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本系列代码地址:https://github.com/HashZhang/spring-cloud-scaffold/tree/master/spring-cloud-iiford

image

server:undertow:# access log相关配置accesslog:# 存放目录,默认为 logsdir: ./log# 是否开启enabled: true# 格式,各种占位符后面会详细说明pattern: '{"transportProtocol":"%{TRANSPORT_PROTOCOL}","scheme":"%{SCHEME}","protocol":"%{PROTOCOL}","method":"%{METHOD}","reqHeaderUserAgent":"%{i,User-Agent}","cookieUserId": "%{c,userId}","queryTest": "%{q,test}","queryString": "%q","relativePath": "%R, %{REQUEST_PATH}, %{RESOLVED_PATH}","requestLine": "%r","uri": "%U","thread": "%I","hostPort": "%{HOST_AND_PORT}","localIp": "%A","localPort": "%p","localServerName": "%v","remoteIp": "%a","remoteHost": "%h","bytesSent": "%b","time":"%{time,yyyy-MM-dd HH:mm:ss.S}","status":"%s","reason":"%{RESPONSE_REASON_PHRASE}","respHeaderUserSession":"%{o,userSession}","respCookieUserId":"%{resp-cookie,userId}","timeUsed":"%Dms, %Ts, %{RESPONSE_TIME}ms, %{RESPONSE_TIME_MICROS} us, %{RESPONSE_TIME_NANOS} ns",}'# 文件前缀,默认为 access_logprefix: access.# 文件后缀,默认为 logsuffix: log# 是否另起日志文件写 access log,默认为 true# 目前只能按照日期进行 rotate,一天一个日志文件rotate: true

image

Undertow 的 accesslog 处理核心类抽象是 io.undertow.server.handlers.accesslog.AccesslogReceiver。由于目前 Undertow 的 AccesslogReceiver 只有一种实现在使用,也就是 io.undertow.server.handlers.accesslog.DefaultAccessLogReceiver

查看 DefaultAccessLogReceiver 的 rotate 时机:

DefaultAccessLogReceiver

/*** 计算 rotate 时间点*/
private void calculateChangeOverPoint() {Calendar calendar = Calendar.getInstance();calendar.set(Calendar.SECOND, 0);calendar.set(Calendar.MINUTE, 0);calendar.set(Calendar.HOUR_OF_DAY, 0);//当前时间日期 + 1,即下一天calendar.add(Calendar.DATE, 1);SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd", Locale.US);currentDateString = df.format(new Date());// if there is an existing default log file, use the date last modified instead of the current dateif (Files.exists(defaultLogFile)) {try {currentDateString = df.format(new Date(Files.getLastModifiedTime(defaultLogFile).toMillis()));} catch(IOException e){// ignore. use the current date if exception happens.}}//rotate 时机是下一天的 0 点changeOverPoint = calendar.getTimeInMillis();
}

image

其实 Undertow 中的 accesslog 占位符,就是之前我们提到的 Undertow Listener 解析请求后抽象的 HTTP server exchange 的属性。

官网文档的表格并不是最全的,并且注意点并没有说明,例如某些占位符必须打开某些 Undertow 特性才能使用等等。这里我们列出下。

首先先提出一个注意点,参数占位符,例如 %{i,你要看的header值} 查看 header 的某个 key 的值。逗号后面注意不要有空格,因为这个空格会算入 key 里面导致拿不到你想要的 key

请求相关属性

描述缩写占位符全名占位符参数占位符源码
请求传输协议,等价于请求协议%{TRANSPORT_PROTOCOL}TransportProtocolAttribute
请求模式,例如 http、https 等%{SCHEME}RequestSchemeAttribute
请求协议,例如 HTTP/1.1%H%{PROTOCOL}RequestProtocolAttribute
请求方法,例如 GET、POST 等%m%{METHOD}RequestMethodAttribute
请求 Header 的某一个值%{i,你要看的header值}RequestHeaderAttribute
Cookie 的某一个值%{c,你要看的cookie值} 或者 %{req-cookie,你要看的cookie值}分别对应 CookieAttributeRequestCookieAttribute
路径参数 PathVariable 由于并没有被 Undertow 的 Listener 或者 Handler 解析处理,所以拦截不到,无法确认是否是一个 PathVariable 还是就是 url 路径。所以,PathVariable 的占位符是不会起作用的%{p, 你想查看的路径参数 key }PathParameterAttribute
请求参数,即 url 的 ? 之后键值对,这里可以选择查看某个 key 的值。%{q, 你想查看的请求参数 key}QueryParameterAttribute
请求参数字符串,即 url 的 ? 之后的所有字符}%q(不包含 ?)%{QUERY_STRING}(不包含 ?);%{BARE_QUERY_STRING}(包含 ?)QueryStringAttribute
请求相对路径(在 Spring Boot 环境下,大多数情况 RequestPath 和 RelativePath 还有 ResolvedPath 是等价的),即除去 host,port,请求参数字符串的路径%R%{RELATIVE_PATH} 或者 %{REQUEST_PATH} 或者 %{RESOLVED_PATH}分别对应 RelativePathAttributeRequestPathAttributeResolvedPathAttribute
请求整体字符串,包括请求方法,请求相对路径,请求参数字符串,请求协议,例如 Get /test?a=b HTTP/1.1%r%{REQUEST_LINE}RequestLineAttribute
请求 URI,包括请求相对路径,请求参数字符串%U%{REQUEST_URL}RequestURLAttribute
处理请求的线程%I%{THREAD_NAME}ThreadNameAttribute

注意:

  1. 路径参数 PathVariable 由于并没有被 Undertow 的 Listener 或者 Handler 解析处理,所以拦截不到,无法确认是否是一个 PathVariable 还是就是 url 路径。所以,PathVariable 的占位符是不会起作用的

请求地址相关

描述缩写占位符全名占位符参数占位符源码
host 和 port,一般就是 HTTP 请求 Header 中的 Host 值,如果 Host 为空则获取本地地址和端口,如果没获取到端口则根据协议用默认端口(http:80,,https:443)%{HOST_AND_PORT}HostAndPortAttribute
请求本地地址 IP%A%{LOCAL_IP}LocalIPAttribute
请求本地端口 Port%p%{LOCAL_PORT}LocalPortAttribute
请求本地主机名,一般就是 HTTP 请求 Header 中的 Host 值,如果 Host 为空则获取本地地址%v%{LOCAL_SERVER_NAME}LocalServerNameAttribute
请求远程主机名,通过连接获取远端的主机地址%h%{REMOTE_HOST}RemoteHostAttribute
请求远程 IP,通过连接获取远端的 IP%a%{REMOTE_IP}RemoteIPAttribute

注意:

  1. 请求的远程地址我们一般不从请求连接获取,而是通过 Http Header 里面的 X-forwarded-for 或者 X-real-ip 等获取,因为现在请求都是通过各种 VPN,负载均衡器发上来的。

响应相关属性

描述缩写占位符全名占位符参数占位符源码
发送的字节数大小,除了 Http Header 以外%b (如果为空就是 -) 或者 %B (如果为空就是 0)%{BYTES_SENT} (如果为空就是 0)BytesSentAttribute
accesslog 时间,这个不是收到请求的时间,而是响应的时间%t%{DATE_TIME}%{time, 你自定义的 java 中 SimpleDateFormat 的格式}DateTimeAttribute
HTTP 响应状态码%s%{RESPONSE_CODE}ResponseCodeAttribute
HTTP 响应原因%{RESPONSE_REASON_PHRASE}ResponseReasonPhraseAttribute
响应 Header 的某一个值%{o,你要看的header值}ResponseHeaderAttribute
响应 Cookie 的某一个值%{resp-cookie,你要看的cookie值}ResponseCookieAttribute
响应时间,默认 undertow 没有开启请求时间内统计,需要打开才能统计响应时间%D(毫秒,例如 56 代表 56ms) %T(秒,例如 5.067 代表 5.067 秒)%{RESPONSE_TIME}(等价于 %D) %{RESPONSE_TIME_MICROS} (微秒) %{RESPONSE_TIME_NANOS}(纳秒)ResponseTimeAttribute

注意:默认 undertow 没有开启请求时间内统计,需要打开才能统计响应时间,如何开启呢?通过注册一个 WebServerFactoryCustomizer 到 Spring ApplicationContext 中即可。请看下面的代码(项目地址:https://github.com/HashZhang/spring-cloud-scaffold/blob/master/spring-cloud-iiford/):

spring.factories(省略无关代码)

# AutoConfiguration
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\com.github.hashjang.spring.cloud.iiford.service.common.auto.UndertowAutoConfiguration

UndertowAutoConfiguration

//设置proxyBeanMethods=false,因为没有 @Bean 的方法互相调用需要每次返回同一个 Bean,没必要代理,关闭增加启动速度
@Configuration(proxyBeanMethods = false)
@Import(WebServerConfiguration.class)
public class UndertowAutoConfiguration {
}

WebServerConfiguration

//设置proxyBeanMethods=false,因为没有 @Bean 的方法互相调用需要每次返回同一个 Bean,没必要代理,关闭增加启动速度
@Configuration(proxyBeanMethods = false)
public class WebServerConfiguration {@Beanpublic WebServerFactoryCustomizer<ConfigurableUndertowWebServerFactory> undertowWebServerAccessLogTimingEnabler(ServerProperties serverProperties) {return new DefaultWebServerFactoryCustomizer(serverProperties);}
}

DefaultWebServerFactoryCustomizer

public class DefaultWebServerFactoryCustomizer implements WebServerFactoryCustomizer<ConfigurableUndertowWebServerFactory> {private final ServerProperties serverProperties;public DefaultWebServerFactoryCustomizer(ServerProperties serverProperties) {this.serverProperties = serverProperties;}@Overridepublic void customize(ConfigurableUndertowWebServerFactory factory) {String pattern = serverProperties.getUndertow().getAccesslog().getPattern();// 如果 accesslog 配置中打印了响应时间,则打开记录请求开始时间配置if (logRequestProcessingTiming(pattern)) {factory.addBuilderCustomizers(builder -> builder.setServerOption(UndertowOptions.RECORD_REQUEST_START_TIME, true));}}private boolean logRequestProcessingTiming(String pattern) {if (StringUtils.isBlank(pattern)) {return false;}//判断 accesslog 是否配置了查看响应时间return pattern.contains(ResponseTimeAttribute.RESPONSE_TIME_MICROS)|| pattern.contains(ResponseTimeAttribute.RESPONSE_TIME_MILLIS)|| pattern.contains(ResponseTimeAttribute.RESPONSE_TIME_NANOS)|| pattern.contains(ResponseTimeAttribute.RESPONSE_TIME_MILLIS_SHORT)|| pattern.contains(ResponseTimeAttribute.RESPONSE_TIME_SECONDS_SHORT);}
}

其他

还有安全相关的属性(SSL 相关,登录认证 Authentication 相关),微服务内部调用一般用不到,我们这里就不赘述了。
其它内置的属性,在 Spring Boot 环境下一般用不到,我们这里就不讨论了。

举例

我们最开始配置的 accesslog 的例子请求返回如下( JSON 格式化之后的结果):

{"transportProtocol": "http/1.1","scheme": "http","protocol": "HTTP/1.1","method": "GET","reqHeaderUserAgent": "PostmanRuntime/7.26.10","cookieUserId": "testRequestCookieUserId","queryTest": "1","queryString": "?test=1&query=2","relativePath": "/test, /test, -","requestLine": "GET /test?test=1&query=2 HTTP/1.1","uri": "/test","thread": "XNIO-2 task-1","hostPort": "127.0.0.1:8102","localIp": "127.0.0.1","localPort": "8102","localServerName": "127.0.0.1","remoteIp": "127.0.0.1","remoteHost": "127.0.0.1","bytesSent": "26","time": "2021-04-08 00:07:50.410","status": "200","reason": "OK","respHeaderUserSession": "testResponseHeaderUserSession","respCookieUserId": "testResponseCookieUserId","timeUsed": "3683ms, 3.683s, 3683ms, 3683149 us, 3683149200 ns",
}

我们这一节详细介绍了如何配置 Undertow 的 accesslog,将 accesslog 各种占位符都罗列了出来,用户可以根据这些信息配置出自己想要的 accesslog 信息以及格式。下一节,我们将详细介绍我们框架中针对 Undertow 的定制代码

微信搜索“我的编程喵”关注公众号,每日一刷,轻松提升技术,斩获各种offer

这篇关于SpringCloud升级之路2020.0.x版-14.UnderTow AccessLog 配置介绍的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/170573

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Linux云服务器手动配置DNS的方法步骤

《Linux云服务器手动配置DNS的方法步骤》在Linux云服务器上手动配置DNS(域名系统)是确保服务器能够正常解析域名的重要步骤,以下是详细的配置方法,包括系统文件的修改和常见问题的解决方案,需要... 目录1. 为什么需要手动配置 DNS?2. 手动配置 DNS 的方法方法 1:修改 /etc/res