快速构建Spring Boot Starter指南

前言

对于正在构建的Spring Boot应用程序,我们不希望每次都从0开始实现某些特定功能。相反,我们希望仅实现功能一次,并根据需要将该功能包含在任何应用中。一次实现,处处引用。

在Spring Boot中,启动器就是完成这种任务的利器。启动器可以轻松的包含一组特定功能,并能轻松的使用它们。

使用Spring Boot启动器的场景:

  • 提供可配置和/或默认的日志配置,或使日志轻易的记录到中心日志服务器。
  • 提供可配置和/或安全配置。
  • 提供可配置和/或默认的错误处理策略。
  • 为消息中间件提供适配器。
  • 集成第三方库并使其可配置为与Spring Boot一起使用

在本文中,我们将构建一个Spring Boot启动器,它允许Spring Boot应用程序使用华为云提供的OBS服务存储和读取数据。

样例代码

样例代码可以obs-starter获取。

Spring Boot启动器基本概念

在深入了解创建Spring Boot启动器的细节之前,让我们讨论有助于理解启动器工作原理的基本概念。

什么是应用上下文?


在一个Spring应用中,应用上下文是组成应用对象(beans)的网络。应用上下文包含web controller,service,repository和一切应用运行需要的对象,这些对象通常都是无状态的。

什么是 Spring 配置?


使用@Configuration注解的类充当添加到应用程序上下文bean的工厂。可能包含多个工厂方法(@Bean),方法的返回值会被Spring自动添加到应用上下文中。

简而言之,Spring Configuration为应用上下文提供beans。

什么是自动配置?


自动配置是一个可以被Spring自动发现的配置类(使用@Configuration 注解)。只要在类路径上找到自动配置,Spring就会对其进行计算,并将配置的贡献(比如,创建Bean)添加到应用程序上下文中。自动配置可以是有条件的,自动配置生效可以依赖外部因素,比如具有特定值的配置(一切皆是受控的)。

什么是自动配置模块?


自动配置模块是包含自动配置类的Maven或Gradle模块。这样,我们就可以构建自动为应用程序上下文做贡献的模块,添加某个功能或提供对某个外部库的访问。我们在Spring Boot应用程序中使用它所要做的唯一工作就是在pom.xmlbuild.gradle中包含一个依赖项。

Spring Boot团队大量使用此方法将Spring Boot与外部库集成。

什么是 Spring Boot 启动器?


Spring Boot Starter是一个Maven或Gradle模块,其唯一目的是提供“使用特定功能”的所有依赖项。这通常意味着它是一个单独的pom.xmlbuild.gradle文件,它包含一个或多个自动配置模块的依赖项以及可能需要的任何其他依赖项。在Spring Boot应用中,使用这些功能只需要包含这个启动器即可(添加一个依赖)。

为OBS服务构建一个启动器

下面以一个具体例子讨论如何实现一个启动器。想象一下我们在微服务环境中工作,所有的微服务都需要从OBS服务存储数据或查询数据,我们希望通过一个启动器来实现该功能并被所有微服务共享。OBS启动器会提供以下功能:

  • 一个ObsWriter bean允许微服务向OBS写数据。
  • 一个ObsReader bean允许微服务从OBS读数据。

支持OBS信息可配置


连接华为云OBS服务需要使用AK、SK和Endpoint三个信息,为了支持不同微服务对接不同的OBS服务(不同租户的OBS,不同region的OBS),AK、SK和Endpoint需要支持配置:

@ConfigurationProperties(prefix = "obs.info")
@Configuration
public class ObsInfoProperties {

    @Value(value = "${ak:default ak}")
    private String ak;

    @Value(value = "${sk:default sk}")
    private String sk;

    @Value(value = "${endPoint:https://obs.cn-north-5.myhuaweicloud.com}")
    private String endPoint;
    
    // getter,setter方法
}

微服务可以通过 application.propertiesapplication.yaml 配置需要使用的OBS服务信息:

application.properties 的文件内容如下:

obs.info.ak = your self ak
obs.info.sk =your self sk
obs.info.endPoint = the endpoint you want to connect

application.yaml 的文件内容如下:

obs:
  info:
    ak: your self ak
    sk: your self sk
    enPoint: your self endpoint

