From b4662434ab1f3d3d4196b62e71905deea715993b Mon Sep 17 00:00:00 2001 From: Julien Ponge Date: Tue, 12 Mar 2024 19:42:40 +0100 Subject: [PATCH] docs: addition of a reactive pitfalls reference This reference page is to be completed with our observations of how users might have incorrect assumptions on how to use Mutiny and reactive in general. --- .../going-reactive-a-few-pitfalls.md | 74 +++++++++++++++++++ documentation/mkdocs.yml | 1 + .../java/guides/reference/PitfallsTest.java | 64 ++++++++++++++++ 3 files changed, 139 insertions(+) create mode 100644 documentation/docs/reference/going-reactive-a-few-pitfalls.md create mode 100644 documentation/src/test/java/guides/reference/PitfallsTest.java diff --git a/documentation/docs/reference/going-reactive-a-few-pitfalls.md b/documentation/docs/reference/going-reactive-a-few-pitfalls.md new file mode 100644 index 000000000..a4fbd2546 --- /dev/null +++ b/documentation/docs/reference/going-reactive-a-few-pitfalls.md @@ -0,0 +1,74 @@ +--- +tags: +- reference +- beginner +--- + +# Going reactive: a few pitfalls + +Don't get us wrong, reactive programming is a fantastic way to write resource-efficient code! + +That being said, reactive programming has a learning curve that should not be taken lightly, and in some cases it is safer to write imperative code that you fully comprehend over reactive code that you don't fully grok. + +We have assembled a few considerations that we think new users should know before they embark into writing complex reactive business logic. + +## Mutiny doesn't auto-magically make your code asynchronous + +This is a common source of confusion for new reactive programmers. +Mutiny itself **does not perform any scheduling work**, except for the [`emitOn` and `runSubscriptionOn` operators](../guides/emit-on-vs-run-subscription-on.md). + +Consider the following code where we _join_ results from multiple asynchronous operations, materialised by the `Uni`-returning `fetch` method: + +```java linenums="1" +{{ insert('java/guides/reference/PitfallsTest.java', 'noMagicJoin') }} +``` + +You might think that the `join` operator schedules the calls to `fetch` to be run concurrently, and then collects the results into a list. +This is not how it works! + +The `join` operator does subscribe to each `Uni` returned by each call to `fetch`. +When it receives a value, it puts it into a list, and when all values have been received, that list is emitted. +The threads involved here are the ones that emit values in `fetch`. +If `fetch` uses async I/O underneath then you should observe true concurrency, but if `fetch` just emits a value right when the subscription happens then you will merely observe a sequential execution of each call to `fetch`, in order. + +## When to prefer `Uni>` over `Multi` + +The reason why `Multi` exists is to model streams over back-pressured sources. +By conforming to the [Reactive Streams protocol](https://www.reactive-streams.org/), a `Multi` respects the control flow requests from its subscribers, avoiding classic problems such as a fast producer and a slow consumer that can yield to memory exhaustion problems. + +That being said, not everything is a stream. +Take the example of relational databases: **databases don't stream!** (for the most parts) + +When you do a query such as `SELECT * FROM ABC WHERE INDEX < 123`, you get result rows. +While you might wrap the results in a `Multi` as a convenience, the network protocol of the database still sends you all `Row` values and is very unlikely to support any notion of back-pressure on a SQL query result. + +This is why `Uni>` is in this case a better representation of an asynchronous operation than `Multi`, because the underlying networked service protocol does not provide you with any back-pressured stream. + +## Creating `Uni` and `Multi` from in-memory data might be suspicious + +You will find lots of occurrences of creating `Uni` and `Multi` from in-memory data in this documentation, as in: + +```java linenums="1" +{{ insert('java/guides/reference/PitfallsTest.java', 'inMemoryData') }} +``` + +This is convenient and expected when creating tests and examples, but this should be a strong warning in production. +Indeed, if we have a method such as the following: + +```java linenums="1" +{{ insert('java/guides/reference/PitfallsTest.java', 'suspiciousPublisher') }} +``` + +then it is clear that there is nothing _"reactive"_ in this code _(sadly, you can find such idioms in some well-known "reactive" client libraries, but we digress)_. + +As a rule of thumb, if your **initial** publisher does not make any I/O operation and it already has the data available in memory, then it is suspicious: + +- if it is a `Uni`, then it does not really model an asynchronous I/O operation because the data is already here, and +- if it is a `Multi` then not only there is no asynchronous I/O operation involved, but there is no need for a back-pressure protocol either (see the previous section). + +What is not suspicious however is to create, say, a `Multi` to perform a transformation operation: + +```java linenums="1" +{{ insert('java/guides/reference/PitfallsTest.java', 'flatmap-ism') }} +``` + diff --git a/documentation/mkdocs.yml b/documentation/mkdocs.yml index 148f6c5dd..7ce1f5292 100644 --- a/documentation/mkdocs.yml +++ b/documentation/mkdocs.yml @@ -59,6 +59,7 @@ nav: - 'reference/what-is-reactive-programming.md' - 'reference/what-makes-mutiny-different.md' - 'reference/uni-and-multi.md' + - 'reference/going-reactive-a-few-pitfalls.md' - 'reference/publications.md' - 'Tags index': 'tags-index.md' - 'API': 'https://javadoc.io/doc/io.smallrye.reactive/mutiny/latest/index.html' diff --git a/documentation/src/test/java/guides/reference/PitfallsTest.java b/documentation/src/test/java/guides/reference/PitfallsTest.java new file mode 100644 index 000000000..b2b6cce81 --- /dev/null +++ b/documentation/src/test/java/guides/reference/PitfallsTest.java @@ -0,0 +1,64 @@ +package guides.reference; + +import io.smallrye.mutiny.Multi; +import io.smallrye.mutiny.Uni; +import org.junit.jupiter.api.Test; + +import java.util.List; + +public class PitfallsTest { + + private Uni fetch(String id) { + return Uni.createFrom().item("Yolo"); + } + + @Test + public void noMagicScheduling() { + // + Uni> data = Uni.join().all( + fetch("abc"), + fetch("def"), + fetch("123")).andFailFast(); + // + } + + @Test + public void inMemoryData() { + // + Uni uni = Uni.createFrom().item(123); + Multi multi = Multi.createFrom().items(1, 2, 3, 4, 5); + // + } + + private List obtainValues(String key) { + return List.of(); + } + + // + public Multi fetchData(String key) { + List strings = obtainValues(key); + return Multi.createFrom().iterable(strings); + } + // + + private Multi streamData(String key) { + return Multi.createFrom().items(""); + } + + private Multi extractData(String line) { + return Multi.createFrom().items(""); + } + + private Multi flatmap() { + // + Multi stream = streamData("abc"); + return stream.onItem().transformToMultiAndMerge(line -> { + if (line.trim().length() > 10) { + return extractData(line); + } else { + return Multi.createFrom().item("[N/A]"); + } + }); + // + } +}