You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Copy file name to clipboardExpand all lines: README.md
+8-7
Original file line number
Diff line number
Diff line change
@@ -80,6 +80,7 @@ followed by a possibly unbounded number of `onNext` signals (as requested by `Su
80
80
| <aname="term_terminal_state">Terminal state</a> | For a Publisher: When `onComplete` or `onError` has been signalled. For a Subscriber: When an `onComplete` or `onError` has been received.|
81
81
| <aname="term_nop">NOP</a> | Execution that has no detectable effect to the calling thread, and can as such safely be called any number of times.|
82
82
| <aname="term_ext_sync">External synchronization</a> | Access coordination for thread safety purposes implemented outside of the constructs defined in this specification, using techniques such as, but not limited to, `atomics`, `monitors`, or `locks`. |
83
+
| <aname="term_thread-safe">Thread-safe</a> | Can be safely invoked synchronously, or asychronously, without requiring external synchronization to ensure program correctness. |
83
84
84
85
### SPECIFICATION
85
86
@@ -97,7 +98,7 @@ public interface Publisher<T> {
97
98
| [:bulb:](#1.1"1.1 explained") |*The intent of this rule is to make it clear that Publishers cannot signal more elements than Subscribers have requested. There’s an implicit, but important, consequence to this rule:Since demand can only be fulfilled after it has been received, there’s a happens-before relationship between requesting elements and receiving elements.*|
98
99
|<a name="1.2">2</a>|A `Publisher` MAY signal fewer `onNext` than requested and terminate the `Subscription` by calling `onComplete` or `onError`.|
99
100
| [:bulb:](#1.2"1.2 explained") |*The intent of this rule is to make it clear that a Publisher cannot guarantee that it will be able to produce the number of elements requested; it simply might not be able to produce them all; it may be in a failed state; it may be empty or otherwise already completed.*|
100
-
|<a name="1.3">3</a>| `onSubscribe`, `onNext`, `onError` and `onComplete` signaled to a `Subscriber` MUST be signaled in a `thread-safe` manner—and if performed by multiple threads—use [external synchronization](#term_ext_sync).|
101
+
|<a name="1.3">3</a>| `onSubscribe`, `onNext`, `onError` and `onComplete` signaled to a `Subscriber` MUST be signaled in a [thread-safe](#term_thread-safe) manner—and if performed by multiple threads—use [external synchronization](#term_ext_sync).|
101
102
| [:bulb:](#1.3"1.3 explained") |*The intent of this rule is to make it clear that [external synchronization](#term_ext_sync) must be employed if the Publisher intends to send signals from multiple/different threads.*|
102
103
|<a name="1.4">4</a>|If a `Publisher` fails it MUST signal an `onError`.|
103
104
| [:bulb:](#1.4"1.4 explained") |*The intent of this rule is to make it clear that a Publisher is responsible for notifying its Subscribersif it detects that it cannot proceed—Subscribers must be given a chance to clean up resources or otherwise deal with the Publisher´s failures.*|
@@ -108,7 +109,7 @@ public interface Publisher<T> {
108
109
|<a name="1.7">7</a>|Once a [terminal state](#term_terminal_state) has been signaled (`onError`, `onComplete`) it is REQUIRED that no further signals occur. |
109
110
| [:bulb:](#1.7"1.7 explained") |*The intent of this rule is to make sure that onError and onComplete are the final states of an interaction between a Publisher and Subscriber pair.*|
110
111
|<a name="1.8">8</a>|If a `Subscription` is cancelled its `Subscriber` MUST eventually stop being signaled. |
111
-
| [:bulb:](#1.8"1.8 explained") |*The intent of this rule is to make sure that Publishers respect a Subscriber’s request to cancel a Subscription when Subscription.cancel() has been called. The reason for*eventually* is because signals can have propagation delay due to being asynchronous.*|
112
+
| [:bulb:](#1.8"1.8 explained") |*The intent of this rule is to make sure that Publishers respect a Subscriber’s request to cancel a Subscription when Subscription.cancel() has been called. The reason for**eventually** is because signals can have propagation delay due to being asynchronous.*|
112
113
|<a name="1.9">9</a>| `Publisher.subscribe` MUST call `onSubscribe` on the provided `Subscriber` prior to any other signals to that `Subscriber` and MUST [return normally](#term_return_normally), except when the provided `Subscriber` is `null` in which case it MUSTthrow a `java.lang.NullPointerException` to the caller, for all other situations the only legal way to signal failure (or reject the `Subscriber`) is by calling `onError` (after calling `onSubscribe`).|
113
114
| [:bulb:](#1.9"1.9 explained") |*The intent of this rule is to make sure that `onSubscribe` is always signalled before any of the other signals, so that initialization logic can be executed by the Subscriber when the signal is received. Also `onSubscribe` MUST only be called at most once, [see [2.12](#2.12)].If the supplied `Subscriber` is `null`, there is nowhere else to signal this but to the caller, which means a `java.lang.NullPointerException` must be thrown. Examples of possible situations:A stateful Publisher can be overwhelmed, bounded by a finite number of underlying resources, exhausted, or in a [terminal state](#term_terminal_state).*|
114
115
|<a name="1.10">10</a>| `Publisher.subscribe` MAY be called as many times as wanted but MUST be with a different `Subscriber` each time [see [2.12](#2.12)].|
@@ -132,7 +133,7 @@ public interface Subscriber<T> {
132
133
|<a name="2.1">1</a>|A `Subscriber` MUST signal demand via `Subscription.request(long n)` to receive `onNext` signals. |
133
134
| [:bulb:](#2.1"2.1 explained") |*The intent of this rule is to establish that it is the responsibility of the Subscriber to signal when, and how many, elements it is able and willing to receive.*|
134
135
|<a name="2.2">2</a>|If a `Subscriber` suspects that its processing of signals will negatively impact its `Publisher`´s responsivity, it is RECOMMENDED that it asynchronously dispatches its signals. |
135
-
| [:bulb:](#2.2"2.2 explained") |*The intent of this rule is that a Subscriber should [not obstruct](#term_non-obstructing) the progress of the Publisher from an execution point-of-view. In other words, the Subscriber should not starve the Publisher from CPU cycles.*|
136
+
| [:bulb:](#2.2"2.2 explained") |*The intent of this rule is that a Subscriber should [not obstruct](#term_non-obstructing) the progress of the Publisher from an execution point-of-view. In other words, the Subscriber should not starve the Publisher from receiving CPU cycles.*|
136
137
|<a name="2.3">3</a>| `Subscriber.onComplete()` and `Subscriber.onError(Throwable t)` MUSTNOT call any methods on the `Subscription` or the `Publisher`.|
137
138
| [:bulb:](#2.3"2.3 explained") |*The intent of this rule is to prevent cycles and race-conditions—between Publisher, Subscription and Subscriber—during the processing of completion signals.*|
138
139
|<a name="2.4">4</a>| `Subscriber.onComplete()` and `Subscriber.onError(Throwable t)` MUST consider the Subscription cancelled after having received the signal. |
@@ -144,7 +145,7 @@ public interface Subscriber<T> {
144
145
|<a name="2.7">7</a>|A `Subscriber` MUST ensure that all calls on its `Subscription` take place from the same thread or provide for respective [external synchronization](#term_ext_sync).|
145
146
| [:bulb:](#2.7"2.7 explained") |*The intent of this rule is to establish that [external synchronization](#term_ext_sync) must be added if a Subscriber will be using a Subscription concurrently by two or more threads.*|
146
147
|<a name="2.8">8</a>|A `Subscriber` MUST be prepared to receive one or more `onNext` signals after having called `Subscription.cancel()` if there are still requested elements pending [see [3.12](#3.12)]. `Subscription.cancel()` does not guarantee to perform the underlying cleaning operations immediately. |
147
-
| [:bulb:](#2.8"2.8 explained") |*The intent of this rule is to highlight that there may be a delay between calling `cancel` the Publisherseeing that.*|
148
+
| [:bulb:](#2.8"2.8 explained") |*The intent of this rule is to highlight that there may be a delay between calling `cancel` and the Publisherobserving that cancellation.*|
148
149
|<a name="2.9">9</a>|A `Subscriber` MUST be prepared to receive an `onComplete` signal with or without a preceding `Subscription.request(long n)` call. |
149
150
| [:bulb:](#2.9"2.9 explained") |*The intent of this rule is to establish that completion is unrelated to the demand flow—this allows for streams which complete early, and obviates the need to *poll*for completion.*|
150
151
|<a name="2.10">10</a>|A `Subscriber` MUST be prepared to receive an `onError` signal with or without a preceding `Subscription.request(long n)` call. |
@@ -172,10 +173,10 @@ public interface Subscription {
172
173
|<a name="3.2">2</a>|The `Subscription` MUST allow the `Subscriber` to call `Subscription.request` synchronously from within `onNext` or `onSubscribe`.|
173
174
| [:bulb:](#3.2"3.2 explained") |*The intent of this rule is to make it clear that implementations of `request` must be reentrant, to avoid stack overflows in the case of mutual recursion between `request` and `onNext` (and eventually `onComplete` / `onError`).This implies that Publishers can be `synchronous`, i.e. signalling `onNext`´s on the thread which calls `request`.*|
174
175
|<a name="3.3">3</a>| `Subscription.request` MUST place an upper bound on possible synchronous recursion between `Publisher` and `Subscriber`.|
175
-
| [:bulb:](#3.3"3.3 explained") |*The intent of this rule is to complement [see [3.2](#3.2)] by placing an upper limit on the mutual recursion between `request` and `onNext` (and eventually `onComplete` / `onError`).Implementations are RECOMMENDED to limit this mutual recursion to a depth of `1` (ONE)—for the sake of conserving stack space. An example for undesirable synchronous, open recursion would be Subscriber.onNext ->Subscription.request ->Subscriber.onNext -> …, as it otherwise will result in blowing the calling Thread´s stack.*|
176
+
| [:bulb:](#3.3"3.3 explained") |*The intent of this rule is to complement [see [3.2](#3.2)] by placing an upper limit on the mutual recursion between `request` and `onNext` (and eventually `onComplete` / `onError`).Implementations are RECOMMENDED to limit this mutual recursion to a depth of `1` (ONE)—for the sake of conserving stack space. An example for undesirable synchronous, open recursion would be Subscriber.onNext ->Subscription.request ->Subscriber.onNext -> …, as it otherwise will result in blowing the calling thread´s stack.*|
176
177
|<a name="3.4">4</a>| `Subscription.request` SHOULD respect the responsivity of its caller by returning in a timely manner. |
177
178
| [:bulb:](#3.4"3.4 explained") |*The intent of this rule is to establish that `request` is intended to be a [non-obstructing](#term_non-obstructing) method, and should be as quick to execute as possible on the calling thread, so avoid heavy computations and other things that would stall the caller´s thread of execution.*|
178
-
|<a name="3.5">5</a>| `Subscription.cancel` MUST respect the responsivity of its caller by returning in a timely manner, MUST be idempotent and MUST be thread-safe. |
179
+
|<a name="3.5">5</a>| `Subscription.cancel` MUST respect the responsivity of its caller by returning in a timely manner, MUST be idempotent and MUST be [thread-safe](#term_thread-safe).|
179
180
| [:bulb:](#3.5"3.5 explained") |*The intent of this rule is to establish that `cancel` is intended to be a [non-obstructing](#term_non-obstructing) method, and should be as quick to execute as possible on the calling thread, so avoid heavy computations and other things that would stall the caller´s thread of execution. Furthermore, it is also important that it is possible to call it multiple times without any adverse effects.*|
180
181
|<a name="3.6">6</a>|After the `Subscription` is cancelled, additional `Subscription.request(long n)` MUST be [NOPs](#term_nop).|
181
182
| [:bulb:](#3.6"3.6 explained") |*The intent of this rule is to establish a causal relationship between cancellation of a subscription and the subsequent non-operation of requesting more elements.*|
|<a name="4.2">2</a>|A `Processor` MAY choose to recover an `onError` signal. If it chooses to do so, it MUST consider the `Subscription` cancelled, otherwise it MUST propagate the `onError` signal to its Subscribers immediately. |
219
220
| [:bulb:](#4.2"4.2 explained") |*The intent of this rule is to inform that it’s possible for implementations to be more than simple transformations.*|
220
221
221
-
While not mandated, it can be a good idea to cancel a `Processors` upstream `Subscription` when/if its last `Subscriber` cancels their `Subscription`,
222
+
While not mandated, it can be a good idea to cancel a `Processor`´s upstream `Subscription` when/if its last `Subscriber` cancels their `Subscription`,
222
223
to let the cancellation signal propagate upstream.
0 commit comments