Broker Agnostic Microservices: Simplifying Messaging Platform Integration with Spring Cloud Stream and Functions

Sugumar Panneerselvam
5 min readMar 14, 2024

--

Photo by Kelly Sikkema on Unsplash

Introduction

Event-driven architecture (EDA) has gained popularity in modern software development due to its ability to build highly scalable and responsive systems. By decoupling components and allowing them to communicate asynchronously through events, EDA enables better scalability, fault tolerance, and flexibility.

However, one challenge in implementing EDA is handling the communication between different components and ensuring interoperability. This is where an Agnostic approach comes into play. Agnostic frameworks, like Spring Cloud Stream, provide a unified way to interact with various messaging systems, such as RabbitMQ, Apache Kafka, or Google Pub/Sub, without needing to worry about the specific implementation details of each platform.

Understanding Agnostic Approach

When considering an “agnostic” approach, one often ponders the frequency of changes in integrations. Yet, the true essence lies not in the frequency of integration changes, but rather in the system’s flexibility and readiness to adapt to change.

For instance, during the migration from Spring Boot 2.x to 3, some systems faced challenges due to inadequate preparation and evaluation for change. This highlights the importance of abstraction in building maintainable systems.

Spring Cloud Stream embraces this concept of abstraction, allowing developers to focus on building robust business logic while the framework handles the complexities of messaging platform integration seamlessly. By adopting an agnostic approach, developers can future-proof their systems and ensure they are adaptable to evolving requirements and technologies.

Problem Statement: Simplifying Messaging Platform Integration

In this article, we’ll examine a straightforward example to demonstrate how Spring Cloud Stream facilitates the effortless configuration of consumers. Consider a scenario of an e-commerce platform where multiple microservices handle different functionalities like inventory management, order processing, and shipping. With event-driven architecture, when a new order is placed, an event is emitted and consumed by relevant microservices asynchronously.

Integrating different messaging platforms into a microservices architecture can be complex and time-consuming. Developers often have to deal with varying APIs, protocols, and configurations for each platform, which distracts them from focusing on the business logic. Spring Cloud Stream addresses this challenge by providing a consistent abstraction layer that hides the complexity of interacting with different messaging systems, allowing developers to concentrate on writing simple, clean code.

Version 3 introduces significant enhancements, making it even easier to create scalable and responsive systems. One of its core features is the support for functions, allowing developers to define simple business logic as functions that react to events.

Foundational concepts

Bindings: These are a set of interfaces that declaratively identify the input and output channels.

Binder: This refers to the messaging middleware implementation, such as Kafka or RabbitMQ.

Channel: This represents the communication conduit between the messaging middleware and the application.

Here’s a three-step approach to creating producers and consumers using Spring Cloud Stream.

Step 1 : Define the event

In this step, we define the structure of the event that will be propagated through the system. Events serve as the building blocks of our event-driven architecture, encapsulating meaningful data that triggers actions within our microservices ecosystem. Here, we create a Java record representing an ‘OrderAcceptedMessage’, which includes essential information such as the order ID. Defining clear and concise events ensures that our system communicates effectively and maintains a shared understanding of the data being processed.

public record OrderAcceptedMessage(Long orderId){

}

Step 2 : Implement the business logic as a simple function

Once we have defined our event, we proceed to implement the business logic that will react to this event. In this step, we encapsulate the processing logic within a simple function. By using Spring Cloud Stream’s functional programming model, we can focus on the core business logic without being burdened by the intricacies of message handling.

@Configuation
public class DispatchingFunctions {
@Bean
public Function<OrderAcceptedMessage, Long> processOrder() {
return orderAcceptedMessage -> {
log.info("The order id {} is processed.", orderAcceptedMessage.orderId());
}
}
}

Step 3 : Configure a consumer to handle the binding and consumption of events

With our business logic defined, we now configure a consumer to handle the binding and consumption of events within our messaging infrastructure.

In this configuration, we specify the input binding for our ‘processOrder’ function, indicating the destination where events of type ‘OrderAcceptedMessage’ will be received. Additionally, we provide the necessary details for connecting to our messaging broker, such as RabbitMQ and following the naming convention: methodname-in-0 eg., processorder-in-0 is logical name only, RabbitMQ doesn’t aware about them.

<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-stream-binder-rabbit</artifactId>
</dependency>
spring:
cloud:
function: processOrder
stream:
bindings:
processOrder-in-0:
destination: order-accepted
rabbitmq:
host: localhost
port: 5673
username: guest
password: guest
connection-timeout: 5s

The main advantage of Spring Cloud Stream is its level of abstraction — it shields developers from having to worry about the underlying messaging infrastructure, whether it’s RabbitMQ, Kafka, or any other supported platform. Integration testing is also simplified with Spring Cloud Stream, allowing developers to verify their applications’ behavior in a technology-agnostic context and even test against specific brokers using tools like Testcontainers. [More on following article]

When you start the application, Spring Cloud Stream automatically creates the exchange and queue based on binding names mentioned in the configuration.

Migration from RabbitMQ to Kafka with Spring Cloud Stream

To illustrate the flexibility of Spring Cloud Stream, let’s consider a scenario where an existing application is using RabbitMQ as its messaging platform and needs to migrate to Kafka.

With Spring Cloud Stream, this migration becomes straightforward. By updating the configuration to use Kafka as the binder, Spring Cloud Stream handles the underlying differences between RabbitMQ and Kafka, allowing the application to seamlessly transition without rewriting significant portions of code.

<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-stream-binder-kafka</artifactId>
</dependency>
spring:
cloud:
function:
definition: processOrder-in-0
stream:
bindings:
numberConsumer-in-0:
destination: order-accepted
kafka:
binder:
brokers: localhost:9092

Conclusion

Spring Cloud Stream offers a powerful solution for building event-driven microservices, simplifying messaging platform integration, and allowing developers to focus on writing clean, scalable code. By abstracting away the complexities of interacting with different messaging systems, Spring Cloud Stream enables developers to create flexible and responsive systems that can adapt to changing business requirements with ease.

The Agnostic approach provided by Spring Cloud Stream ensures interoperability across various messaging platforms, making it an ideal choice for building robust and scalable event-driven architectures. As organizations continue to embrace microservices and event-driven design patterns, tools like Spring Cloud Stream will play a crucial role in accelerating development and driving innovation in the software industry.

Next Steps: Exploring Detailed Use Cases with Working Prototypes

In our next article, we will delve deeper into the capabilities of Spring Cloud Stream and explore detailed use cases with working prototypes. We will demonstrate how Spring Cloud Stream can be leveraged to build complex event-driven architectures that address real-world challenges in domains such as e-commerce and finance.

Stay tuned for an exciting journey into the world of event-driven architecture with Spring Cloud Stream and Function.

Related article — https://medium.com/@sugumar.p/checking-business-logic-without-message-broker-needs-a431517a84f3

--

--