提供一个自动配置


作为启动器功能的切入点,我们首先提供一个@Configuration class:

@Configuration
@EnableConfigurationProperties(ObsInfoProperties.class)
public class ObsAutoConfiguration {

    @Autowired
    private ObsInfoProperties obsProperties;

    @Bean
    public ObsWriter obsWriter() {
        String ak = obsProperties.getAk();
        Assert.notNull(ak, "ak must not be null");
        String sk = obsProperties.getSk();
        Assert.notNull(sk, "sk must not be null");
        String endPoint = obsProperties.getEndPoint();
        Assert.notNull(endPoint, "endPoint must not be null");
        return new ObsWriter(ak, sk, endPoint);
    }

    @Bean
    public ObsReader obsReader() {
        String ak = obsProperties.getAk();
        Assert.notNull(ak, "ak must not be null");
        String sk = obsProperties.getSk();
        Assert.notNull(sk, "sk must not be null");
        String endPoint = obsProperties.getEndPoint();
        Assert.notNull(endPoint, "endPoint must not be null");
        return new ObsReader(ak, sk, endPoint);
    }
}

这个配置类包含了OBS功能需要的所有类,ObsWriterObsReader 两个Bean依赖 ObsInfoProperties。为了使配置类成为一个自动配置类,需要在src/main/resource/META-INFO目录下添加 spring.factories 文件,文件的内容为:

# 键值对的值为自动配置类的全限定名
org.springframework.boot.autoconfigure.EnableAutoConfiguration=io.github.ctlove0523.obsstart.ObsAutoConfiguration

ObsWriterObsReader的完整定义可以从Github获取:

ObsWriter 只包含属性的定义如下:

public class ObsWriter {
    private ObsClient obsClient;
    private String ak;
    private String sk;
    private String endPoint;
}

ObsReader 只包含属性的定义如下:

public class ObsReader {
    private ObsClient obsClient;
    private String ak;
    private String sk;
    private String endPoint;
}

让OBS功能成为可选功能

在任何时候允许关闭Spring Boot启动器的某些功能都是非常不错的想法。尤其是涉及对接外部系统时,比如对接华为云的存储服务OBS等。在研发内部环境或者研发测试环境,第三方的服务可能都无法使用,在这种情况下,关闭该功能是更好的选择。

在Spring Boot中有很多方法可以让一个 Bean 成为可选的 Bean ,在这里只是选取最简单的注解方式说明如何让一个Bean成为可选Bean,也就是如何关闭OBS功能,修改后的ObsAutoConfiguration 如下:

@Configuration
@EnableConfigurationProperties(ObsInfoProperties.class)
@ConditionalOnProperty(value = "obs.enabled", havingValue = "true")
public class ObsAutoConfiguration {

    @Autowired
    private ObsInfoProperties obsProperties;

    @Bean
    public ObsWriter obsWriter() {
        // 不详细展示代码细节
        return new ObsWriter(ak, sk, endPoint);
    }

    @Bean
    public ObsReader obsReader() {
        // 不详细展示代码细节
        return new ObsReader(ak, sk, endPoint);
    }
}

EnableConfigurationProperties 告诉Spring Boot,当且仅当obs.enabled 的值为true 时才创建并加载Bean

使用OBS启动器

现在OBS启动器已经开发完成,新构建或正在构建的Spring Boot应用只要包含OBS启动器,就可以使用OBS的功能,如果Spring Boot应用是基于Maven构建,只需要增加如下依赖项目即可开箱即用:

    <dependencies>
        <dependency>
            <groupId>io.github.ctlove0523</groupId>
            <artifactId>obs-spring-boot-starter</artifactId>
            <version>0.0.3-SNAPSHOT</version>
        </dependency>
    </dependencies>

客户端可以在自己的配置文件中对AK,SK,Endpoint,是否开启OBS功能等进行配置:

application.properties

obs.info.ak = client use ak
obs.info.sk = client use sk
obs.info.endPoint = client want to connect endpoint
# 开启OBS功能
obs.enabled = true

参考文章

  1. Creating Your Own Starter
  2. Quick Guide to Building a Spring Boot Starter
  3. 华为OBS
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容