1. 声明式 REST 客户端:Feign
Feign 是一个声明性 Web 服务客户端。
它使编写 Web 服务客户端变得更加容易。
要使用 Feign,请创建一个接口并对其进行注释。
它具有可插拔的注释支持,包括 Feign 注释和 JAX-RS 注释。
Feign 还支持可插拔编码器和解码器。
Spring Cloud 增加了对 Spring MVC 注释和使用相同的HttpMessageConverters
在 Spring Web 中默认使用。
Spring Cloud 集成了 Eureka、Spring Cloud CircuitBreaker 以及 Spring Cloud LoadBalancer,以便在使用 Feign 时提供负载均衡的 http 客户端。
1.1. 如何包含 feign
要将 Feign 包含在您的项目中,请使用 starter with grouporg.springframework.cloud
和工件 IDspring-cloud-starter-openfeign
.有关使用当前 Spring Cloud Release Train 设置构建系统的详细信息,请参阅 Spring Cloud 项目页面。
Spring Boot 应用程序示例
@SpringBootApplication
@EnableFeignClients
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
@FeignClient("stores")
public interface StoreClient {
@RequestMapping(method = RequestMethod.GET, value = "/stores")
List<Store> getStores();
@RequestMapping(method = RequestMethod.GET, value = "/stores")
Page<Store> getStores(Pageable pageable);
@RequestMapping(method = RequestMethod.POST, value = "/stores/{storeId}", consumes = "application/json")
Store update(@PathVariable("storeId") Long storeId, Store store);
@RequestMapping(method = RequestMethod.DELETE, value = "/stores/{storeId:\\d+}")
void delete(@PathVariable Long storeId);
}
在@FeignClient
annotation String 值(上面的 “stores”)是任意的客户端名称,用于创建 Spring Cloud LoadBalancer 客户端。
您还可以使用url
属性
(绝对值或只是一个主机名)。Bean 在
Application Context 是接口的完全限定名称。
要指定您自己的别名值,您可以使用qualifiers
价值
的@FeignClient
注解。
上面的负载均衡器客户端将需要发现物理地址
对于 “stores” 服务。如果您的应用程序是 Eureka 客户端,则
它将解析 Eureka 服务注册表中的服务。如果你
不想使用 Eureka,可以配置服务器列表
在外部配置中使用SimpleDiscoveryClient
.
Spring Cloud OpenFeign 支持 Spring Cloud LoadBalancer 的阻塞模式可用的所有功能。您可以在项目文档中阅读有关它们的更多信息。
要使用@EnableFeignClients 注释开启@Configuration -annotated-classes 中,请确保指定客户端所在的位置,例如:@EnableFeignClients(basePackages = "com.example.clients") 或明确列出它们:@EnableFeignClients(clients = InventoryServiceFeignClient.class)
|
1.2. 覆盖 Feign 默认值
Spring Cloud 的 Feign 支持中的一个中心概念是命名客户端的概念。每个假客户端都是组件集合的一部分,这些组件协同工作以按需联系远程服务器,并且集合具有一个名称,您作为应用程序开发人员使用@FeignClient
注解。Spring Cloud 创建一个新的 ensemble 作为ApplicationContext
按需为每个指定客户端使用FeignClientsConfiguration
.这包含(除其他外)一个feign.Decoder
一个feign.Encoder
和feign.Contract
.
可以使用contextId
属性的@FeignClient
注解。
Spring Cloud 允许您通过声明额外的配置(在FeignClientsConfiguration
) 使用@FeignClient
.例:
@FeignClient(name = "stores", configuration = FooConfiguration.class)
public interface StoreClient {
//..
}
在这种情况下,客户端由已有的FeignClientsConfiguration
与任何 IN 一起FooConfiguration
(其中后者将覆盖前者)。
FooConfiguration 不需要使用@Configuration .但是,如果是,请注意将其从任何@ComponentScan 否则将包含此配置,因为它将成为feign.Decoder ,feign.Encoder ,feign.Contract 等。这可以通过将其放在一个单独的、不重叠的包中来避免@ComponentScan 或@SpringBootApplication ,或者可以在@ComponentScan . |
用contextId 属性的@FeignClient 注解,以及更改
这ApplicationContext ensemble 的 API 中,它将覆盖客户端名称的别名
它将用作为该客户端创建的配置 Bean 名称的一部分。 |
以前,使用url 属性,不需要name 属性。用name 现在是必需的。 |
占位符在name
和url
属性。
@FeignClient(name = "${feign.name}", url = "${feign.url}")
public interface StoreClient {
//..
}
Spring Cloud OpenFeign 默认为 feign (BeanType
beanName 的 Bean 名称:ClassName
):
-
Decoder
feignDecoder 的ResponseEntityDecoder
(它将SpringDecoder
) -
Encoder
feignEncoder 的SpringEncoder
-
Logger
feignLogger 中:Slf4jLogger
-
MicrometerCapability
micrometerCapability:如果feign-micrometer
位于 Classpath 上,并且MeterRegistry
可用 -
CachingCapability
cachingCapability:如果@EnableCaching
annotation 的 intent 中。可以通过以下方式禁用feign.cache.enabled
. -
Contract
feignContract 的SpringMvcContract
-
Feign.Builder
feignBuilder 中:FeignCircuitBreaker.Builder
-
Client
feignClient:如果 Spring Cloud LoadBalancer 在 Classpath 上,则FeignBlockingLoadBalancerClient
被使用。 如果它们都不在 Classpath 上,则使用默认的假客户端。
spring-cloud-starter-openfeign 支持spring-cloud-starter-loadbalancer .但是,与可选依赖项一样,如果要使用它,则需要确保已将其添加到项目中。 |
OkHttpClient 和 ApacheHttpClient 以及 ApacheHC5 假客户端可以通过设置feign.okhttp.enabled
或feign.httpclient.enabled
或feign.httpclient.hc5.enabled
自true
,并将它们放在 Classpath 上。
您可以通过提供以下任一 bean 来自定义使用的 HTTP 客户端org.apache.http.impl.client.CloseableHttpClient
使用 Apache 或okhttp3.OkHttpClient
当使用 OK HTTP 或org.apache.hc.client5.http.impl.classic.CloseableHttpClient
使用 Apache HC5 时。
Spring Cloud OpenFeign 默认情况下不为 feign 提供以下 bean,但仍从应用程序上下文中查找这些类型的 bean 以创建 feign 客户端:
-
Logger.Level
-
Retryer
-
ErrorDecoder
-
Request.Options
-
Collection<RequestInterceptor>
-
SetterFactory
-
QueryMapEncoder
-
Capability
(MicrometerCapability
和CachingCapability
默认提供)
一个 beanRetryer.NEVER_RETRY
将Retryer
,这将禁用重试。
请注意,此重试行为与 Feign 默认行为不同,后者将自动重试 IOExceptions。
将它们视为与网络相关的暂时性异常,以及从 ErrorDecoder 引发的任何 RetryableException。
创建其中一种类型的 Bean 并将其放置在@FeignClient
配置(例如FooConfiguration
above) 允许您覆盖所描述的每个 bean。例:
@Configuration
public class FooConfiguration {
@Bean
public Contract feignContract() {
return new feign.Contract.Default();
}
@Bean
public BasicAuthRequestInterceptor basicAuthRequestInterceptor() {
return new BasicAuthRequestInterceptor("user", "password");
}
}
这将替换SpringMvcContract
跟feign.Contract.Default
并添加一个RequestInterceptor
到 的集合RequestInterceptor
.
@FeignClient
也可以使用 Configuration Properties 进行配置。
application.yml
feign:
client:
config:
feignName:
connectTimeout: 5000
readTimeout: 5000
loggerLevel: full
errorDecoder: com.example.SimpleErrorDecoder
retryer: com.example.SimpleRetryer
defaultQueryParameters:
query: queryValue
defaultRequestHeaders:
header: headerValue
requestInterceptors:
- com.example.FooRequestInterceptor
- com.example.BarRequestInterceptor
decode404: false
encoder: com.example.SimpleEncoder
decoder: com.example.SimpleDecoder
contract: com.example.SimpleContract
capabilities:
- com.example.FooCapability
- com.example.BarCapability
queryMapEncoder: com.example.SimpleQueryMapEncoder
metrics.enabled: false
feignName
在此示例中,指的是@FeignClient
value
,该 ID 也与@FeignClient
name
和@FeignClient
contextId
.在负载均衡方案中,它还对应于serviceId
的 Server 应用程序。为 decoders、retryer 和其他类指定的类必须在 Spring 上下文中具有 bean 或具有默认构造函数。
默认配置可以在@EnableFeignClients
属性defaultConfiguration
以与上述类似的方式。区别在于此配置将应用于所有 feign 客户端。
如果您更喜欢使用配置属性来配置所有@FeignClient
中,您可以使用default
假名。
您可以使用feign.client.config.feignName.defaultQueryParameters
和feign.client.config.feignName.defaultRequestHeaders
指定将随名为feignName
.
application.yml
feign:
client:
config:
default:
connectTimeout: 5000
readTimeout: 5000
loggerLevel: basic
如果我们同时创建@Configuration
bean 和 configuration 属性,则 configuration 属性将胜出。
它将覆盖@Configuration
值。但是,如果您想将优先级更改为@Configuration
,
您可以更改feign.client.default-to-properties
自false
.
如果我们想创建多个具有相同名称或 url 的假客户端
这样它们就会指向同一个服务器,但每个服务器都有不同的自定义配置
我们必须使用contextId
属性的@FeignClient
为了避免姓名
这些配置 bean 的冲突。
@FeignClient(contextId = "fooClient", name = "stores", configuration = FooConfiguration.class)
public interface FooClient {
//..
}
@FeignClient(contextId = "barClient", name = "stores", configuration = BarConfiguration.class)
public interface BarClient {
//..
}
也可以将 FeignClient 配置为不从父上下文继承 bean。
您可以通过覆盖inheritParentConfiguration()
在FeignClientConfigurer
要返回的 beanfalse
:
@Configuration
public class CustomConfiguration {
@Bean
public FeignClientConfigurer feignClientConfigurer() {
return new FeignClientConfigurer() {
@Override
public boolean inheritParentConfiguration() {
return false;
}
};
}
}
默认情况下,Feign 客户端不对斜杠字符进行编码。您可以通过设置/ feign.client.decodeSlash 自false . |
1.2.1.SpringEncoder
配置
在SpringEncoder
我们提供的,我们设定的null
charset 用于二进制内容类型,而 charset 用于二进制内容类型UTF-8
对于所有其他的。
您可以修改此行为,从Content-Type
header charset 来代替,方法是将feign.encoder.charset-from-content-type
自true
.
1.3. 超时处理
我们可以在默认客户端和命名客户端上配置超时。OpenFeign 使用两个超时参数:
-
connectTimeout
防止由于服务器处理时间长而阻止调用方。 -
readTimeout
从建立连接时开始应用,并在返回响应时间过长时触发。
如果服务器未运行或不可用,则数据包会导致连接被拒绝。通信以错误消息或回退结束。这可能发生在connectTimeout 如果设置得非常低。执行查找和接收此类数据包所花费的时间会导致此延迟的很大一部分。它可能会根据涉及 DNS 查找的远程主机而更改。 |
1.4. 手动创建假客户端
在某些情况下,可能需要以一种不是 可以使用上述方法。在这种情况下,您可以使用 Feign Builder API 创建客户端。下面是一个示例 这会创建两个具有相同接口的假客户端,但为每个客户端配置一个 一个单独的请求拦截器。
@Import(FeignClientsConfiguration.class)
class FooController {
private FooClient fooClient;
private FooClient adminClient;
@Autowired
public FooController(Client client, Encoder encoder, Decoder decoder, Contract contract, MicrometerCapability micrometerCapability) {
this.fooClient = Feign.builder().client(client)
.encoder(encoder)
.decoder(decoder)
.contract(contract)
.addCapability(micrometerCapability)
.requestInterceptor(new BasicAuthRequestInterceptor("user", "user"))
.target(FooClient.class, "https://PROD-SVC");
this.adminClient = Feign.builder().client(client)
.encoder(encoder)
.decoder(decoder)
.contract(contract)
.addCapability(micrometerCapability)
.requestInterceptor(new BasicAuthRequestInterceptor("admin", "admin"))
.target(FooClient.class, "https://PROD-SVC");
}
}
在上面的例子中FeignClientsConfiguration.class 是默认配置
由 Spring Cloud OpenFeign 提供。 |
PROD-SVC 是 Client 将向其发出请求的服务的名称。 |
假动作Contract object 定义哪些 annotation 和 values 在接口上有效。这
自动接线Contract bean 支持 SpringMVC 注解,而不是
默认的 Feign 本机注释。 |
您还可以使用Builder`to configure FeignClient not to inherit beans from the parent context.
You can do this by overriding calling `inheritParentContext(false)
在Builder
.
1.5. Feign Spring Cloud CircuitBreaker 支持
如果 Spring Cloud CircuitBreaker在 Classpath 上,并且feign.circuitbreaker.enabled=true
,Feign 将用断路器包装所有方法。
要在每个客户端上禁用 Spring Cloud CircuitBreaker 支持,请创建一个 vanillaFeign.Builder
使用 “prototype” 范围,例如:
@Configuration
public class FooConfiguration {
@Bean
@Scope("prototype")
public Feign.Builder feignBuilder() {
return Feign.builder();
}
}
断路器名称遵循此模式<feignClientClassName>#<calledMethod>(<parameterTypes>)
.调用@FeignClient
跟FooClient
interface 和被调用的 interface 方法没有参数bar
则断路器名称将为FooClient#bar()
.
自 2020.0.2 起,熔断器名称模式已从<feignClientName>_<calledMethod> .
用CircuitBreakerNameResolver 在 2020.0.4 中引入,断路器名称可以保留旧模式。 |
提供 beanCircuitBreakerNameResolver
中,您可以更改断路器名称模式。
@Configuration
public class FooConfiguration {
@Bean
public CircuitBreakerNameResolver circuitBreakerNameResolver() {
return (String feignClientName, Target<?> target, Method method) -> feignClientName + "_" + method.getName();
}
}
要启用 Spring Cloud CircuitBreaker 组,请将feign.circuitbreaker.group.enabled
property 设置为true
(默认情况下false
).
1.6. 使用配置属性配置 CircuitBreaker
您可以通过配置属性配置 CircuitBreaker。待办事项集feign.circuitbreaker.alphanumeric-ids.enabled
自true
.因为
您不能使用、 、#
(
)
,
在 Configuration property names 中,我们需要
更改 OpenFeign 生成的断路器 ID 的命名规则。以上内容
property 将为您执行此作。
例如,如果您有这个 Feign 客户端
@FeignClient(url = "http://localhost:8080")
public interface DemoClient {
@GetMapping("demo")
String getDemo();
}
您可以通过执行以下作,使用配置属性对其进行配置
feign:
circuitbreaker:
enabled: true
alphanumeric-ids:
enabled: true
resilience4j:
circuitbreaker:
instances:
DemoClientgetDemo:
minimumNumberOfCalls: 69
timelimiter:
instances:
DemoClientgetDemo:
timeoutDuration: 10s
1.7. 假装 Spring Cloud CircuitBreaker 回退
Spring Cloud CircuitBreaker 支持回退的概念:当电路打开或出现错误时执行的默认代码路径。要为给定的@FeignClient
将fallback
属性设置为实现回退的类名。您还需要将实现声明为 Spring Bean。
@FeignClient(name = "test", url = "http://localhost:${server.port}/", fallback = Fallback.class)
protected interface TestClient {
@RequestMapping(method = RequestMethod.GET, value = "/hello")
Hello getHello();
@RequestMapping(method = RequestMethod.GET, value = "/hellonotfound")
String getException();
}
@Component
static class Fallback implements TestClient {
@Override
public Hello getHello() {
throw new NoFallbackAvailableException("Boom!", new RuntimeException());
}
@Override
public String getException() {
return "Fixed response";
}
}
如果需要访问触发回退触发器的原因,可以使用fallbackFactory
属性@FeignClient
.
@FeignClient(name = "testClientWithFactory", url = "http://localhost:${server.port}/",
fallbackFactory = TestFallbackFactory.class)
protected interface TestClientWithFactory {
@RequestMapping(method = RequestMethod.GET, value = "/hello")
Hello getHello();
@RequestMapping(method = RequestMethod.GET, value = "/hellonotfound")
String getException();
}
@Component
static class TestFallbackFactory implements FallbackFactory<FallbackWithFactory> {
@Override
public FallbackWithFactory create(Throwable cause) {
return new FallbackWithFactory();
}
}
static class FallbackWithFactory implements TestClientWithFactory {
@Override
public Hello getHello() {
throw new NoFallbackAvailableException("Boom!", new RuntimeException());
}
@Override
public String getException() {
return "Fixed response";
}
}
1.8. Feign 和@Primary
当将 Feign 与 Spring Cloud CircuitBreaker 回退一起使用时,其中ApplicationContext
相同类型。这将导致@Autowired
不起作用,因为没有一个 bean,或者一个标记为 primary。为了解决这个问题,Spring Cloud OpenFeign 将所有 Feign 实例标记为@Primary
,因此 Spring Framework 将知道要注入哪个 bean。在某些情况下,这可能并不可取。要关闭此行为,请将primary
属性@FeignClient
设置为 false。
@FeignClient(name = "hello", primary = false)
public interface HelloClient {
// methods here
}
1.9. 假继承支持
Feign 通过单继承接口支持样板 API。 这允许将常见作分组到方便的 base interfaces 中。
public interface UserService {
@RequestMapping(method = RequestMethod.GET, value ="/users/{id}")
User getUser(@PathVariable("id") long id);
}
@RestController
public class UserResource implements UserService {
}
package project.user;
@FeignClient("users")
public interface UserClient extends UserService {
}
@FeignClient 不应在服务器和客户端之间共享接口,并添加@FeignClient 接口与@RequestMapping 不再支持 on class 级别。 |
1.10. 假装请求 / 响应压缩
您可以考虑为您的 假装请求。您可以通过启用以下属性之一来执行此作:
feign.compression.request.enabled=true
feign.compression.response.enabled=true
假请求压缩为您提供的设置类似于您为 Web 服务器设置的设置:
feign.compression.request.enabled=true
feign.compression.request.mime-types=text/xml,application/xml,application/json
feign.compression.request.min-request-size=2048
这些属性允许您选择压缩媒体类型和最小请求阈值长度。
由于 OkHttpClient 使用“透明”压缩,因此如果content-encoding 或accept-encoding header 存在,则当feign.okhttp.OkHttpClient 存在于 Classpath 中,并且feign.okhttp.enabled 设置为true . |
1.11. 假装日志记录
为创建的每个 Feign 客户端创建一个 Logger。默认情况下,Logger 的名称是用于创建 Feign 客户端的接口的完整类名。假日志记录仅响应DEBUG
水平。
logging.level.project.user.UserClient: DEBUG
这Logger.Level
对象,告诉 Feign 要记录多少。选项包括:
-
NONE
、无日志记录 (DEFAULT)。 -
BASIC
,仅记录请求方法和 URL 以及响应状态代码和执行时间。 -
HEADERS
,记录基本信息以及请求和响应标头。 -
FULL
,记录请求和响应的标头、正文和元数据。
例如,以下代码会将Logger.Level
自FULL
:
@Configuration
public class FooConfiguration {
@Bean
Logger.Level feignLoggerLevel() {
return Logger.Level.FULL;
}
}
1.12. Feign Capability 支持
Feign 功能公开核心 Feign 组件,以便可以修改这些组件。例如,功能可以采用Client
,装饰它,并将装饰的实例返回给 Feign。
对 metrics 库的支持就是一个很好的真实示例。请参阅 Feign metrics。
创建一个或多个Capability
bean 并将它们放在@FeignClient
configuration 允许您注册它们并修改相关客户端的行为。
@Configuration
public class FooConfiguration {
@Bean
Capability customCapability() {
return new CustomCapability();
}
}
1.13. 假量度
如果满足以下所有条件,则MicrometerCapability
创建并注册 bean,以便您的 Feign 客户端将指标发布到 Micrometer:
-
feign-micrometer
位于 Classpath 上 -
一个
MeterRegistry
Bean 可用 -
feign metrics 属性设置为
true
(默认情况下)-
feign.metrics.enabled=true
(适用于所有客户) -
feign.client.config.feignName.metrics.enabled=true
(对于单个客户)
-
如果您的应用程序已经使用 Micrometer,则启用指标就像将feign-micrometer 添加到您的 Classpath 中。 |
您还可以通过以下任一方式禁用该功能:
-
以外
feign-micrometer
从 Classpath 中 -
将其中一个 Feign metrics 属性设置为
false
-
feign.metrics.enabled=false
-
feign.client.config.feignName.metrics.enabled=false
-
feign.metrics.enabled=false 禁用对所有 Feign 客户端的度量支持,而不管客户端级别标志的值如何:feign.client.config.feignName.metrics.enabled .
如果要为每个客户端启用或禁用 merics,请不要设置feign.metrics.enabled 并使用feign.client.config.feignName.metrics.enabled . |
您还可以自定义MicrometerCapability
通过注册您自己的 Bean:
@Configuration
public class FooConfiguration {
@Bean
public MicrometerCapability micrometerCapability(MeterRegistry meterRegistry) {
return new MicrometerCapability(meterRegistry);
}
}
1.14. 假装缓存
如果@EnableCaching
annotation 时,使用CachingCapability
创建并注册 bean,以便您的 Feign 客户端识别@Cache*
其接口上的注释:
public interface DemoClient {
@GetMapping("/demo/{filterParam}")
@Cacheable(cacheNames = "demo-cache", key = "#keyParam")
String demoEndpoint(String keyParam, @PathVariable String filterParam);
}
您还可以通过 property 禁用该功能feign.cache.enabled=false
.
1.15. 假装@QueryMap支持
OpenFeign 餐厅@QueryMap
注解支持将 POJO 用作
GET 参数映射。不幸的是,默认的 OpenFeign QueryMap 注释是
与 Spring 不兼容,因为它缺少value
财产。
Spring Cloud OpenFeign 提供了一个等效的@SpringQueryMap
注解,其中
用于将 POJO 或 Map 参数注释为查询参数 map。
例如,Params
class 定义参数param1
和param2
:
// Params.java
public class Params {
private String param1;
private String param2;
// [Getters and setters omitted for brevity]
}
以下假客户端使用Params
类@SpringQueryMap
注解:
@FeignClient("demo")
public interface DemoTemplate {
@GetMapping(path = "/demo")
String demoEndpoint(@SpringQueryMap Params params);
}
如果您需要对生成的查询参数 map 进行更多控制,可以实现自定义的QueryMapEncoder
豆。
1.16. HATEOAS 支持
Spring 提供了一些 API 来创建遵循 HATEOAS 原则、Spring Hateoas 和 Spring Data REST 的 REST 表示。
如果您的项目使用org.springframework.boot:spring-boot-starter-hateoas
起动机
或org.springframework.boot:spring-boot-starter-data-rest
starter 中,默认情况下会启用 Feign HATEOAS 支持。
启用 HATEOAS 支持后,允许 Feign 客户端进行序列化 并反序列化 HATEOAS 表示模型:EntityModel、CollectionModel 和 PagedModel。
@FeignClient("demo")
public interface DemoTemplate {
@GetMapping(path = "/stores")
CollectionModel<Store> getStores();
}
1.17. Spring @MatrixVariable 支持
Spring Cloud OpenFeign 为 Spring 提供支持@MatrixVariable
注解。
如果将 map 作为方法参数传递,则@MatrixVariable
路径段是通过将映射中的键值对与 .=
如果传递了不同的对象,则name
在@MatrixVariable
annotation (如果已定义) 或带注释的变量名称为
使用 .=
- 重要
-
尽管在服务器端, Spring 不要求用户将路径段占位符命名为与矩阵变量名称相同的名称,但由于在客户端太模糊,Spring Cloud OpenFeign 要求您添加一个名称与
name
在@MatrixVariable
annotation (如果已定义) 或带注释的变量名称。
例如:
@GetMapping("/objects/links/{matrixVars}")
Map<String, List<String>> getObjects(@MatrixVariable Map<String, List<String>> matrixVars);
请注意,变量 name 和 path segment placeholder 都称为matrixVars
.
@FeignClient("demo")
public interface DemoTemplate {
@GetMapping(path = "/stores")
CollectionModel<Store> getStores();
}
1.18. 假装CollectionFormat
支持
我们支持feign.CollectionFormat
通过提供@CollectionFormat
注解。
你可以通过传递所需的feign.CollectionFormat
作为注释值。
在以下示例中,CSV
format 而不是默认的EXPLODED
以处理该方法。
@FeignClient(name = "demo")
protected interface DemoFeignClient {
@CollectionFormat(feign.CollectionFormat.CSV)
@GetMapping(path = "/test")
ResponseEntity performRequest(String test);
}
1.19. 反应式支持
由于 OpenFeign 项目目前不支持反应式客户端,例如 Spring WebClient,因此 Spring Cloud OpenFeign 也不支持。
在此之前,我们建议使用 feign-reactive 来支持 Spring WebClient。
1.19.1. 早期初始化错误
根据您使用 Feign 客户端的方式,您可能会在启动应用程序时看到初始化错误。
要临时解决这个问题,您可以使用ObjectProvider
当 Autowiring 您的 Client 端时。
@Autowired
ObjectProvider<TestFeignClient> testFeignClient;
1.20. Spring Data 支持
您可以考虑启用 Jackson Modules 以获得支持org.springframework.data.domain.Page
和org.springframework.data.domain.Sort
译码。
feign.autoconfiguration.jackson.enabled=true
1.21. 弹簧@RefreshScope
支持
如果启用了 Feign client refresh,则每个 Feign 客户端都是使用feign.Request.Options
作为刷新范围的 Bean。这意味着connectTimeout
和readTimeout
可以通过以下方式针对任何 Feign 客户端实例进行刷新POST /actuator/refresh
.
默认情况下,Feign 客户端中的刷新行为处于禁用状态。使用以下属性启用刷新行为:
feign.client.refresh-enabled=true
请勿注释@FeignClient 与@RefreshScope 注解。 |
1.22. OAuth2 支持
可以通过设置以下标志来启用 OAuth2 支持:
feign.oauth2.enabled=true
1.22.1. 已弃用的 OAuth2 支持
当标志设置为true
和spring-security-oauth2-autoconfigure
存在于 Classpath 中,并且存在 OAuth2 Client Context 资源详细信息,并且OAuth2ClientContext
和OAuth2ProtectedResourceDetails
beans 存在,一个类OAuth2FeignRequestInterceptor
已创建。在每个请求之前,侦听器会解析所需的访问令牌并将其作为标头包含在内。
有时,当为 Feign 客户端启用负载均衡时,您可能还希望使用负载均衡来获取访问令牌。为此,您应该确保负载均衡器位于 Classpath(spring-cloud-starter-loadbalancer)上,并通过设置以下标志显式启用 OAuth2FeignRequestInterceptor 的负载平衡:
feign.oauth2.load-balanced=true
- 警告
-
如上所述的 OAuth2 支持现已弃用,因为
spring-security-oauth2-autoconfigure
已达到使用寿命。请改用下面描述的模式。
1.22.2. 当前的 OAuth2 支持
当feign.client.refresh-enabled
flag 设置为 true,并且spring-security-oauth2-client
存在于 Classpath 中,则 classOAuth2AccessTokenInterceptor
已创建。在每个请求之前,侦听器会解析所需的访问令牌并将其作为标头包含在内。OAuth2AccessTokenInterceptor
使用OAuth2AuthorizedClientManager
获取OAuth2AuthorizedClient
,它持有OAuth2AccessToken
.如果用户指定了 OAuth2clientRegistrationId
使用feign.oauth2.clientRegistrationId
属性,它将用于检索令牌。如果未检索到令牌,或者clientRegistrationId
未指定,则serviceId
从 中检索url
将使用 host segment。
- 提示
-
使用
serviceId
作为 OAuth2 客户端 registrationId 对于负载均衡的 Feign 客户端来说很方便。对于非负载均衡的 API,基于 property 的clientRegistrationId
是一种合适的方法。 - 提示
-
如果您不想使用
OAuth2AuthorizedClientManager
,则只需在配置中实例化此类型的 bean。