Skip to content
项目
群组
代码片段
帮助
当前项目
正在载入...
登录 / 注册
切换导航面板
W
wangxiaolu-sfa-gateway
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
图表
比较
统计图
议题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
日程
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
图像
聊天
创建新问题
作业
提交
问题看板
Open sidebar
sfa
wangxiaolu-sfa-gateway
Commits
6a9b1a35
提交
6a9b1a35
authored
10月 11, 2024
作者:
李秋林
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
gateway服务添加request、response日志
上级
75d009c3
隐藏空白字符变更
内嵌
并排
正在显示
5 个修改的文件
包含
392 行增加
和
23 行删除
+392
-23
pom.xml
pom.xml
+6
-0
ConfigController.java
...ain/java/com/sfa/gateway/controller/ConfigController.java
+0
-22
AuthFilter.java
src/main/java/com/sfa/gateway/filter/AuthFilter.java
+2
-1
GatewayLogFilter.java
src/main/java/com/sfa/gateway/filter/GatewayLogFilter.java
+319
-0
GatewayLog.java
src/main/java/com/sfa/gateway/pojo/GatewayLog.java
+65
-0
没有找到文件。
pom.xml
浏览文件 @
6a9b1a35
...
...
@@ -91,6 +91,12 @@
<artifactId>
mysql-connector-java
</artifactId>
</dependency>
<dependency>
<groupId>
org.projectlombok
</groupId>
<artifactId>
lombok
</artifactId>
<optional>
true
</optional>
</dependency>
</dependencies>
<build>
...
...
src/main/java/com/sfa/gateway/controller/ConfigController.java
deleted
100644 → 0
浏览文件 @
75d009c3
package
com
.
sfa
.
gateway
.
controller
;
import
org.springframework.beans.factory.annotation.Value
;
import
org.springframework.cloud.context.config.annotation.RefreshScope
;
import
org.springframework.web.bind.annotation.GetMapping
;
import
org.springframework.web.bind.annotation.RequestMapping
;
import
org.springframework.web.bind.annotation.RestController
;
@RestController
@RequestMapping
(
"/config"
)
@RefreshScope
public
class
ConfigController
{
@Value
(
"${spring.redis.database}"
)
private
String
useLocalCache
;
@GetMapping
(
"/get"
)
public
String
get
()
{
return
useLocalCache
;
}
}
\ No newline at end of file
src/main/java/com/sfa/gateway/filter/AuthFilter.java
浏览文件 @
6a9b1a35
...
...
@@ -130,6 +130,6 @@ public class AuthFilter implements GlobalFilter, Ordered
@Override
public
int
getOrder
()
{
return
-
200
;
return
-
1
;
}
}
\ No newline at end of file
src/main/java/com/sfa/gateway/filter/GatewayLogFilter.java
0 → 100644
浏览文件 @
6a9b1a35
package
com
.
sfa
.
gateway
.
filter
;
import
com.alibaba.fastjson.JSON
;
import
com.sfa.gateway.pojo.GatewayLog
;
import
org.springframework.beans.factory.annotation.Autowired
;
import
org.springframework.cloud.gateway.filter.factory.rewrite.CachedBodyOutputMessage
;
import
org.springframework.context.ApplicationEventPublisher
;
import
org.springframework.util.ObjectUtils
;
import
reactor.core.publisher.Flux
;
import
reactor.core.publisher.Mono
;
import
java.nio.charset.StandardCharsets
;
import
java.util.*
;
import
lombok.extern.slf4j.Slf4j
;
import
org.apache.commons.lang3.StringUtils
;
import
org.reactivestreams.Publisher
;
import
org.springframework.cloud.gateway.filter.GatewayFilterChain
;
import
org.springframework.cloud.gateway.filter.GlobalFilter
;
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
;
import
org.springframework.core.io.buffer.DataBuffer
;
import
org.springframework.core.io.buffer.DataBufferFactory
;
import
org.springframework.core.io.buffer.DataBufferUtils
;
import
org.springframework.core.io.buffer.DefaultDataBufferFactory
;
import
org.springframework.http.HttpHeaders
;
import
org.springframework.http.HttpStatus
;
import
org.springframework.http.MediaType
;
import
org.springframework.http.codec.HttpMessageReader
;
import
org.springframework.http.server.reactive.ServerHttpRequest
;
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
;
import
org.springframework.web.reactive.function.server.ServerRequest
;
import
org.springframework.web.server.ServerWebExchange
;
@Slf4j
@Component
public
class
GatewayLogFilter
implements
GlobalFilter
,
Ordered
{
@Autowired
private
ApplicationEventPublisher
applicationEventPublisher
;
private
final
List
<
HttpMessageReader
<?>>
messageReaders
=
HandlerStrategies
.
withDefaults
().
messageReaders
();
private
static
final
String
CONTENT_TYPE
=
"application/json"
;
// 请求来源应用
private
static
final
String
REQUEST_ORIGIN_APP
=
"Request-Origin-App"
;
// 自定义请求头,转发之前删除自定义请求头
private
static
final
List
<
String
>
CUSTOM_HEADERS
=
Arrays
.
asList
(
"sign"
,
"timestamp"
,
"random"
,
"Request-Origin-App"
);
@Override
public
Mono
<
Void
>
filter
(
ServerWebExchange
exchange
,
GatewayFilterChain
chain
)
{
ServerHttpRequest
request
=
exchange
.
getRequest
();
// 请求路径
String
requestPath
=
request
.
getPath
().
pathWithinApplication
().
value
();
// 获取路由信息
Route
route
=
getGatewayRoute
(
exchange
);
// String ipAddress = IpUtil.getIpAddr(request);
GatewayLog
gatewayLog
=
new
GatewayLog
();
gatewayLog
.
setOrigin
(
request
.
getHeaders
().
getFirst
(
REQUEST_ORIGIN_APP
));
gatewayLog
.
setSchema
(
request
.
getURI
().
getScheme
());
gatewayLog
.
setRequestMethod
(
request
.
getMethodValue
());
gatewayLog
.
setRequestPath
(
requestPath
);
gatewayLog
.
setTargetServer
(
route
.
getUri
().
toString
());
gatewayLog
.
setStartTime
(
System
.
currentTimeMillis
());
// gatewayLog.setIp(ipAddress);
gatewayLog
.
setRouteConfig
(
JSON
.
toJSONString
(
route
));
Map
<
String
,
Object
>
headers
=
new
HashMap
<>();
for
(
String
key
:
request
.
getHeaders
().
keySet
())
{
headers
.
put
(
key
,
request
.
getHeaders
().
getFirst
(
key
));
}
gatewayLog
.
setHeaders
(
JSON
.
toJSONString
(
headers
));
MediaType
mediaType
=
request
.
getHeaders
().
getContentType
();
if
(
request
.
getHeaders
().
getContentType
()
!=
null
)
{
gatewayLog
.
setRequestContentType
(
request
.
getHeaders
().
getContentType
().
toString
());
}
if
(
mediaType
!=
null
&&
(
MediaType
.
APPLICATION_FORM_URLENCODED
.
isCompatibleWith
(
mediaType
)
||
MediaType
.
APPLICATION_JSON
.
isCompatibleWith
(
mediaType
))){
return
writeBodyLog
(
exchange
,
chain
,
gatewayLog
);
}
else
{
return
writeBasicLog
(
exchange
,
chain
,
gatewayLog
);
}
}
@Override
public
int
getOrder
()
{
// 必须小于等于-2,否则无法获取相应结果
return
-
2
;
}
/**
* 获取路由信息
* @param exchange
* @return
*/
private
Route
getGatewayRoute
(
ServerWebExchange
exchange
)
{
return
exchange
.
getAttribute
(
ServerWebExchangeUtils
.
GATEWAY_ROUTE_ATTR
);
}
private
Mono
<
Void
>
writeBasicLog
(
ServerWebExchange
exchange
,
GatewayFilterChain
chain
,
GatewayLog
gatewayLog
)
{
StringBuilder
builder
=
new
StringBuilder
();
MultiValueMap
<
String
,
String
>
queryParams
=
exchange
.
getRequest
().
getQueryParams
();
gatewayLog
.
setRequestBody
(
getUrlParamsByMap
(
queryParams
));
// 修改Header
ServerHttpRequest
mutableReq
=
exchange
.
getRequest
().
mutate
().
headers
(
httpHeaders
->
{
// 删除自定义header
for
(
String
customHeader
:
CUSTOM_HEADERS
)
{
httpHeaders
.
remove
(
customHeader
);
}
}).
build
();
//获取响应体
ServerHttpResponseDecorator
decoratedResponse
=
recordResponseLog
(
exchange
,
gatewayLog
);
return
chain
.
filter
(
exchange
.
mutate
().
request
(
mutableReq
).
response
(
decoratedResponse
).
build
())
.
then
(
Mono
.
fromRunnable
(()
->
{
// 打印日志
writeAccessLog
(
gatewayLog
);
}));
}
/**
* 解决 request body 只能读取一次问题,
* 参考: org.springframework.cloud.gateway.filter.factory.rewrite.ModifyRequestBodyGatewayFilterFactory
* @param exchange
* @param chain
* @param gatewayLog
* @return
*/
private
Mono
<
Void
>
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
);
});
// 通过 BodyInserter 插入 body(支持修改body), 避免 request body 只能获取一次
BodyInserter
bodyInserter
=
BodyInserters
.
fromPublisher
(
modifiedBody
,
String
.
class
);
HttpHeaders
headers
=
new
HttpHeaders
();
headers
.
putAll
(
exchange
.
getRequest
().
getHeaders
());
// the new content type will be computed by bodyInserter
// and then set in the request decorator
headers
.
remove
(
HttpHeaders
.
CONTENT_LENGTH
);
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
);
}));
}));
}
/**
* 打印日志
* @param gatewayLog 网关日志
*/
private
void
writeAccessLog
(
GatewayLog
gatewayLog
)
{
// applicationEventPublisher.publishEvent(gatewayLog);
log
.
info
(
"------------ 请求响应 start ------------"
);
log
.
info
(
"请求路径:{}:{}{}"
,
gatewayLog
.
getRequestMethod
(),
gatewayLog
.
getTargetServer
(),
gatewayLog
.
getRequestPath
());
log
.
info
(
"Header配置:{}"
,
gatewayLog
.
getHeaders
());
log
.
info
(
"请求参数:{}"
,
gatewayLog
.
getRequestBody
());
log
.
info
(
"响应信息:{}"
,
gatewayLog
.
getResponseData
());
log
.
info
(
"------------ 请求结束 {} end ------------"
,
gatewayLog
.
getResponseTime
());
}
/**
* 请求装饰器,重新计算 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"
);
}
// 删除自定义header
for
(
String
customHeader
:
CUSTOM_HEADERS
)
{
headers
.
remove
(
customHeader
);
}
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
.
setEndTime
(
responseTime
.
getTime
());
// 计算执行时间
long
executeTime
=
(
responseTime
.
getTime
()
-
gatewayLog
.
getStartTime
());
gatewayLog
.
setExecuteTime
(
executeTime
);
gatewayLog
.
setStatus
(
response
.
getStatusCode
().
value
()
==
200
?
"成功"
:
"失败"
);
// 获取响应类型,如果是 json 就打印
String
originalResponseContentType
=
exchange
.
getAttribute
(
ServerWebExchangeUtils
.
ORIGINAL_RESPONSE_CONTENT_TYPE_ATTR
);
if
(
Objects
.
equals
(
this
.
getStatusCode
(),
HttpStatus
.
OK
)
&&
StringUtils
.
isNotBlank
(
originalResponseContentType
)
&&
originalResponseContentType
.
contains
(
CONTENT_TYPE
))
{
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
);
String
responseResult
=
new
String
(
content
,
StandardCharsets
.
UTF_8
);
gatewayLog
.
setResponseData
(
responseResult
);
return
bufferFactory
.
wrap
(
content
);
}));
}
}
// if body is not a flux. never got there.
return
super
.
writeWith
(
body
);
}
};
}
/**
* 将map参数转换成url参数
* @param map
* @return
*/
private
String
getUrlParamsByMap
(
MultiValueMap
<
String
,
String
>
map
)
{
if
(
ObjectUtils
.
isEmpty
(
map
))
{
return
""
;
}
StringBuilder
sb
=
new
StringBuilder
();
for
(
Map
.
Entry
<
String
,
List
<
String
>>
entry
:
map
.
entrySet
())
{
sb
.
append
(
entry
.
getKey
()).
append
(
"="
).
append
(
entry
.
getValue
().
get
(
0
));
sb
.
append
(
"&"
);
}
String
s
=
sb
.
toString
();
if
(
s
.
endsWith
(
"&"
))
{
s
=
StringUtils
.
substringBeforeLast
(
s
,
"&"
);
}
return
s
;
}
}
\ No newline at end of file
src/main/java/com/sfa/gateway/pojo/GatewayLog.java
0 → 100644
浏览文件 @
6a9b1a35
package
com
.
sfa
.
gateway
.
pojo
;
import
lombok.*
;
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public
class
GatewayLog
{
/**请求来源**/
private
String
origin
;
/**访问实例*/
private
String
targetServer
;
/**请求路径*/
private
String
requestPath
;
/**请求方法*/
private
String
requestMethod
;
/**协议 */
private
String
schema
;
/**请求类型 */
private
String
requestContentType
;
/**请求头 */
private
String
headers
;
/**请求体*/
private
String
requestBody
;
/**响应体*/
private
String
responseData
;
/**请求ip*/
private
String
ip
;
/**IP所属城市*/
private
String
city
;
/**开始时间*/
private
Long
startTime
;
/**结束时间*/
private
Long
endTime
;
/**请求时间*/
private
String
requestTime
;
/**响应时间*/
private
String
responseTime
;
/**执行时间*/
private
long
executeTime
;
/**路由配置*/
private
String
routeConfig
;
/**响应状态*/
private
String
status
;
}
\ No newline at end of file
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论