Skip to content
项目
群组
代码片段
帮助
当前项目
正在载入...
登录 / 注册
切换导航面板
Y
yudao-cloud
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
图表
比较
统计图
议题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
日程
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
图像
聊天
创建新问题
作业
提交
问题看板
Open sidebar
hblj
yudao-cloud
Commits
88352fe0
提交
88352fe0
authored
6月 07, 2022
作者:
YunaiV
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
gateway:优化 AccessLogFilter 访问日志
上级
f4c83c01
隐藏空白字符变更
内嵌
并排
正在显示
3 个修改的文件
包含
217 行增加
和
147 行删除
+217
-147
AccessLogFilter.java
...iocoder/yudao/gateway/filter/logging/AccessLogFilter.java
+95
-123
GatewayLog.java
...a/cn/iocoder/yudao/gateway/filter/logging/GatewayLog.java
+77
-18
WebFrameworkUtils.java
...java/cn/iocoder/yudao/gateway/util/WebFrameworkUtils.java
+45
-6
没有找到文件。
yudao-gateway/src/main/java/cn/iocoder/yudao/gateway/filter/logging/AccessLogFilter.java
浏览文件 @
88352fe0
package
cn
.
iocoder
.
yudao
.
gateway
.
filter
.
logging
;
import
cn.hutool.core.util.ObjectUtil
;
import
cn.iocoder.yudao.framework.common.util.date.DateUtils
;
import
cn.iocoder.yudao.framework.common.util.json.JsonUtils
;
import
cn.iocoder.yudao.gateway.util.WebFrameworkUtils
;
import
com.alibaba.nacos.common.utils.StringUtils
;
import
lombok.extern.slf4j.Slf4j
;
import
org.reactivestreams.Publisher
;
import
org.springframework.cloud.gateway.filter.GatewayFilterChain
;
import
org.springframework.cloud.gateway.filter.GlobalFilter
;
import
org.springframework.cloud.gateway.filter.factory.rewrite.CachedBodyOutputMessage
;
import
org.springframework.cloud.gateway.route.Route
;
import
org.springframework.cloud.gateway.support.BodyInserterContext
;
import
org.springframework.cloud.gateway.support.ServerWebExchangeUtils
;
import
org.springframework.core.Ordered
;
...
...
@@ -24,7 +26,6 @@ import org.springframework.http.server.reactive.ServerHttpRequestDecorator;
import
org.springframework.http.server.reactive.ServerHttpResponse
;
import
org.springframework.http.server.reactive.ServerHttpResponseDecorator
;
import
org.springframework.stereotype.Component
;
import
org.springframework.util.MultiValueMap
;
import
org.springframework.web.reactive.function.BodyInserter
;
import
org.springframework.web.reactive.function.BodyInserters
;
import
org.springframework.web.reactive.function.server.HandlerStrategies
;
...
...
@@ -36,7 +37,6 @@ import reactor.core.publisher.Mono;
import
java.nio.charset.StandardCharsets
;
import
java.util.Date
;
import
java.util.List
;
import
java.util.Map
;
@Slf4j
@Component
...
...
@@ -52,54 +52,35 @@ public class AccessLogFilter implements GlobalFilter, Ordered {
@Override
@SuppressWarnings
(
"unchecked"
)
public
Mono
<
Void
>
filter
(
ServerWebExchange
exchange
,
GatewayFilterChain
chain
)
{
// 将 Request 中可以直接获取到的参数,设置到网关日志
ServerHttpRequest
request
=
exchange
.
getRequest
();
// 请求路径
String
requestPath
=
request
.
getPath
().
pathWithinApplication
().
value
();
Route
route
=
getGatewayRoute
(
exchange
);
// String ipAddress = WebUtils.getServerHttpRequestIpAddress(request);
String
ipAddress
=
"127.0.0.1"
;
// TODO traceId
GatewayLog
gatewayLog
=
new
GatewayLog
();
gatewayLog
.
setRoute
(
WebFrameworkUtils
.
getGatewayRoute
(
exchange
));
gatewayLog
.
setSchema
(
request
.
getURI
().
getScheme
());
gatewayLog
.
setRequestMethod
(
request
.
getMethodValue
());
gatewayLog
.
setRequestPath
(
requestPath
);
gatewayLog
.
setTargetServer
(
route
.
getId
());
gatewayLog
.
setRequestTime
(
new
Date
());
gatewayLog
.
setIp
(
ipAddress
);
gatewayLog
.
setRequestUrl
(
request
.
getURI
().
getRawPath
());
gatewayLog
.
setQueryParams
(
request
.
getQueryParams
());
gatewayLog
.
setRequestHeaders
(
request
.
getHeaders
());
gatewayLog
.
setStartTime
(
new
Date
());
gatewayLog
.
setUserIp
(
WebFrameworkUtils
.
getClientIP
(
exchange
));
// 继续 filter 过滤
MediaType
mediaType
=
request
.
getHeaders
().
getContentType
();
if
(
MediaType
.
APPLICATION_FORM_URLENCODED
.
isCompatibleWith
(
mediaType
)
||
MediaType
.
APPLICATION_JSON
.
isCompatibleWith
(
mediaType
)){
return
writeBodyLog
(
exchange
,
chain
,
gatewayLog
);
}
else
{
return
writeBasicLog
(
exchange
,
chain
,
gatewayLog
);
if
(
MediaType
.
APPLICATION_FORM_URLENCODED
.
isCompatibleWith
(
mediaType
)
||
MediaType
.
APPLICATION_JSON
.
isCompatibleWith
(
mediaType
))
{
// 适合 JSON 和 Form 提交的请求
return
filterWithRequestBody
(
exchange
,
chain
,
gatewayLog
);
}
return
filterWithoutRequestBody
(
exchange
,
chain
,
gatewayLog
);
}
private
Mono
<
Void
>
writeBasicLog
(
ServerWebExchange
exchange
,
GatewayFilterChain
chain
,
GatewayLog
accessLog
)
{
StringBuilder
builder
=
new
StringBuilder
();
MultiValueMap
<
String
,
String
>
queryParams
=
exchange
.
getRequest
().
getQueryParams
();
for
(
Map
.
Entry
<
String
,
List
<
String
>>
entry
:
queryParams
.
entrySet
())
{
builder
.
append
(
entry
.
getKey
()).
append
(
"="
).
append
(
StringUtils
.
join
(
entry
.
getValue
(),
","
));
}
accessLog
.
setRequestBody
(
builder
.
toString
());
//获取响应体
private
Mono
<
Void
>
filterWithoutRequestBody
(
ServerWebExchange
exchange
,
GatewayFilterChain
chain
,
GatewayLog
accessLog
)
{
// 包装 Response,用于记录 Response Body
ServerHttpResponseDecorator
decoratedResponse
=
recordResponseLog
(
exchange
,
accessLog
);
return
chain
.
filter
(
exchange
.
mutate
().
response
(
decoratedResponse
).
build
())
.
then
(
Mono
.
fromRunnable
(()
->
{
// 打印日志
writeAccessLog
(
accessLog
);
}));
.
then
(
Mono
.
fromRunnable
(()
->
writeAccessLog
(
accessLog
)));
// 打印日志
}
/**
* 解决 request body 只能读取一次问题,
* 参考: org.springframework.cloud.gateway.filter.factory.rewrite.ModifyRequestBodyGatewayFilterFactory
...
...
@@ -109,14 +90,12 @@ public class AccessLogFilter implements GlobalFilter, Ordered {
* @return
*/
@SuppressWarnings
(
"unchecked"
)
private
Mono
writeBodyLog
(
ServerWebExchange
exchange
,
GatewayFilterChain
chain
,
GatewayLog
gatewayLog
)
{
ServerRequest
serverRequest
=
ServerRequest
.
create
(
exchange
,
messageReaders
);
Mono
<
String
>
modifiedBody
=
serverRequest
.
bodyToMono
(
String
.
class
)
.
flatMap
(
body
->{
gatewayLog
.
setRequestBody
(
body
);
return
Mono
.
just
(
body
);
});
private
Mono
filterWithRequestBody
(
ServerWebExchange
exchange
,
GatewayFilterChain
chain
,
GatewayLog
gatewayLog
)
{
ServerRequest
serverRequest
=
ServerRequest
.
create
(
exchange
,
messageReaders
);
Mono
<
String
>
modifiedBody
=
serverRequest
.
bodyToMono
(
String
.
class
).
flatMap
(
body
->
{
gatewayLog
.
setRequestBody
(
body
);
return
Mono
.
just
(
body
);
});
// 通过 BodyInserter 插入 body(支持修改body), 避免 request body 只能获取一次
BodyInserter
bodyInserter
=
BodyInserters
.
fromPublisher
(
modifiedBody
,
String
.
class
);
...
...
@@ -128,21 +107,15 @@ public class AccessLogFilter implements GlobalFilter, Ordered {
CachedBodyOutputMessage
outputMessage
=
new
CachedBodyOutputMessage
(
exchange
,
headers
);
return
bodyInserter
.
insert
(
outputMessage
,
new
BodyInserterContext
())
.
then
(
Mono
.
defer
(()
->
{
// 重新封装请求
ServerHttpRequest
decoratedRequest
=
requestDecorate
(
exchange
,
headers
,
outputMessage
);
// 记录响应日志
ServerHttpResponseDecorator
decoratedResponse
=
recordResponseLog
(
exchange
,
gatewayLog
);
// 记录普通的
return
chain
.
filter
(
exchange
.
mutate
().
request
(
decoratedRequest
).
response
(
decoratedResponse
).
build
())
.
then
(
Mono
.
fromRunnable
(()
->
{
// 打印日志
writeAccessLog
(
gatewayLog
);
}));
}));
return
bodyInserter
.
insert
(
outputMessage
,
new
BodyInserterContext
()).
then
(
Mono
.
defer
(()
->
{
// 包装 Request,用于缓存 Request Body
ServerHttpRequest
decoratedRequest
=
requestDecorate
(
exchange
,
headers
,
outputMessage
);
// 包装 Response,用于记录 Response Body
ServerHttpResponseDecorator
decoratedResponse
=
recordResponseLog
(
exchange
,
gatewayLog
);
// 记录普通的
return
chain
.
filter
(
exchange
.
mutate
().
request
(
decoratedRequest
).
response
(
decoratedResponse
).
build
())
.
then
(
Mono
.
fromRunnable
(()
->
writeAccessLog
(
gatewayLog
)));
// 打印日志
}));
}
/**
...
...
@@ -152,93 +125,43 @@ public class AccessLogFilter implements GlobalFilter, Ordered {
* @param gatewayLog 网关日志
*/
private
void
writeAccessLog
(
GatewayLog
gatewayLog
)
{
log
.
info
(
gatewayLog
.
toString
());
}
private
Route
getGatewayRoute
(
ServerWebExchange
exchange
)
{
return
exchange
.
getAttribute
(
ServerWebExchangeUtils
.
GATEWAY_ROUTE_ATTR
);
log
.
info
(
"[writeAccessLog][日志内容:{}]"
,
JsonUtils
.
toJsonString
(
gatewayLog
));
}
/**
* 请求装饰器,重新计算 headers
* @param exchange
* @param headers
* @param outputMessage
* @return
*/
private
ServerHttpRequestDecorator
requestDecorate
(
ServerWebExchange
exchange
,
HttpHeaders
headers
,
CachedBodyOutputMessage
outputMessage
)
{
return
new
ServerHttpRequestDecorator
(
exchange
.
getRequest
())
{
@Override
public
HttpHeaders
getHeaders
()
{
long
contentLength
=
headers
.
getContentLength
();
HttpHeaders
httpHeaders
=
new
HttpHeaders
();
httpHeaders
.
putAll
(
super
.
getHeaders
());
if
(
contentLength
>
0
)
{
httpHeaders
.
setContentLength
(
contentLength
);
}
else
{
// TODO: this causes a 'HTTP/1.1 411 Length Required' // on
// httpbin.org
httpHeaders
.
set
(
HttpHeaders
.
TRANSFER_ENCODING
,
"chunked"
);
}
return
httpHeaders
;
}
@Override
public
Flux
<
DataBuffer
>
getBody
()
{
return
outputMessage
.
getBody
();
}
};
}
/**
* 记录响应日志
* 通过 DataBufferFactory 解决响应体分段传输问题。
*/
private
ServerHttpResponseDecorator
recordResponseLog
(
ServerWebExchange
exchange
,
GatewayLog
gatewayLog
)
{
ServerHttpResponse
response
=
exchange
.
getResponse
();
DataBufferFactory
bufferFactory
=
response
.
bufferFactory
();
return
new
ServerHttpResponseDecorator
(
response
)
{
@Override
public
Mono
<
Void
>
writeWith
(
Publisher
<?
extends
DataBuffer
>
body
)
{
if
(
body
instanceof
Flux
)
{
Date
responseTime
=
new
Date
();
gatewayLog
.
setResponseTime
(
responseTime
);
DataBufferFactory
bufferFactory
=
response
.
bufferFactory
();
// 计算执行时间
long
executeTime
=
(
responseTime
.
getTime
()
-
gatewayLog
.
getRequestTime
().
getTime
());
gatewayLog
.
setExecuteTime
(
executeTime
);
gatewayLog
.
setEndTime
(
new
Date
());
gatewayLog
.
setDuration
((
int
)
DateUtils
.
diff
(
gatewayLog
.
getEndTime
(),
gatewayLog
.
getStartTime
()));
// 设置其它字段
gatewayLog
.
setResponseHeaders
(
response
.
getHeaders
());
// 获取响应类型,如果是 json 就打印
String
originalResponseContentType
=
exchange
.
getAttribute
(
ServerWebExchangeUtils
.
ORIGINAL_RESPONSE_CONTENT_TYPE_ATTR
);
if
(
ObjectUtil
.
equal
(
this
.
getStatusCode
(),
HttpStatus
.
OK
)
if
(
ObjectUtil
.
equal
(
getStatusCode
(),
HttpStatus
.
OK
)
&&
StringUtils
.
isNotBlank
(
originalResponseContentType
)
&&
originalResponseContentType
.
contains
(
"application/json"
))
{
Flux
<?
extends
DataBuffer
>
fluxBody
=
Flux
.
from
(
body
);
return
super
.
writeWith
(
fluxBody
.
buffer
().
map
(
dataBuffers
->
{
// 合并多个流集合,解决返回体分段传输
DataBufferFactory
dataBufferFactory
=
new
DefaultDataBufferFactory
();
DataBuffer
join
=
dataBufferFactory
.
join
(
dataBuffers
);
byte
[]
content
=
new
byte
[
join
.
readableByteCount
()];
join
.
read
(
content
);
// 释放掉内存
DataBufferUtils
.
release
(
join
);
// 设置 response body 到网关日志
byte
[]
content
=
readContent
(
dataBuffers
);
String
responseResult
=
new
String
(
content
,
StandardCharsets
.
UTF_8
);
gatewayLog
.
setResponseBody
(
responseResult
);
gatewayLog
.
setResponseData
(
responseResult
);
// 响应
return
bufferFactory
.
wrap
(
content
);
}));
}
...
...
@@ -248,4 +171,53 @@ public class AccessLogFilter implements GlobalFilter, Ordered {
}
};
}
// ========== 参考 ModifyRequestBodyGatewayFilterFactory 中的方法 ==========
/**
* 请求装饰器,支持重新计算 headers、body 缓存
*
* @param exchange 请求
* @param headers 请求头
* @param outputMessage body 缓存
* @return 请求装饰器
*/
private
ServerHttpRequestDecorator
requestDecorate
(
ServerWebExchange
exchange
,
HttpHeaders
headers
,
CachedBodyOutputMessage
outputMessage
)
{
return
new
ServerHttpRequestDecorator
(
exchange
.
getRequest
())
{
@Override
public
HttpHeaders
getHeaders
()
{
long
contentLength
=
headers
.
getContentLength
();
HttpHeaders
httpHeaders
=
new
HttpHeaders
();
httpHeaders
.
putAll
(
super
.
getHeaders
());
if
(
contentLength
>
0
)
{
httpHeaders
.
setContentLength
(
contentLength
);
}
else
{
// TODO: this causes a 'HTTP/1.1 411 Length Required' // on
// httpbin.org
httpHeaders
.
set
(
HttpHeaders
.
TRANSFER_ENCODING
,
"chunked"
);
}
return
httpHeaders
;
}
@Override
public
Flux
<
DataBuffer
>
getBody
()
{
return
outputMessage
.
getBody
();
}
};
}
// ========== 参考 ModifyResponseBodyGatewayFilterFactory 中的方法 ==========
private
byte
[]
readContent
(
List
<?
extends
DataBuffer
>
dataBuffers
)
{
// 合并多个流集合,解决返回体分段传输
DataBufferFactory
dataBufferFactory
=
new
DefaultDataBufferFactory
();
DataBuffer
join
=
dataBufferFactory
.
join
(
dataBuffers
);
byte
[]
content
=
new
byte
[
join
.
readableByteCount
()];
join
.
read
(
content
);
// 释放掉内存
DataBufferUtils
.
release
(
join
);
return
content
;
}
}
yudao-gateway/src/main/java/cn/iocoder/yudao/gateway/filter/logging/GatewayLog.java
浏览文件 @
88352fe0
package
cn
.
iocoder
.
yudao
.
gateway
.
filter
.
logging
;
import
lombok.Data
;
import
org.springframework.cloud.gateway.route.Route
;
import
org.springframework.util.MultiValueMap
;
import
java.util.Date
;
import
java.util.Map
;
/**
* 网关的访问日志
*/
@Data
public
class
GatewayLog
{
/**访问实例*/
private
String
targetServer
;
/**请求路径*/
private
String
requestPath
;
/**请求方法*/
private
String
requestMethod
;
/**协议 */
/**
* 链路追踪编号
*/
private
String
traceId
;
/**
* 用户编号
*/
private
Long
userId
;
/**
* 用户类型
*/
private
Integer
userType
;
/**
* 路由
*
* 类似 ApiAccessLogCreateReqDTO 的 applicationName
*/
private
Route
route
;
/**
* 协议
*/
private
String
schema
;
/**请求体*/
/**
* 请求方法名
*/
private
String
requestMethod
;
/**
* 访问地址
*/
private
String
requestUrl
;
/**
* 查询参数
*/
private
MultiValueMap
<
String
,
String
>
queryParams
;
/**
* 请求体
*/
private
String
requestBody
;
/**响应体*/
private
String
responseData
;
/**请求ip*/
private
String
ip
;
/**请求时间*/
private
Date
requestTime
;
/**响应时间*/
private
Date
responseTime
;
/**执行时间*/
private
long
executeTime
;
/**
* 请求头
*/
private
MultiValueMap
<
String
,
String
>
requestHeaders
;
/**
* 用户 IP
*/
private
String
userIp
;
/**
* 响应体
*
* 类似 ApiAccessLogCreateReqDTO 的 resultCode + resultMsg
*/
private
String
responseBody
;
/**
* 响应头
*/
private
MultiValueMap
<
String
,
String
>
responseHeaders
;
/**
* 开始请求时间
*/
private
Date
startTime
;
/**
* 结束请求时间
*/
private
Date
endTime
;
/**
* 执行时长,单位:毫秒
*/
private
Integer
duration
;
}
yudao-gateway/src/main/java/cn/iocoder/yudao/gateway/util/WebFrameworkUtils.java
浏览文件 @
88352fe0
package
cn
.
iocoder
.
yudao
.
gateway
.
util
;
import
cn.hutool.core.map.MapUtil
;
import
cn.hutool.core.net.NetUtil
;
import
cn.hutool.core.util.ArrayUtil
;
import
cn.hutool.core.util.StrUtil
;
import
cn.hutool.extra.servlet.ServletUtil
;
import
cn.iocoder.yudao.framework.common.util.json.JsonUtils
;
import
cn.iocoder.yudao.module.system.api.oauth2.dto.OAuth2AccessTokenCheckRespDTO
;
import
com.fasterxml.jackson.core.JsonProcessingException
;
import
lombok.extern.slf4j.Slf4j
;
import
org.springframework.cloud.gateway.route.Route
;
import
org.springframework.cloud.gateway.support.ServerWebExchangeUtils
;
import
org.springframework.core.io.buffer.DataBufferFactory
;
import
org.springframework.http.HttpHeaders
;
import
org.springframework.http.MediaType
;
...
...
@@ -15,9 +16,6 @@ import org.springframework.http.server.reactive.ServerHttpResponse;
import
org.springframework.web.server.ServerWebExchange
;
import
reactor.core.publisher.Mono
;
import
java.util.HashMap
;
import
java.util.Map
;
/**
* Web 工具类
*
...
...
@@ -71,4 +69,45 @@ public class WebFrameworkUtils {
}));
}
/**
* 获得客户端 IP
*
* 参考 {@link ServletUtil} 的 getClientIP 方法
*
* @param exchange 请求
* @param otherHeaderNames 其它 header 名字的数组
* @return 客户端 IP
*/
public
static
String
getClientIP
(
ServerWebExchange
exchange
,
String
...
otherHeaderNames
)
{
String
[]
headers
=
{
"X-Forwarded-For"
,
"X-Real-IP"
,
"Proxy-Client-IP"
,
"WL-Proxy-Client-IP"
,
"HTTP_CLIENT_IP"
,
"HTTP_X_FORWARDED_FOR"
};
if
(
ArrayUtil
.
isNotEmpty
(
otherHeaderNames
))
{
headers
=
ArrayUtil
.
addAll
(
headers
,
otherHeaderNames
);
}
// 方式一,通过 header 获取
String
ip
;
for
(
String
header
:
headers
)
{
ip
=
exchange
.
getRequest
().
getHeaders
().
getFirst
(
header
);
if
(!
NetUtil
.
isUnknown
(
ip
))
{
return
NetUtil
.
getMultistageReverseProxyIp
(
ip
);
}
}
// 方式二,通过 remoteAddress 获取
if
(
exchange
.
getRequest
().
getRemoteAddress
()
==
null
)
{
return
null
;
}
ip
=
exchange
.
getRequest
().
getRemoteAddress
().
getHostString
();
return
NetUtil
.
getMultistageReverseProxyIp
(
ip
);
}
/**
* 获得请求匹配的 Route 路由
*
* @param exchange 请求
* @return 路由
*/
public
static
Route
getGatewayRoute
(
ServerWebExchange
exchange
)
{
return
exchange
.
getAttribute
(
ServerWebExchangeUtils
.
GATEWAY_ROUTE_ATTR
);
}
}
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论