DEV Community

Cover image for 🍃Taste Of Spring Reactive Stack
Abdulcelil Cercenazi
Abdulcelil Cercenazi

Posted on

🍃Taste Of Spring Reactive Stack

Spring's Servlet Stack ⚙️

The most common use of Spring nowadays is the servlet stack usage. This method assigns a new thread to each new incoming request and follows a thread blocking execution behavior.

Those two properties make Servlet Spring a bad candidate for future backend servers that will serve millions of requests in very brief amounts of time.

So what's the solution?☹️

Spring's Reactive Stack ♻️

The most two important properties of the reactive stack

▶️ Asynchronous

if the thread reaches a line of code that requires it to wait for a response, the thread doesn't sit around waiting, it handles other work while the response becomes available.

▶️ Declarative

it's a programming paradigm that expresses the logic of a computation(What to do) without describing its control flow(How to do).
for example "For each element in this list, apply this logic and collect the result into a list"

How does it work? 🧐

It uses two main constructs called the Mono and Flux.

Mono 👈

Emits at most one item via the onNext signal then terminates with an onComplete signal (successful Mono, with or without value), or only emits a single onError signal (failed Mono)

Flux 👈

Emits 0 to N elements, and then completes (successfully or with an error).

Furthermore, it provides two options

1️⃣ Streaming

  • with backpressure control (the client and server can talk about slowing down the data flow if the client is having trouble keeping up with the data flow speed)

2️⃣ Non-streaming

Simple Demo 👀

We are going to write a backend server, which populates a single dummy Post entity and saves it into the database. We will offer some endpoints to perform CRUD operations on that entity and add new Posts.
let's start with the database. we will use MongoDB simply because it's reactive in nature and integrated well with our Reactive stack.

🛂

@Data @AllArgsConstructor @NoArgsConstructor
@Builder @Document
public class Post {
    @Id
    private String id;
    private Date publicationDate;
    private String title;
    private User owner;
}
Enter fullscreen mode Exit fullscreen mode

📂

The Spring Data repository

@Repository
public interface PostRepository extends ReactiveCrudRepository<Post, String> {
}
Enter fullscreen mode Exit fullscreen mode

👨🏿‍✈️

The Post Service

@Service
@RequiredArgsConstructor
@Slf4j
public class PostService {
    private final PostRepository postRepository;
    // check the code below
}

// So, let's take a look at the reactive methods
// the methods from portRepository provide methods that return Flux and Mono objects
// since it extends the ReactiveCrudRepository interface

public Flux<Post> getAllPosts(){
    return postRepository.findAll();
}
public Mono<Post> getPostById(String id) {
    return postRepository.findById(id);
}
public Mono<Post> addNewPost(Post postFromClient) {
    return postRepository.save(postFromClient);
}
Enter fullscreen mode Exit fullscreen mode

The delete Post method needed a bit more code

public Mono<ResponseEntity<Object>> deletePost(String id) {
    return postRepository.
            findById(id).
            map(Optional::ofNullable). // => get Optional 
            defaultIfEmpty(Optional.empty()). // 
            doOnSuccess(post -> checkAndDeletePost(post, id)).
            thenReturn(ResponseEntity.ok().build());
}
private void checkAndDeletePost(Optional<Post> post, String id) {
    post.ifPresentOrElse(
        p -> postRepository.delete(p).subscribe(),
        () -> log.info(String.format("%s Post doesn't exist", id)));
}
Enter fullscreen mode Exit fullscreen mode

The update Post method

public Mono<ResponseEntity<Object>> updatePost(Post updatedPost){
    return postRepository.
        findById(updatedPost.getId()).
        map(Optional::ofNullable).
        defaultIfEmpty(Optional.empty()).
        doOnSuccess(oldPost ->
                checkAndUpdatePost(oldPost, updatedPost)).
        thenReturn(ResponseEntity.ok().build());
}

private void checkAndUpdatePost(Optional<Post> oldPost, Post updatedPost) {
    oldPost.ifPresentOrElse(
        oldP -> postRepository.save(updatedPost).subscribe(),
        () -> log.info(String.format("%s Post doesn't exist",
                                    updatedPost.getId())));
}
Enter fullscreen mode Exit fullscreen mode

Code on GitHub

Resources to go deeper

GOTO 2019 • Reactive Spring • Josh Long

Top comments (0)