preface

As we all know, the biggest feature of Spring Boot is automatic assembly, which simplifies dependency and enables us to build projects quickly. After using Spring Boot, we can use some other framework in our project by simply introducing the corresponding Starter dependency.

Have you actually developed some basic features that you need to use in your Spring Boot project?

These features may be needed in other projects, and it is not a good idea to write the same features in another project.

We can make our functionality into a corresponding Starter module and add it directly as a Maven dependency when we need to use it in a project. This way, not only do we not have to develop it repeatedly, but we also have better version management when the functionality is upgraded.

Build a spring-boot-starter

Let’s build a spring-boot-starter. For the sake of description, assume that the primary function of our starter is to easily send and receive asynchronous events. Note that the purpose of this article is to build a Starter, so the implementation details of sending and receiving asynchronous events will not be described.

Basic concept

Before we can start building, we need to synchronize a few basic concepts.

Application Context

If you know anything about Spring, you probably know the Application Context, which is a container used to manage Spring beans, It contains all the Controller, Service, Repository, Component objects in our project.

@Configuration

@Configuration is an annotation in Spring. The class annotated with this annotation is like Spring’s XML Configuration file. Its main purpose is to provide some Bean objects to the Application Context. You can think of a class labeled @Configuration as a factory of Bean objects.

Auto-Configuration

There is an @enableAutoConfiguration annotation in Spring Boot that reads the classes specified under EnableAutoConfiguration in the Spring. factories file, To initialize all @bean methods under the specified class, and initialize the Bean. And you can specify conditions that determine whether you want to auto-assemble Bean objects.


Create a project

Next we’ll start our Starter build, which we’ll call spring-boot-starter-EventMessage.

Suppose we need to implement a microservice environment that allows asynchronous communication between services. Our spring-boot-starter-EventMessage provides the following functions:

  • First of all, we need to have oneEventPublisherObject that is used to send events to the message center;
  • And then you have to have oneEventListenerClass to subscribe to specific events from the message center;

Because our Starter will be used in multiple microservice applications, our Starter will need to be able to build using Maven for other applications to introduce.

Start by creating a Maven project spring-boot-starter-EventMessage.

The pom.xml file is as follows:


      
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.heiz123</groupId>
    <artifactId>spring-boot-starter-eventmessage</artifactId>
    <version>1.0.0</version>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
            <version>2.4.2</version>
        </dependency>
    </dependencies>

</project>
Copy the code

Let the starter autoassemble

Next we need to provide an entry to our Starter feature. We need to provide a @Configuration class:

@Configuration
public class EventAutoConfiguration {

    @Bean
    public EventPublisher eventPublisher(List<EventListener> listeners) {
        return new EventPublisher(listeners);
    }

Copy the code

EventAutoConfiguration includes all the @Bean definitions we need to provide the Starter feature. Because our Starter only needs to provide event publishing functionality, we need to add an EventPublisher object to the ApplicationContext.

Our EventPublisher needs to know all eventListeners so it can pass events to them, so we let Spring inject a list of all available EventListeners in ApplicationContext.

To get our Starter to automatically configure, we need to create a Spring.factories file in the meta-INF/directory.

To configure our Configuration class in this file:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.heiz123.event.EventAutoConfiguration
Copy the code

Spring Boot searches all spring.Factories files in your classpath and loads the configuration declared in them.

With the EventAutoConfiguration class, we now have an entry point that can automatically assemble spring-boot-starter.

Make Starter support optional

In a real development scenario, if a feature needs to be turned off and not turned on until the test environment or online environment, our starter should support turning it off or off.

We can use Conditional annotations for Spring Boot to make our Configuration optional:

@Configuration
@ConditionalOnProperty(value = "eventstarter.enabled", havingValue = "true")
@ConditionalOnClass(name = "com.heiz123.RabbitConnector")
@Slf4j
public class EventAutoConfiguration {

