spring-web模块包含以下对反应式Web应用程序的基本支持:
对于服务器请求处理,有两个级别的支持。
对于客户端,有一个基本的ClientHttpConnector协定,以执行具有非阻塞I / O和响应流反压力的HTTP请求,以及用于Reactor Netty和响应式Jetty HttpClient的适配器。 应用程序中使用的更高级别的WebClient基于此基本协定。
对于客户端和服务器,编解码器用于序列化和反序列化HTTP请求和响应内容。
HttpHandler是具有单个方法的简单协定,用于处理请求和响应。 它故意是最小的,它的主要也是唯一的目的是成为对不同HTTP服务器API的最小抽象。
下表描述了受支持的服务器API:
下表描述了服务器依赖性(另请参阅受支持的版本):
下面的代码段显示了如何将HttpHandler适配器与每个服务器API一起使用:
Reactor Netty
HttpHandler handler = ... ReactorHttpHandlerAdapter adapter = new ReactorHttpHandlerAdapter(handler); HttpServer.create(host, port).newHandler(adapter).block();
Undertow
HttpHandler handler = ... UndertowHttpHandlerAdapter adapter = new UndertowHttpHandlerAdapter(handler); Undertow server = Undertow.builder().addHttpListener(port, host).setHandler(adapter).build(); server.start();
Tomcat
HttpHandler handler = ... Servlet servlet = new TomcatHttpHandlerAdapter(handler); Tomcat server = new Tomcat(); File base = new File(System.getProperty("java.io.tmpdir")); Context rootContext = server.addContext("", base.getAbsolutePath()); Tomcat.addServlet(rootContext, "main", servlet); rootContext.addServletMappingDecoded("/", "main"); server.setHost(host); server.setPort(port); server.start();
Jetty
HttpHandler handler = ... Servlet servlet = new JettyHttpHandlerAdapter(handler); Server server = new Server(); ServletContextHandler contextHandler = new ServletContextHandler(server, ""); contextHandler.addServlet(new ServletHolder(servlet), "/"); contextHandler.start(); ServerConnector connector = new ServerConnector(server); connector.setHost(host); connector.setPort(port); server.addConnector(connector); server.start();
Servlet 3.1+ Container
要将其作为WAR部署到任何Servlet 3.1+容器,您可以扩展WAR并将其包括在AbstractReactiveWebInitializer中。 该类使用ServletHttpHandlerAdapter包装HttpHandler并将其注册为Servlet。
org.springframework.web.server包基于HttpHandler契约构建,以提供通用的Web API,以通过多个WebExceptionHandler,多个WebFilter和单个WebHandler组件的链来处理请求。 通过简单地指向自动检测组件的Spring ApplicationContext和/或通过向构建器注册组件,可以将该链与WebHttpHandlerBuilder放在一起。
尽管HttpHandler的目标很简单,即抽象化不同HTTP服务器的使用,但WebHandler API的目的是提供Web应用程序中常用的更广泛的功能集,例如:
Special bean types
下表列出了WebHttpHandlerBuilder可以在Spring ApplicationContext中自动检测的组件,或者可以直接向其注册的组件:
表格数据
ServerWebExchange 公开以下访问表单数据的方法:
Mono<MultiValueMap<String, String>> getFormData();
DefaultServerWebExchange使用配置的HttpMessageReader将表单数据(application / x-www-form-urlencoded)解析为MultiValueMap。 默认情况下,FormHttpMessageReader配置为由ServerCodecConfigurer Bean使用(请参阅Web Handler API)。
Multipart Data
ServerWebExchange公开以下访问多部分数据的方法:
Mono <MultiValueMap <String,Part >> getMultipartData();
DefaultServerWebExchange使用配置的HttpMessageReader <MultiValueMap <String,Part >>将multipart / form-data内容解析为MultiValueMap。 当前,Synchronoss NIO Multipart是唯一受支持的第三方库,并且是我们知道的用于非阻塞解析多部分请求的唯一库。 通过ServerCodecConfigurer bean启用它(请参阅Web Handler API)。
要以流方式解析多部分数据,可以使用从HttpMessageReader <Part>返回的Flux <Part>。 例如,在带注解的控制器中,使用@RequestPart意味着按名称对各个部分进行类似于Map的访问,因此需要完全解析多部分数据。 相反,您可以使用@RequestBody将内容解码为Flux <Part>而不收集到MultiValueMap。
Forwarded Headers
当请求通过代理(例如负载平衡器)进行处理时,主机,端口和方案可能会更改,这使得从客户端角度创建指向正确的主机,端口和方案的链接带来了挑战。
RFC 7239定义了代理可以用来提供有关原始请求的信息的HTTP转发头。还有其他非标准标头,包括X-Forwarded-Host,X-Forwarded-Port,X-Forwarded-Proto,X-Forwarded-Ssl和X-Forwarded-Prefix。
ForwardedHeaderFilter是一个Servlet过滤器,它根据转发的标头修改请求的主机,端口和方案,然后删除这些标头。
对于转发的标头,存在安全方面的考虑,因为应用程序无法知道标头是由代理添加的,还是由恶意客户端添加的。这就是为什么应配置信任边界处的代理以删除来自外部的不受信任的转发标头的原因。您还可以使用removeOnly = true配置ForwardedHeaderFilter,在这种情况下,它将删除但不使用标头。
在5.1版本中,ForwardedHeaderFilter被ForwardedHeaderTransformer弃用并取代,因此可以在创建交换之前更早地处理转发的标头。 如果仍然配置了过滤器,则将其从过滤器列表中删除,而改用ForwardedHeaderTransformer。
在WebHandler API中,可以使用WebFilter在其余过滤器和目标WebHandler的其余处理链之前和之后应用拦截样式的逻辑。 使用WebFlux Config时,注册WebFilter就像将其声明为Spring bean一样简单,并且(可选)通过在bean声明上使用@Order或实现Ordered来表达优先级。
CORS
Spring WebFlux通过控制器上的注解为CORS配置提供了细粒度的支持。 但是,当您将其与Spring Security结合使用时,我们建议您依赖内置的CorsFilter,该产品必须在Spring Security的过滤器链之前订购。
有关更多详细信息,请参见有关CORS和CORS WebFilter的部分。
在WebHandler API中,可以使用WebExceptionHandler来处理WebFilter实例链和目标WebHandler链中的异常。 使用WebFlux Config时,注册WebExceptionHandler就像将其声明为Spring bean一样简单,并且(可选)通过在bean声明上使用@Order或实现Ordered来表达优先级。
下表描述了可用的WebExceptionHandler实现:
spring-web和spring-core模块提供支持,通过具有Reactive Streams背压的非阻塞I / O,可以将字节内容与更高级别的对象之间的字节序列进行序列化和反序列化。以下介绍了此支持:
spring-core模块提供byte [],ByteBuffer,DataBuffer,Resource和String编码器和解码器实现。 spring-web模块提供了Jackson JSON,Jackson Smile,JAXB2,Protocol Buffers和其他编码器和解码器,以及仅Web的HTTP消息读取器和写入器实现,用于表单数据,多部分内容,服务器发送的事件等。
ClientCodecConfigurer和ServerCodecConfigurer通常用于配置和自定义要在应用程序中使用的编解码器。请参阅有关配置HTTP消息编解码器的部分。
Jackson JSON
存在Jackson库时,都支持JSON和二进制JSON(Smile)。
Jackson2Decoder的工作方式如下:
Jackson2Encoder的工作方式如下:
默认情况下,Jackson2Encoder和Jackson2Decoder都不支持String类型的元素。 相反,默认假设是一个字符串或一系列字符串表示要由CharSequenceEncoder呈现的序列化JSON内容。 如果您需要从Flux <String>呈现JSON数组,请使用Flux#collectToList()并对Mono <List <String >>进行编码。
Form Data
FormHttpMessageReader和FormHttpMessageWriter支持对“ application / x-www-form-urlencoded”内容进行解码和编码。
在经常需要从多个位置访问表单内容的服务器端,ServerWebExchange提供了专用的getFormData()方法,该方法通过FormHttpMessageReader解析内容,然后缓存结果以进行重复访问。请参阅WebHandler API部分中的表单数据。
一旦使用getFormData(),就无法再从请求正文中读取原始原始内容。因此,应用程序应始终通过ServerWebExchange来访问缓存的表单数据,而不是从原始请求正文中进行读取。
Multipart
MultipartHttpMessageReader和MultipartHttpMessageWriter支持对“ multipart / form-data”内容进行解码和编码。反过来,MultipartHttpMessageReader委托给另一个HttpMessageReader进行实际解析为Flux <Part>,然后将这些部分简单地收集到MultiValueMap中。目前,Synchronoss NIO Multipart用于实际解析。
在可能需要从多个位置访问多部分表单内容的服务器端,ServerWebExchange提供了专用的getMultipartData()方法,该方法通过MultipartHttpMessageReader解析内容,然后缓存结果以进行重复访问。请参阅WebHandler API部分中的多部分数据。
一旦使用getMultipartData(),就无法再从请求正文中读取原始原始内容。因此,应用程序必须始终使用getMultipartData()来重复,类似地图地访问零件,否则必须依靠SynchronossPartHttpMessageReader来一次性访问Flux <Part>。
Streaming
在流式传输到HTTP响应(例如text / event-stream,application / stream + json)时,定期发送数据很重要,这样才能尽快(而不是稍后)可靠地检测到断开连接的客户端。这样的发送可以是仅注解,空的SSE事件或任何其他可以有效充当心跳的“无操作”数据。
DataBuffer
DataBuffer是WebFlux中字节缓冲区的表示形式。该参考书的Spring Core部分在有关数据缓冲区和编解码器的部分中有更多内容。要理解的关键点是,在诸如Netty之类的某些服务器上,字节缓冲区被池化并且对引用计数进行计数,并且在消耗字节缓冲区时必须将其释放以避免内存泄漏。
WebFlux应用程序通常不需要关心此类问题,除非它们直接使用或产生数据缓冲区,而不是依赖于编解码器与更高级别的对象进行转换。或者,除非他们选择创建自定义编解码器。对于这种情况,请查看“数据缓冲区和编解码器”中的信息,尤其是有关“使用数据缓冲区”的部分。
Spring WebFlux中的DEBUG级别日志记录被设计为紧凑,最小化和人性化。它侧重于一遍又一遍有用的高价值信息,而其他信息则仅在调试特定问题时才有用。
TRACE级别的日志记录通常遵循与DEBUG相同的原则(例如,也不应成为firehose),但可用于调试任何问题。另外,某些日志消息在TRACE vs DEBUG上可能显示不同级别的详细信息。
良好的日志记录来自使用日志的经验。如果您发现任何不符合既定目标的东西,请告诉我们。
Log Id
在WebFlux中,单个请求可以在多个线程上执行,并且线程ID对于关联属于特定请求的日志消息没有用。这就是为什么WebFlux日志消息默认情况下带有特定于请求的ID的原因。
在服务器端,日志ID存储在ServerWebExchange属性(LOG_ID_ATTRIBUTE)中,而可从ServerWebExchange#getLogPrefix()获得基于该ID的全格式前缀。在WebClient端,日志ID存储在ClientRequest属性(LOG_ID_ATTRIBUTE)中,而完全格式的前缀可从ClientRequest#logPrefix()获得。
Sensitive Data
调试和跟踪日志记录可以记录敏感信息。这就是默认情况下屏蔽表单参数和标题的原因,并且必须显式启用它们的完整日志记录。
以下示例显示了如何对服务器端请求执行此操作:
@Configuration @EnableWebFlux class MyConfig implements WebFluxConfigurer { @Override public void configureHttpMessageCodecs(ServerCodecConfigurer configurer) { configurer.defaultCodecs().enableLoggingRequestDetails(true); } }
以下示例显示了如何针对客户端请求执行此操作:
Consumer<ClientCodecConfigurer> consumer = configurer -> configurer.defaultCodecs().enableLoggingRequestDetails(true); WebClient webClient = WebClient.builder() .exchangeStrategies(ExchangeStrategies.builder().codecs(consumer).build()) .build();
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8