spring整合feign和hystrix實現RPC熔斷降級和監控
背景
在spring應用中,原始的openfeign生成的代理物件是沒有直接在spring容器中注入的,要使用openfeign的原生使用方式如下所示。
這種方式,對一個有追求的程式設計師來說很不優雅,本文來解決將feign整合到spring中的問題,實現feign的優雅使用。
解決上一個問題,之後我們還問自己一個問題,feign的作用是http的rpc呼叫,但是http的呼叫如何優雅的實現服務熔斷和降級呢?很明顯用hystrix,這裡又有問題了,hystrix和feign分別生成的動態代理物件是不相容的,feign是需要手動建立代理物件,而hystrix可以通過spring建立或者通過command api來建立,如何將 feign和hystrix一起整合到spring中呢?
帶著這兩個問題我們繼續。
feign
首先feign的介紹可以參考上一篇博文:添feign的基本使用和原始碼解析,講解了feign的使用和工作流程和原始碼解析。
回顧一下feign使用,Feign.builder() .target()會建立GitHub類的動態代理物件GitHubProxy即github例項
public class MyApp {
public static void main(String[] args) {
GitHub github = Feign.builder()
.decoder(new GsonDecoder())
.target(GitHub.class, "https://api.github.com");
/**
*@Param註解的引數會按照名字填充{owner}{repo}
HousonCao/Hystrix
*/
github.contributors("HousonCao", "Hystrix");
}
}
hystrix
hystric使用和原理介紹參考:
https://mp.csdn.net/console/editor/html/111477766
spring整合feign和hystrix
本文重點闡述如何將第三方框架整合到spring,以spring整合hystrix-feign擴充套件為例展示spring整合三方框架的流程。
首先說明一下需求:
feign在springboot中是可以和feign和hystrix一起使用的,但是在spring中需要自己整合到spring,我們基於
和他的擴充套件
實現spring整合feign和hystrix,達到宣告式的http RPC呼叫,同時支援RxJava響應式程式設計的目標。我們思考一下:mybatis是如何實現和spring整合的呢,我們要做的和mybatis很類似,甚至可以照葫蘆畫瓢。
public void setHttpClient(OkHttpClientService httpClient) {
this.httpClient = httpClient;
}
public Class getFallback() {
return fallback;
}
public void setFallback(Class fallback) {
this.fallback = fallback;
}
public FeignRegistry getFeignRegistry() {
return feignRegistry;
}
public void setFeignRegistry(FeignRegistry feignRegistry) {
this.feignRegistry = feignRegistry;
}
/**
* {@inheritDoc}
*/
@Override
public T getObject() throws Exception {
if (feignRegistry.getMapper(hystrixFeignInterface) == null) {
notNull(this.hystrixFeignInterface, "Property 'mapperInterface' is required");
// create feign hystrix client instance
LOGGER.info("建立feign客戶端引數:Uri:{}",uri);
Object feignClient = TraceHystrixFeign.builder()
.setterFactory(setterFactory)
.encoder(encoder)
.decoder(decoder)
.logger(logger)
.logLevel(Level.FULL)
.retryer(new NoRetryer())
.client(new OkHttpClient(httpClient.getmOkHttpClient()))
.options(new Options(httpClient.getConnectTimeout(), httpClient.getReadTimeout()))
.target(hystrixFeignInterface, uri, new FallbackFactory() {
@Override
public Object create(Throwable cause) {
try {
return BeanUtils.instantiateClass(fallback.getConstructor(Throwable.class), cause);
} catch (NoSuchMethodException e) {
LOGGER.warn("建立fallback失敗",e);
throw new RuntimeException(e);
}
}
});
// add feign hystrix interface target instance to set
feignRegistry.addMapper(hystrixFeignInterface, feignClient);
}
// get from set
return (T) feignRegistry.getMapper(hystrixFeignInterface);
}
/**
* {@inheritDoc}
*/
@Override
public Class<T> getObjectType() {
return this.hystrixFeignInterface;
}
/**
* {@inheritDoc}
*/
@Override
public boolean isSingleton() {
return true;
}
public Class<T> getHystrixFeignInterface() {
return hystrixFeignInterface;
}
public void setHystrixFeignInterface(Class<T> hystrixFeignInterface) {
this.hystrixFeignInterface = hystrixFeignInterface;
}
}
bean定義完了需要定義註解,讓客戶端能配置factorybean的屬性。這裡直接將@hystrixCommand的部分屬性搬過來。
```java
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
public @interface HystrixFeign {
String feignRegistry() default "feignRegistry";
Class fallbackRef() ;
/**
* feign的請求body編碼器
* @return
*/
String encoderRef() default "";
/**
* feign 響應解碼器
* @return
*/
String decoderRef() default "";
String httpClient() default "";
String uri() default "";
/**
* The command group key is used for grouping together commands such as for reporting,
* alerting, dashboards or team/library ownership.
* <p/>
* default => the runtime class name of annotated method
*
* @return group key
*/
String groupKey() default "";
/**
* Hystrix command key.
* <p/>
* default => the name of annotated method. for example:
* <code>
* ...
* @HystrixCommand
* public User getUserById(...)
* ...
* the command name will be: 'getUserById'
* </code>
*
* @return command key
*/
String commandKey() default "";
/**
* The thread-pool key is used to represent a
* HystrixThreadPool for monitoring, metrics publishing, caching and other such uses.
*
* @return thread pool key
*/
String threadPoolKey() default "";
/**
* Specifies a method to process fallback logic.
* A fallback method should be defined in the same class where is HystrixCommand.
* Also a fallback method should have same signature to a method which was invoked as hystrix command.
* for example:
* <code>
* @HystrixCommand(fallbackMethod = "getByIdFallback")
* public String getById(String id) {...}
*
* private String getByIdFallback(String id) {...}
* </code>
* Also a fallback method can be annotated with {@link HystrixCommand}
* <p/>
* default => see {@link com.netflix.hystrix.contrib.javanica.command.GenericCommand#getFallback()}
*
* @return method name
*/
String fallbackMethod() default "";
/**
* Specifies command properties.
*
* @return command properties
*/
HystrixProperty[] commandProperties() default {};
/**
* Specifies thread pool properties.
*
* @return thread pool properties
*/
HystrixProperty[] threadPoolProperties() default {};
/**
* Defines exceptions which should be ignored.
* Optionally these can be wrapped in HystrixRuntimeException if raiseHystrixExceptions contains RUNTIME_EXCEPTION.
*
* @return exceptions to ignore
*/
Class<? extends Throwable>[] ignoreExceptions() default {};
/**
* Specifies the mode that should be used to execute hystrix observable command.
* For more information see {@link ObservableExecutionMode}.
*
* @return observable execution mode
*/
ObservableExecutionMode observableExecutionMode() default ObservableExecutionMode.EAGER;
/**
* When includes RUNTIME_EXCEPTION, any exceptions that are not ignored are wrapped in HystrixRuntimeException.
*
* @return exceptions to wrap
*/
HystrixException[] raiseHystrixExceptions() default {};
/**
* Specifies default fallback method for the command. If both {@link #fallbackMethod} and {@link #defaultFallback}
* methods are specified then specific one is used.
* note: default fallback method cannot have parameters, return type should be compatible with command return type.
*
* @return the name of default fallback method
*/
String defaultFallback() default "";
}
定義完屬性,我們繼續
bean掃描
-
繼承Spring提供的ClassPathBeanDefinitionScanner 進行beandefinition的掃描,實現doScan方法,生成beandefinition,返回beandefinition集合進行註冊。類似spring的@component註解需要有個掃描的過程。
public class ClassPathHystrixFeignScanner extends ClassPathBeanDefinitionScanner { public static final Logger LOGGER = LoggerFactory.getLogger(ClassPathHystrixFeignScanner.class); private Class<? extends Annotation> annotationClass; private HystrixFeignFactoryBean mapperFactoryBean = new HystrixFeignFactoryBean(); public ClassPathHystrixFeignScanner(BeanDefinitionRegistry registry) { super(registry, false); } public void setAnnotationClass(Class<? extends Annotation> annotationClass) { this.annotationClass = annotationClass; } public void setMapperFactoryBean(HystrixFeignFactoryBean mapperFactoryBean) { this.mapperFactoryBean = (mapperFactoryBean != null ? mapperFactoryBean : new HystrixFeignFactoryBean()); } /** * Configures parent scanner to search for the right interfaces. It can search for all interfaces or just for those * that extends a markerInterface or/and those annotated with the annotationClass */ public void registerFilters() { boolean acceptAllInterfaces = true; // if specified, use the given annotation and / or marker interface if (this.annotationClass != null) { addIncludeFilter(new AnnotationTypeFilter(this.annotationClass)); acceptAllInterfaces = false; } // override AssignableTypeFilter to ignore matches on the actual marker interface if (acceptAllInterfaces) { // default include filter that accepts all classes addIncludeFilter(new TypeFilter() { @Override public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException { return true; } }); } // exclude package-info.java addExcludeFilter(new TypeFilter() { @Override public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException { String className = metadataReader.getClassMetadata().getClassName(); return className.endsWith("package-info"); } }); } /** * Calls the parent search that will search and register all the candidates. Then the registered objects are post * processed to set them as MapperFactoryBeans */ @Override public Set<BeanDefinitionHolder> doScan(String... basePackages) { Set<BeanDefinitionHolder> beanDefinitions = super.doScan(basePackages); if (beanDefinitions.isEmpty()) { logger.warn("No hystrix feign was found in '" + Arrays.toString(basePackages) + "' package. Please check your configuration."); } else { processBeanDefinitions(beanDefinitions); } return beanDefinitions; } private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) { GenericBeanDefinition definition; for (BeanDefinitionHolder holder : beanDefinitions) { definition = (GenericBeanDefinition) holder.getBeanDefinition(); definition.setAutowireMode(AbstractBeanDefinition.AUTOWIRE_BY_NAME); if (logger.isDebugEnabled()) { logger.debug("Creating HystrixFeignFactoryBean with name '" + holder.getBeanName() + "' and '" + definition.getBeanClassName() + "' HystrixFeign Interface"); } // the hystrixFeign interface is the original class of the bean // but, the actual class of the bean is HystrixFeignFactoryBean definition.getPropertyValues().add("hystrixFeignInterface", definition.getBeanClassName()); //TODO 解析interface的方法註解獲取hystrix配置 String realBeanClassName = definition.getBeanClassName(); Class<?> realBeadClass = null; try { realBeadClass = Class.forName(realBeanClassName); } catch (ClassNotFoundException e) { LOGGER.warn("掃描hystrix feign失敗",e); } HystrixFeign annotation = realBeadClass.getAnnotation(HystrixFeign.class); if (annotation == null) { LOGGER.warn("HystrixFeign 註解不存在:{}", realBeadClass); continue; } GenericSetterBuilder.Builder setterBuilder = GenericSetterBuilder.builder() .groupKey(annotation.groupKey()) .threadPoolKey(annotation.threadPoolKey()) .commandKey(annotation.commandKey()) .commandProperties(Arrays.asList(annotation.commandProperties())) .threadPoolProperties(Arrays.asList(annotation.threadPoolProperties())); GenericSetterBuilder build = setterBuilder.build(); Setter setter = build.build(); definition.getPropertyValues().add("setterFactory", new HystrixFeignSetterFactory(setter)); String uri = annotation.uri(); String uriProperty = getEnvironment().resolveRequiredPlaceholders(uri); int indexOfSep= uri.indexOf(":"); if (indexOfSep!=-1){ String urlKey = uri.substring(2, indexOfSep); String remoteUri= ConfigManager.getString(urlKey); if(StringUtils.isNotEmpty(remoteUri)){ definition.getPropertyValues().add("uri", remoteUri); }else{ definition.getPropertyValues().add("uri", uriProperty); } } definition.getPropertyValues().add("httpClient", new RuntimeBeanReference(annotation.httpClient())); if (StringUtils.isNotEmpty(annotation.decoderRef())) { definition.getPropertyValues().add("decoder", new RuntimeBeanReference(annotation.decoderRef())); } definition.getPropertyValues().add("fallback", annotation.fallbackRef()); definition.getPropertyValues().add("feignRegistry", new RuntimeBeanReference(annotation.feignRegistry())); if (StringUtils.isNotEmpty(annotation.encoderRef())) { definition.getPropertyValues().add("encoder", new RuntimeBeanReference(annotation.encoderRef())); } if (StringUtils.isNotEmpty(annotation.decoderRef())) { definition.getPropertyValues().add("decoder", new RuntimeBeanReference(annotation.decoderRef())); } definition.setBeanClass(this.mapperFactoryBean.getClass()); // definition.setAutowireMode(AbstractBeanDefinition.AUTOWIRE_BY_TYPE); } } /** * {@inheritDoc} */ @Override protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) { return beanDefinition.getMetadata().isInterface() && beanDefinition.getMetadata().isIndependent(); } /** * {@inheritDoc} */ @Override protected boolean checkCandidate(String beanName, BeanDefinition beanDefinition) { if (super.checkCandidate(beanName, beanDefinition)) { return true; } else { logger.warn("Skipping HystrixFeignFactoryBean with name '" + beanName + "' and '" + beanDefinition.getBeanClassName() + "' mapperInterface" + ". Bean already defined with the same name!"); return false; } } }
beandefinition註冊
-
首先需要將要生成的例項的beandefinition註冊到spring的beandefinition容器BeanDefinitionRegistry,如何實現呢。
-
實現方式有兩種:
-
- 繼承ImportBeanDefinitionRegistrar在配置類上使用@Import(HystrixFeignScannerRegistrar.class)即可在啟動時載入Registrar類例項,在registerBeanDefinitions方法中將scanner掃描的beandefinitions註冊到registry
- 實現BeanDefinitionRegistryPostProcessor,spring啟動時呼叫refresh方法會呼叫postProcessBeanDefinitionRegistry執行註冊beandefinition的程式碼,在這裡執行ClassPathBeanDefinitionScanner的掃描邏輯,將生成的beandefinition進行註冊。
-
我們選擇第一種實現:原始碼如下:
public class HystrixFeignScannerRegistrar implements ImportBeanDefinitionRegistrar, ResourceLoaderAware {
private ResourceLoader resourceLoader;
/**
* {@inheritDoc}
*/
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
AnnotationAttributes annoAttrs = AnnotationAttributes.fromMap(importingClassMetadata.getAnnotationAttributes(
HystrixFeignScan.class.getName()));
ClassPathHystrixFeignScanner scanner = new ClassPathHystrixFeignScanner(registry);
// this check is needed in Spring 3.1
if (resourceLoader != null) {
scanner.setResourceLoader(resourceLoader);
}
Class<? extends Annotation> annotationClass = annoAttrs.getClass("annotationClass");
if (!Annotation.class.equals(annotationClass)) {
scanner.setAnnotationClass(annotationClass);
}
Class<? extends HystrixFeignFactoryBean> mapperFactoryBeanClass = annoAttrs.getClass("factoryBean");
if (!HystrixFeignFactoryBean.class.equals(mapperFactoryBeanClass)) {
scanner.setMapperFactoryBean(BeanUtils.instantiateClass(mapperFactoryBeanClass));
}
List<String> basePackages = new ArrayList<String>();
for (String pkg : annoAttrs.getStringArray("value")) {
if (StringUtils.hasText(pkg)) {
basePackages.add(pkg);
}
}
for (String pkg : annoAttrs.getStringArray("basePackages")) {
if (StringUtils.hasText(pkg)) {
basePackages.add(pkg);
}
}
scanner.registerFilters();
scanner.doScan(StringUtils.toStringArray(basePackages));
}
/**
* {@inheritDoc}
*/
@Override
public void setResourceLoader(ResourceLoader resourceLoader) {
this.resourceLoader = resourceLoader;
}
}
配置啟動掃描
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(HystrixFeignScannerRegistrar.class)
public @interface EnableHystrixFeign {
String[] value() default {};
String[] basePackages() default {};
Class<? extends Annotation> annotationClass() default Annotation.class;
Class<? extends HystrixFeignFactoryBean> factoryBean() default HystrixFeignFactoryBean.class;
}
使用舉例
@HystrixFeign(commandKey = "expressApi",
threadPoolKey = "expressPool",
uri = "${test.url:http://localhost:8080/test_hystrix_feign}",
fallbackRef = ExpressApi.ExpressFallback.class,
httpClient = "okHttpClient",
commandProperties = {
@HystrixProperty(name = "execution.isolation.thread.timeoutInMilliseconds", value = "1000"),
},
threadPoolProperties = {
@HystrixProperty(name = "coreSize", value = "20"),
@HystrixProperty(name = "maximumSize", value = "20")
})
public interface ExpressApi {
@RequestLine("POST ")
@Headers({"Content-Type: application/json", "host: localhost:8080"})
String getExpressInfos(ExpressQueryParam queryParam);
@Slf4j
@AllArgsConstructor
class ExpressFallback implements ExpressApi {
Throwable t;
@Override
public String getExpressInfos(ExpressQueryParam queryParam) {
return null;
}
}
}
相關文章
- 微服務SpringCloud之熔斷監控Hystrix Dashboard和Turbine微服務SpringGCCloud
- SpringCloud-Hystrix 服務降級、熔斷SpringGCCloud
- springcloud(五):熔斷監控Hystrix DashboardSpringGCCloud
- 微服務11:熔斷、降級的Hystrix實現(附原始碼)微服務原始碼
- Spring Cloud入門教程-Hystrix斷路器實現容錯和降級SpringCloud
- 跟我學SpringCloud | 第五篇:熔斷監控Hystrix Dashboard和TurbineSpringGCCloud
- SpringMvc整合開源流量監控、限流、熔斷降級、負載保護元件SentinelSpringMVC負載元件
- 分散式RPC框架Dubbo實現服務治理:整合Kryo實現高速序列化,整合Hystrix實現熔斷器分散式RPC框架
- SpringCloud Netflix (五) : Hystrix 服務熔斷和服務降級SpringGCCloud
- Spring cloud(4)-熔斷(Hystrix)SpringCloud
- Spring Cloud中Hystrix、Ribbon及Feign的熔斷關係是什麼?SpringCloud
- 服務的熔斷和降級的區別
- Spring Cloud實戰系列(四) - 熔斷器HystrixSpringCloud
- Sentinel限流熔斷降級
- 擴充套件Spring Cloud Feign 實現自動降級套件SpringCloud
- Hystrix--熔斷
- Spring Cloud Feign 熔斷機制填坑SpringCloud
- 熔斷和降級的初步詳解實現(NET Core控制檯輸出講解Polly)
- Sentinel入門到實操 (限流熔斷降級)
- springcloud(六):熔斷監控TurbineSpringGCCloud
- 12.SpringCloudAlibabaSentinel實現熔斷和限流SpringGCCloud
- Sentinel全域性Feign預設熔斷設計實現
- Spring Cloud 快速入門(四)Hystrix Dashboard 監控儀表盤、Turbine 聚合監控、服務降級報警機制SpringCloud
- 五. SpringCloud服務降級與熔斷SpringGCCloud
- 分散式熔斷降級平臺aegis分散式
- sentinel流量控制和熔斷降級執行流程之原始碼分析原始碼
- springcloud之Hystrix熔斷器SpringGCCloud
- springcloud(四):熔斷器HystrixSpringGCCloud
- Sentinel 成為 Spring Cloud 官方推薦的主流熔斷降級方案SpringCloud
- 一個故事理解限流熔斷降級
- 面試官:說說降級、熔斷、限流面試
- spring cloud優雅的處理feign熔斷異常SpringCloud
- 《重磅 | Sentinel 成為 Spring Cloud 官方推薦的主流熔斷降級方案》SpringCloud
- 面試官:熔斷降級原理是什麼?面試
- Spring Cloud Alibaba:Sentinel實現熔斷與限流SpringCloud
- Spring Boot整合Hystrix實現服務容錯Spring Boot
- SpringCloud微服務實戰——搭建企業級開發框架(十五):整合Sentinel高可用流量管理框架【熔斷降級】SpringGCCloud微服務框架
- 微服務熔斷限流Hystrix之Dashboard微服務