Spring-Cloud Zuul 破坏了转发的多部分请求文件名中的 UTF-8 符号
Spring-Cloud Zuul breaks UTF-8 symbols in forwarded multipart request filename
这是我第一次在 SO 上,所以请耐心等待我的第一个问题。
我想我遇到了某种配置问题,但经过一天的实验后我被卡住了。我们的应用程序基于 Spring-Cloud [Brixton 发布]。我们有这样的配置:Portal(服务于 angular-based web-ui 的 Web 应用程序),它有 zuul 代理,单个路由配置到我们的网关服务,像这样:
zuul:
ignoredServices: '*'
prefix: /api
routes:
api-proxy:
path: /**
serviceId: api-gateway
配置了另一个 Zuul 并将请求中继到内部业务逻辑服务:
zuul:
ignoredServices: '*'
routes:
service1:
path: /service1/**
serviceId: service1
service2:
path: /service2/**
serviceId: service2
所有这些配置都没有问题。
我现在面临的问题是文件上传多部分请求。更准确地说 - 那些多部分请求,当要上传的文件具有来自 UTF-8
的非拉丁符号(例如 ąčęėįš)时。当请求到达必须处理 @RequestPart MultipartFile file
的服务时,则在上述符号的位置添加 file.getOriginalFilename()
returns 问号。现在,我尝试直接将此类文件上传到此类控制器,并且文件名没有问号,即没有损坏,这表明,当代理中继进入时,Zuul 过滤器中某处发生了一些错误的 interpretation/parsing 多部分请求请求。
也许有人对 Zuul 有类似的经验,可以指导我一些方法来解决这个问题?
我自己 运行 遇到了同样的问题,并创建了以下问题:
https://jira.spring.io/browse/SPR-15396
希望这在 Spring 4.3.8 中可以配置。
同时,您必须创建一个 FormBodyWrapperFilter
类型的 bean(覆盖 ZuulConfiguration
中的那个)。在构造函数中,您传递了 FormHttpMessageConverter
的副本,它从 FormHttpMessageConverter
扩展而来,并将 FormHttpMessageConverter.MultipartHttpOutputMessage#getAsciiBytes(String)
中使用的编码更改为 UTF-8(您可能还想删除对javax-mail,除非你在类路径上有它)。您需要最新版本的 Spring Cloud Netflix 才能执行此操作。
示例:
@Bean
FormBodyWrapperFilter formBodyWrapperFilter() {
return new FormBodyWrapperFilter(new MyFormHttpMessageConverter());
}
然后你创建一个FormHttpMessageConverter
的副本,并改变下面的方法:
private byte[] getAsciiBytes(String name) {
try {
// THIS IS THE ONLY MODIFICATION:
return name.getBytes("UTF-8");
} catch (UnsupportedEncodingException ex) {
// Should not happen - US-ASCII is always supported.
throw new IllegalStateException(ex);
}
}
即使使用较新的版本修改响应后仍然存在问题
使用 spring 启动 2.3.8.RELEASE
通过强制执行以下 spring 属性
设法修复它
server.servlet.encoding.force= true
server.servlet.encoding.charset= UTF-8
这是我第一次在 SO 上,所以请耐心等待我的第一个问题。
我想我遇到了某种配置问题,但经过一天的实验后我被卡住了。我们的应用程序基于 Spring-Cloud [Brixton 发布]。我们有这样的配置:Portal(服务于 angular-based web-ui 的 Web 应用程序),它有 zuul 代理,单个路由配置到我们的网关服务,像这样:
zuul:
ignoredServices: '*'
prefix: /api
routes:
api-proxy:
path: /**
serviceId: api-gateway
配置了另一个 Zuul 并将请求中继到内部业务逻辑服务:
zuul:
ignoredServices: '*'
routes:
service1:
path: /service1/**
serviceId: service1
service2:
path: /service2/**
serviceId: service2
所有这些配置都没有问题。
我现在面临的问题是文件上传多部分请求。更准确地说 - 那些多部分请求,当要上传的文件具有来自 UTF-8
的非拉丁符号(例如 ąčęėįš)时。当请求到达必须处理 @RequestPart MultipartFile file
的服务时,则在上述符号的位置添加 file.getOriginalFilename()
returns 问号。现在,我尝试直接将此类文件上传到此类控制器,并且文件名没有问号,即没有损坏,这表明,当代理中继进入时,Zuul 过滤器中某处发生了一些错误的 interpretation/parsing 多部分请求请求。
也许有人对 Zuul 有类似的经验,可以指导我一些方法来解决这个问题?
我自己 运行 遇到了同样的问题,并创建了以下问题:
https://jira.spring.io/browse/SPR-15396
希望这在 Spring 4.3.8 中可以配置。
同时,您必须创建一个 FormBodyWrapperFilter
类型的 bean(覆盖 ZuulConfiguration
中的那个)。在构造函数中,您传递了 FormHttpMessageConverter
的副本,它从 FormHttpMessageConverter
扩展而来,并将 FormHttpMessageConverter.MultipartHttpOutputMessage#getAsciiBytes(String)
中使用的编码更改为 UTF-8(您可能还想删除对javax-mail,除非你在类路径上有它)。您需要最新版本的 Spring Cloud Netflix 才能执行此操作。
示例:
@Bean
FormBodyWrapperFilter formBodyWrapperFilter() {
return new FormBodyWrapperFilter(new MyFormHttpMessageConverter());
}
然后你创建一个FormHttpMessageConverter
的副本,并改变下面的方法:
private byte[] getAsciiBytes(String name) {
try {
// THIS IS THE ONLY MODIFICATION:
return name.getBytes("UTF-8");
} catch (UnsupportedEncodingException ex) {
// Should not happen - US-ASCII is always supported.
throw new IllegalStateException(ex);
}
}
即使使用较新的版本修改响应后仍然存在问题
使用 spring 启动 2.3.8.RELEASE
通过强制执行以下 spring 属性
server.servlet.encoding.force= true
server.servlet.encoding.charset= UTF-8