SpringBoot

SpringBoot 优雅控制API版本

2022-08-20  本文已影响0人  极简博客

在实际工作中,由于接口功能做了不兼容变更,因此需要在原接口基础上增加版本控制,以示区分。

何时进行版本控制?这里引用REST API版本控制描述

只有在进行重大更改时才需要对API进行升级。突破性变化包括:

中断更改应始终导致更改API或内容响应类型的主版本号。

不间断更改(例如添加新端点或新响应参数)不需要更改主版本号。但是,在进行更改以跟踪可能正在接收缓存版本数据或可能遇到其他API问题的客户时,跟踪API的次要版本会很有帮助。

1. 常见版本控制策略

方式 说明
子域名 不同版本使用不同域名,v1.test.com,v2.test.com
请求URL 同一域名,不同路径,test.com/v1/,test.com/v2/
请求参数 根据参数区分相应功能

2. 设计建议

3. 实现

通过重写RequestMappingHandlerMapping更改URL分发

public class VersionConfig extends WebMvcConfigurationSupport {
    /**
     * 注解@Version(2)将分发至 /v2/** 默认: /v1/** || /**
     *
     * @param contentNegotiationManager
     * @param conversionService
     * @param resourceUrlProvider
     * @return
     */
    @Override
    public RequestMappingHandlerMapping requestMappingHandlerMapping(ContentNegotiationManager contentNegotiationManager, FormattingConversionService conversionService, ResourceUrlProvider resourceUrlProvider) {
        return new RequestMappingHandlerMapping() {
            @Override
            protected RequestMappingInfo getMappingForMethod(Method method, Class<?> type) {
                RequestMappingInfo rmi = super.getMappingForMethod(method, type);
                if (Objects.isNull(rmi)) {
                    return null;
                }
                Version version = Optional.ofNullable(AnnotationUtils.findAnnotation(method, Version.class)).orElseGet(() -> AnnotationUtils.findAnnotation(type, Version.class));
                return Objects.nonNull(version)
                        ? RequestMappingInfo.paths(Arrays.stream(version.value()).mapToObj(i -> "v" + i).toArray(String[]::new)).build().combine(rmi)
                        : RequestMappingInfo.paths(new String[]{"v1", ""}).build().combine(rmi);
            }
        };
    }
}

示例代码

上一篇下一篇

猜你喜欢

热点阅读