# spore-spring-boot-starter
**Repository Path**: EcoNetsTech/spore-spring-boot-starter
## Basic Information
- **Project Name**: spore-spring-boot-starter
- **Description**: A spring-boot starter for retrofit, supports rapid integration and feature enhancements.(适用于retrofit的spring-boot-starter,支持快速集成和功能增强)
- **Primary Language**: Unknown
- **License**: Apache-2.0
- **Default Branch**: master
- **Homepage**: None
- **GVP Project**: No
## Statistics
- **Stars**: 2
- **Forks**: 0
- **Created**: 2024-02-21
- **Last Updated**: 2024-04-01
## Categories & Tags
**Categories**: Uncategorized
**Tags**: None
## README
## spore-spring-boot-starter
[](https://www.apache.org/licenses/LICENSE-2.0.html)
[](https://github.com/EcoNetsTech/spore-spring-boot-starterr/releases)
[](https://docs.oracle.com/javase/8/docs/index.html)
[](https://docs.spring.io/spring-boot/docs/2.1.5.RELEASE/reference/htmlsingle/)
**适用于retrofit的spring-boot-starter,支持快速集成和功能增强**。
1. *Spring Boot 3.x 项目,请使用spore-spring-boot-starter 3.x*。
2. *Spring Boot 1.x/2.x 项目,请使用spore-spring-boot-starter 2.x*。
> 麻烦大家能给一颗star✨,您的star是我们持续更新的动力!
github项目地址:[https://github.com/EcoNetsTech/spore-spring-boot-starter](https://github.com/EcoNetsTech/spore-spring-boot-starter)
## 功能特性
- [x] [自定义OkHttpClient](#自定义OkHttpClient)
- [x] [注解式拦截器](#注解式拦截器)
- [x] [日志打印](#日志打印)
- [x] [请求重试](#请求重试)
- [x] [微服务之间的HTTP调用](#微服务之间的HTTP调用)
- [x] [全局拦截器](#全局拦截器)
- [x] [调用适配器](#调用适配器)
- [x] [数据转换器](#数据转码器)
- [x] [元注解](#元注解)
- [x] [其他功能示例](#其他功能示例)
## 快速开始
### 引入依赖
```xml
cn.econets.ximutech
spore-spring-boot-starter
2.0.1
```
**如果启动失败,大概率是依赖冲突,烦请引入或者排除相关依赖**。
### 定义HTTP接口
**接口必须使用`@SporeClient`注解标记**!HTTP相关注解可参考官方文档:[retrofit官方文档](https://square.github.io/retrofit/)。
```java
@SporeClient(baseUrl = "${test.baseUrl}")
public interface HttpApi {
@GET("/")
HitokotoVO getContent();
}
```
### 注入使用
**将接口注入到其它Service中即可使用!**
```java
@Service
public class TestService {
@Autowired
private HttpApi httpApi;
public void test() {
// 使用`httpApi`发起HTTP请求
}
}
```
**默认情况下,自动使用`SpringBoot`扫描路径进行`SporeClient`注册**。你也可以在配置类加上`@EnableSporeClients`手工指定扫描路径。
## HTTP请求相关注解
`HTTP`请求相关注解,全部使用了`Retrofit`原生注解,以下是一个简单说明:
| 注解分类|支持的注解 |
|------------|-----------|
|请求方式|`@GET` `@HEAD` `@POST` `@PUT` `@DELETE` `@OPTIONS` `@HTTP`|
|请求头|`@Header` `@HeaderMap` `@Headers`|
|Query参数|`@Query` `@QueryMap` `@QueryName`|
|path参数|`@Path`|
|form-encoded参数|`@Field` `@FieldMap` `@FormUrlEncoded`|
| 请求体 |`@Body`|
|文件上传|`@Multipart` `@Part` `@PartMap`|
|url参数|`@Url`|
> 详细信息可参考官方文档:[retrofit官方文档](https://square.github.io/retrofit/)
## 配置属性
组件支持了多个可配置的属性,用来应对不同的业务场景,具体可支持的配置属性及默认值如下:
**注意:应用只需要配置要更改的配置项**!
```yaml
retrofit:
# 全局转换器工厂
global-converter-factories:
- retrofit2.converter.jackson.JacksonConverterFactory
# 全局调用适配器工厂
global-call-adapter-factories:
- cn.econets.ximutech.spore.retrofit.adapter.BodyCallAdapterFactory
# 全局日志打印配置
global-log:
# 启用日志打印
enable: true
# 全局日志打印级别
log-level: info
# 全局日志打印策略
log-strategy: basic
# 全局重试配置
global-retry:
# 是否启用全局重试
enable: false
# 全局重试间隔时间
interval-ms: 100
# 全局最大重试次数
max-retries: 2
# 全局重试规则
retry-rules:
- response_status_not_2xx
- occur_io_exception
# 全局超时时间配置
global-timeout:
# 全局读取超时时间
read-timeout-ms: 10000
# 全局写入超时时间
write-timeout-ms: 10000
# 全局连接超时时间
connect-timeout-ms: 10000
# 全局完整调用超时时间
call-timeout-ms: 0
```
## 高级功能
### 超时时间配置
如果仅仅需要修改`OkHttpClient`的超时时间,可以通过`@SporeClient`相关字段修改,或者全局超时配置修改。
### 自定义OkHttpClient
如果需要修改`OkHttpClient`其它配置,可以通过自定义`OkHttpClient`来实现,步骤如下:
1. 实现`OkHttpClientRegistrar`接口,调用`OkHttpClientRegistry#register()`方法注册`OkHttpClient`。
```java
@Component
@Slf4j
public class CustomOkHttpClientRegistrar implements OkHttpClientRegistrar {
@Override
public void register(OkHttpClientRegistry registry) {
OkHttpClient okHttpClient = new OkHttpClient.Builder()
.addInterceptor(chain -> {
log.info("========== 自定义okHttpClient的拦截器 ============");
return chain.proceed(chain.request());
})
.build();
registry.register("customOkHttpClient", okHttpClient);
}
}
```
2. 通过`@SporeClient.sourceOkHttpClient`指定当前接口要使用的`OkHttpClient`。
```java
@SporeClient(baseUrl = "${test.baseUrl}", sourceOkHttpClient = "customOkHttpClient")
public interface CustomOkHttpTestApi {
@GET("/get")
Result get();
}
```
> 注意:组件不会直接使用指定的`OkHttpClient`,而是基于该`OkHttpClient`创建一个新的。
### 注解式拦截器
组件提供了**注解式拦截器**,支持基于url路径匹配拦截,使用的步骤如下:
1. 继承`BasePathMatchInterceptor`
2. 使用`@Intercept`注解指定要使用的拦截器
> 如果需要使用多个拦截器,在接口上标注多个`@Intercept`注解即可。
下面以"给指定请求的url后面拼接timestamp时间戳"为例,介绍下如何使用注解式拦截器。
#### 继承`BasePathMatchInterceptor`编写拦截处理器
```java
@Component
@Scope("prototype")
public class TokenInterceptor extends BasePathMatchInterceptor {
@Override
protected Response doIntercept(Chain chain) throws IOException {
System.out.println("============ 进入token拦截器 ===============");
Request request = chain.request();
HttpUrl newUrl = request.url().newBuilder()
.addQueryParameter("token", UUID.randomUUID().toString())
.build();
Request newRequest = request.newBuilder()
.url(newUrl)
.build();
return chain.proceed(newRequest);
}
}
```
#### 接口上使用`@Intercept`进行标注
```java
@SporeClient(baseUrl = "${test.baseUrl}")
@Intercept(handler = TokenInterceptor.class, include = {"/api/**"}, exclude = "/api/test1")
public interface HttpApi {
@GET("/get")
Result get();
}
```
上面的`@Intercept`配置表示:拦截`HttpApi`接口下`/api/**`路径下(排除`/api/test1`)的请求,拦截处理器使用`TokenInterceptor`。
## 全局拦截器
### 全局应用拦截器
如果我们需要对整个系统的的`HTTP`请求执行统一的拦截处理,可以实现全局拦截器`GlobalInterceptor`, 并配置成`spring Bean`。
```java
@Component
public class CustomGlobalInterceptor implements GlobalInterceptor {
@Override
public Response intercept(Chain chain) throws IOException {
System.out.println("===========执行全局拦截器===========");
Request request = chain.request();
Request newRequest = request.newBuilder()
.addHeader("traceId", UUID.randomUUID().toString())
.build();
return chain.proceed(newRequest);
}
}
```
### 日志打印
组件支持支持全局日志打印和声明式日志打印。
#### 全局日志打印
默认情况下,全局日志打印是开启的,默认配置如下:
```yaml
retrofit:
# 全局日志打印配置
global-log:
# 启用日志打印
enable: true
# 全局日志打印级别
log-level: info
# 全局日志打印策略
log-strategy: basic
```
四种日志打印策略含义如下:
1. `NONE`:No logs.
2. `BASIC`:Logs request and response lines.
3. `HEADERS`:Logs request and response lines and their respective headers.
4. `BODY`:Logs request and response lines and their respective headers and bodies (if present).
#### 声明式日志打印
如果只需要部分请求才打印日志,可以在相关接口或者方法上使用`@SporeLogging`注解。
#### 日志打印自定义扩展
如果需要修改日志打印行为,可以继承`LoggingInterceptor`,并将其配置成`Spring bean`。
#### 聚合日志打印
如果需要将同一个请求的日志聚合在一起打印,可配置`AggregateLoggingInterceptor`。
```java
@Bean
public LoggingInterceptor loggingInterceptor(RetrofitProperties retrofitProperties){
return new AggregateLoggingInterceptor(retrofitProperties.getGlobalLog());
}
```
### 请求重试
组件支持支持全局重试和声明式重试。
#### 全局重试
全局重试默认关闭,默认配置项如下:
```yaml
retrofit:
# 全局重试配置
global-retry:
# 是否启用全局重试
enable: false
# 全局重试间隔时间
interval-ms: 100
# 全局最大重试次数
max-retries: 2
# 全局重试规则
retry-rules:
- response_status_not_2xx
- occur_io_exception
```
重试规则支持三种配置:
1. `RESPONSE_STATUS_NOT_2XX`:响应状态码不是`2xx`时执行重试
2. `OCCUR_IO_EXCEPTION`:发生IO异常时执行重试
3. `OCCUR_EXCEPTION`:发生任意异常时执行重试
#### 声明式重试
如果只有一部分请求需要重试,可以在相应的接口或者方法上使用`@Retry`注解。
#### 请求重试自定义扩展
如果需要修改请求重试行为,可以继承`RetryInterceptor`,并将其配置成`Spring bean`。
### 错误解码器
在`HTTP`发生请求错误(包括发生异常或者响应数据不符合预期)的时候,错误解码器可将`HTTP`相关信息解码到自定义异常中。你可以在`@SporeClient`注解的`errorDecoder()`
指定当前接口的错误解码器,自定义错误解码器需要实现`ErrorDecoder`接口:
### 微服务之间的HTTP调用
#### 继承`ServiceInstanceChooser`
用户可以自行实现`ServiceInstanceChooser`接口,完成服务实例的选取逻辑,并将其配置成`Spring Bean`。对于`Spring Cloud`
应用,可以使用如下实现。
```java
@Service
public class SpringCloudServiceInstanceChooser implements ServiceInstanceChooser {
private LoadBalancerClient loadBalancerClient;
@Autowired
public SpringCloudServiceInstanceChooser(LoadBalancerClient loadBalancerClient) {
this.loadBalancerClient = loadBalancerClient;
}
@Override
public URI choose(String serviceId) {
ServiceInstance serviceInstance = loadBalancerClient.choose(serviceId);
Assert.notNull(serviceInstance, "can not found service instance! serviceId=" + serviceId);
return serviceInstance.getUri();
}
}
```
#### 指定`serviceId`和`path`
```java
@SporeClient(serviceId = "${jy-helicarrier-api.serviceId}", path = "/m/count")
public interface ApiCountService {}
```
## 全局拦截器
### 全局应用拦截器
如果我们需要对整个系统的的`HTTP`请求执行统一的拦截处理,可以实现全局拦截器`GlobalInterceptor`, 并配置成`spring Bean`。
```java
@Component
public class SourceGlobalInterceptor implements GlobalInterceptor {
@Autowired
private TestService testService;
@Override
public Response intercept(Chain chain) throws IOException {
Request request = chain.request();
Request newReq = request.newBuilder()
.addHeader("source", "test")
.build();
testService.test();
return chain.proceed(newReq);
}
}
```
### 全局网络拦截器
实现`NetworkInterceptor`接口,并配置成`spring Bean`。
## 调用适配器
`Retrofit`可以通过`CallAdapterFactory`将`Call`对象适配成接口方法的返回值类型。组件扩展了一些`CallAdapterFactory`实现:
1. `BodyCallAdapterFactory`
- 同步执行`HTTP`请求,将响应体内容适配成方法的返回值类型。
- 任意方法返回值类型都可以使用`BodyCallAdapterFactory`,优先级最低。
2. `ResponseCallAdapterFactory`
- 同步执行`HTTP`请求,将响应体内容适配成`Retrofit.Response`返回。
- 只有方法返回值类型为`Retrofit.Response`,才可以使用`ResponseCallAdapterFactory`。
3. 响应式编程相关`CallAdapterFactory`
**`Retrofit`会根据方法返回值类型选择对应的`CallAdapterFactory`执行适配处理**,目前支持的返回值类型如下:
- `String`:将`Response Body`适配成`String`返回。
- 基础类型(`Long`/`Integer`/`Boolean`/`Float`/`Double`):将`Response Body`适配成上述基础类型
- 任意`Java`类型: 将`Response Body`适配成对应的`Java`对象返回
- `CompletableFuture`: 将`Response Body`适配成`CompletableFuture`对象返回
- `Void`: 不关注返回类型可以使用`Void`
- `Response`: 将`Response`适配成`Response`对象返回
- `Call`: 不执行适配处理,直接返回`Call`对象
- `Mono`: `Project Reactor`响应式返回类型
- `Single`:`Rxjava`响应式返回类型(支持`Rxjava2/Rxjava3`)
- `Completable`:`Rxjava`响应式返回类型,`HTTP`请求没有响应体(支持`Rxjava2/Rxjava3`)
```java
@SporeClient(baseUrl = "${test.baseUrl}")
public interface HttpApi {
@POST("getString")
String getString(@Body Person person);
@GET("person")
Result getPerson(@Query("id") Long id);
@GET("person")
CompletableFuture> getPersonCompletableFuture(@Query("id") Long id);
@POST("savePerson")
Void savePersonVoid(@Body Person person);
@GET("person")
Response> getPersonResponse(@Query("id") Long id);
@GET("person")
Call> getPersonCall(@Query("id") Long id);
@GET("person")
Mono> monoPerson(@Query("id") Long id);
@GET("person")
Single> singlePerson(@Query("id") Long id);
@GET("ping")
Completable ping();
}
```
可以通过继承`CallAdapter.Factory`扩展`CallAdapter`。
组件支持通过`retrofit.global-call-adapter-factories`配置全局调用适配器工厂:
```yaml
retrofit:
# 全局转换器工厂(组件扩展的`CallAdaptorFactory`工厂已经内置,这里请勿重复配置)
global-call-adapter-factories:
# ...
```
针对每个Java接口,还可以通过`@SporeClient.callAdapterFactories`指定当前接口采用的`CallAdapter.Factory`。
> 建议:将`CallAdapter.Factory`配置成`Spring Bean`
### 数据转码器
`Retrofit`使用`Converter`将`@Body`注解的对象转换成`Request Body`,将`Response Body`转换成一个`Java`对象,可以选用以下几种`Converter`:
- [Gson](https://github.com/google/gson): com.squareup.Retrofit:converter-gson
- [Jackson](https://github.com/FasterXML/jackson): com.squareup.Retrofit:converter-jackson
- [Moshi](https://github.com/square/moshi/): com.squareup.Retrofit:converter-moshi
- [Protobuf](https://developers.google.com/protocol-buffers/): com.squareup.Retrofit:converter-protobuf
- [Wire](https://github.com/square/wire): com.squareup.Retrofit:converter-wire
- [Simple XML](http://simple.sourceforge.net/): com.squareup.Retrofit:converter-simplexml
- [JAXB](https://docs.oracle.com/javase/tutorial/jaxb/intro/index.html): com.squareup.retrofit2:converter-jaxb
- fastJson:com.alibaba.fastjson.support.retrofit.Retrofit2ConverterFactory
组件支持通过`retrofit.global-converter-factories`配置全局`Converter.Factory`,默认的是`retrofit2.converter.jackson.JacksonConverterFactory`。
如果需要修改`Jackson`配置,自行覆盖`JacksonConverterFactory`的`bean`配置即可。
```yaml
retrofit:
# 全局转换器工厂
global-converter-factories:
- retrofit2.converter.jackson.JacksonConverterFactory
```
针对每个`Java`接口,还可以通过`@SporeClient.converterFactories`指定当前接口采用的`Converter.Factory`。
> 建议:将`Converter.Factory`配置成`Spring Bean`。
### 元注解
`@SporeClient`、`@Retry`、`@SporeLogging`、`@Resilience4jDegrade`等注解支持元注解、继承以及`@AliasFor`。
```java
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Inherited
@SporeClient(baseUrl = "${test.baseUrl}")
@SporeLogging(logLevel = LogLevel.WARN)
@Retry(intervalMs = 200)
public @interface MyRetrofitClient {
@AliasFor(annotation = RetrofitClient.class, attribute = "converterFactories")
Class extends Converter.Factory>[] converterFactories() default {GsonConverterFactory.class};
@AliasFor(annotation = Logging.class, attribute = "logStrategy")
LogStrategy logStrategy() default LogStrategy.BODY;
}
```
## 其他功能示例
### form参数
```java
@FormUrlEncoded
@POST("token/verify")
Object tokenVerify(@Field("source") String source,@Field("signature") String signature,@Field("token") String token);
@FormUrlEncoded
@POST("message")
CompletableFuture