    @Bean
    public EventPublisher eventPublisher(List<EventListener> listeners) {
        log.info("Event-message autowiring...");
        return newEventPublisher(listeners); }}Copy the code

The @conditionalonProperty annotation adds EventAutoConfiguration to the ApplicationContext only if the Eventstarter. Enabled attribute is set to true. In this way, the application that uses our starter can dynamically control whether our starter is enabled through the eventStarter. Enabled property.

@ ConditionalOnClass annotation is used only in the classpath. Com heiz123. Activate the automatic configuration RabbitConnector class.

Make the Starter configurable

In our EventAutoConfiguration, we need to inject all eventListeners in the application into our EventPublisher, but it is possible that the application using our Starter does not want all eventListeners to be processed. If only a few events are concerned, we should enable users to easily configure them.

For example, in the application.properties file or application.yml file, you can do the following:

eventstarter.listener.enabled-events:foo,bar
Copy the code

Yml file:

eventstarter:
  listener:
    enabled-events:
      - foo
      - bar
Copy the code

To make these properties easily accessible in our Starter code, we can provide a @ConfigurationProperties class:

@ConfigurationProperties(prefix = "eventstarter.listener")
class EventListenerProperties {

    /** * the type of all events that will be passed in EventListener. * Other events will be ignored */
    private List<String> enabledEvents = Collections.emptyList();

    public List<String> getEnabledEvents(a) {
        return enabledEvents;
    }

    public void setEnabledEvents(List<String> enabledEvents) {
        this.enabledEvents = enabledEvents; }}Copy the code

Then we through the use of @ EnableConfigurationProperties annotation our entry point configuration to enable EventListenerProperties class:

@Configuration
@EnableConfigurationProperties(EventListenerProperties.class)
@Slf4j
public class EventAutoConfiguration {

    @Bean
    public EventPublisher eventPublisher(List<EventListener> listeners) {
        log.info("Event-message autowiring...");
        return newEventPublisher(listeners); }}Copy the code

Finally, we can inject the EventListenerProperties object where needed, such as filtering out unwanted events in the abstract EventListener class:

public abstract class EventListener {

    private final EventListenerProperties properties;

    public EventListener(EventListenerProperties properties) {
        this.properties = properties;
    }

    public void receive(Event event) {
        // If the event is Eanbled and subscribed.
        if(isEnabled(event) && isSubscribed(event)) { onEvent(event); }}private boolean isSubscribed(Event event) {
        return event.getType().equals(getSubscribedEventType());
    }

    private boolean isEnabled(Event event) {
        return properties.getEnabledEvents().contains(event.getType());
    }

    protected abstract String getSubscribedEventType(a);

    protected abstract void onEvent(Event event);
}
Copy the code

At this point, our Starter can basically be used by others.

Begin to use

Practice is the sole criterion for testing truth. Let’s introduce our spring-boot-starter-EventMessage in a new project.

Start by creating a spring-boot project, which we’ll call test-starter.

Then we just need to add the Starter dependency coordinate information in Maven.

        <dependency>
            <groupId>com.heiz123</groupId>
            <artifactId>spring-boot-starter-eventmessage</artifactId>
            <version>1.0.0</version>
        </dependency>
Copy the code

Next, launch our SpringBoot project. The startup classes are as follows:

@SpringBootApplication
public class Main {

    public static void main(String[] args) {
        try {
            SpringApplication.run(Main.class, args);
        } catch(Throwable t) { t.printStackTrace(); }}}Copy the code

Running results:

Spring-boot-starter-eventmessage has been successfully integrated.

conclusion

We can wrap certain features and functions into the Starter program for easy and quick use in any other Spring Boot application. There are just a few simple steps to building a Starter:

Provides the automatic Configuration class Configuration, which allows it to be disabled and enabled, and provides configurable parameters for some variable information.

That’s all for this episode. If it helped, give me a thumbs up!

I am xiao Hei, a programmer in the Internet “casual”.

Water does not compete, you are in the flow