All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ratpack.stream.Streams Maven / Gradle / Ivy

There is a newer version: 2.0.0-rc-1
Show newest version
/*
 * Copyright 2014 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package ratpack.stream;

import org.reactivestreams.Publisher;
import ratpack.exec.ExecController;
import ratpack.exec.Promise;
import ratpack.exec.internal.DefaultExecution;
import ratpack.func.Action;
import ratpack.func.Function;
import ratpack.func.Predicate;
import ratpack.registry.Registry;
import ratpack.stream.internal.*;
import ratpack.util.Types;

import java.time.Duration;
import java.util.List;
import java.util.concurrent.ScheduledExecutorService;

/**
 * Some lightweight utilities for working with reactive streams.
 * 
*

Reactive Streams is an initiative to provide a standard for asynchronous stream processing with non-blocking back pressure on the JVM.

*

http://www.reactive-streams.org

*
*

* Ratpack uses the Reactive Streams API when consuming streams of data (e.g {@link ratpack.http.Response#sendStream(Publisher)}). *

*

* This class provides some useful reactive utilities that integrate other parts of the Ratpack API with Reactive Stream types. * It is not designed to be a fully featured reactive toolkit. * If you require more features than provided here, consider using Ratpack's RxJava or Reactor integration. *

*

* The methods in this class are available as Groovy Extensions. * When using Groovy, applications can utilize the static methods provided in this class as instance-level methods against the first argument in their variable argument list. *

*/ public class Streams { /** * Wraps the publisher in Ratpack's {@link TransformablePublisher} to make composing a pipeline easier. *

* The return publisher is effectively the same publisher in terms of the {@link Publisher#subscribe(org.reactivestreams.Subscriber)} method. * * @param publisher the publisher to wrap * @param the type of item the publisher emits * @return a wrapped publisher */ public static TransformablePublisher transformable(Publisher publisher) { if (publisher instanceof TransformablePublisher) { return Types.cast(publisher); } else { return new DefaultTransformablePublisher<>(publisher); } } /** * Converts an iterable to a publishable. *

* Upon subscription, a new iterator will be created from the iterable. * Values are pulled from the iterator in accordance with the requests from the subscriber. *

* Any exception thrown by the iterable/iterator will be forwarded to the subscriber. * * @param iterable the data source * @param the type of item emitted * @return a publisher for the given iterable */ public static TransformablePublisher publish(Iterable iterable) { return new IterablePublisher<>(iterable); } /** * Creates a new publisher, backed by the given data producing function. *

* As subscribers request data of the returned stream, the given function is invoked. * The function returns the item to send downstream. * If the function returns {@code null}, the stream is terminated. * If the function throws an exception, the stream is terminated and the error is sent downstream. *

{@code
   * import ratpack.stream.Streams;
   * import ratpack.test.exec.ExecHarness;
   *
   * import java.util.Arrays;
   * import java.util.List;
   * import static org.junit.Assert.*;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     List strings = ExecHarness.yieldSingle(execControl ->
   *       Streams.yield(r -> {
   *         if (r.getRequestNum() < 2) {
   *           return Long.toString(r.getRequestNum());
   *         } else {
   *           return null;
   *         }
   *       }).toList()
   *     ).getValue();
   *
   *     assertEquals(Arrays.asList("0", "1"), strings);
   *   }
   * }
   * }
*

* If the value producing function is asynchronous, use {@link #flatYield(Function)}. * * @param producer the data source * @param the type of item emitted * @see #flatYield * @return a publisher backed by the given producer */ public static TransformablePublisher yield(Function producer) { return new YieldingPublisher<>(producer); } /** * Creates a new publisher, backed by the given asynchronous data producing function. *

* As subscribers request data of the returned stream, the given function is invoked. * The function returns a promise for the item to send downstream. * If the promise provides a value of {@code null}, the stream is terminated. * If the promise produces an error, the stream is terminated and the error is sent downstream. * If the promise producing function throws an exception, the stream is terminated and the error is sent downstream. *

{@code
   * import ratpack.stream.Streams;
   * import ratpack.exec.Promise;
   * import ratpack.test.exec.ExecHarness;
   *
   * import java.util.Arrays;
   * import java.util.List;
   * import static org.junit.Assert.*;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     List strings = ExecHarness.yieldSingle(execControl ->
   *       Streams.flatYield(r -> {
   *         if (r.getRequestNum() < 2) {
   *           return Promise.value(Long.toString(r.getRequestNum()));
   *         } else {
   *           return Promise.value(null);
   *         }
   *       }).toList()
   *     ).getValue();
   *
   *     assertEquals(Arrays.asList("0", "1"), strings);
   *   }
   * }
   * }
*

* If the value producing function is not asynchronous, use {@link #yield(Function)}. * * @param producer the data source * @param the type of item emitted * @see #yield * @return a publisher backed by the given producer */ public static TransformablePublisher flatYield(Function> producer) { return new FlatYieldingPublisher<>(producer); } /** * Creates a new publisher, that indefinitely streams the given object to all subscribers. *

* This is rarely useful for anything other than testing. * * @param item the item to indefinitely stream * @param the type of item emitted * @return a publisher that indefinitely streams the given item */ public static TransformablePublisher constant(final T item) { return yield(yieldRequest -> item); } /** * Returns a publisher that publishes items from the given input publisher after transforming each item via the given function. *

* The returned publisher does not perform any flow control on the data stream. *

* If the given transformation errors, the exception will be forwarded to the subscriber and the subscription to the input stream will be cancelled. * * @param input the stream of input data * @param function the transformation * @param the type of input item * @param the type of output item * @return a publisher that applies the given transformation to each item from the input stream */ public static TransformablePublisher map(Publisher input, Function function) { return new MapPublisher<>(input, function); } /** * Returns a publisher that filters items from the given input stream by applying the given filter predicate. *

* The returned stream is {@link #buffer buffered}, which means that if the downstream requests, say 5 items, which is filtered into only 3 items * the publisher will ask for more from the upstream to meet the downstream demand. *

{@code
   * import org.reactivestreams.Publisher;
   * import ratpack.stream.Streams;
   * import ratpack.stream.TransformablePublisher;
   * import ratpack.test.exec.ExecHarness;
   *
   * import java.util.Arrays;
   * import java.util.List;
   *
   * import static org.junit.Assert.*;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     List result = ExecHarness.yieldSingle(execControl -> {
   *       TransformablePublisher evens = Streams.publish(Arrays.asList(1, 2, 3, 4, 5, 6)).filter(i -> i % 2 == 0);
   *       return evens.toList();
   *     }).getValue();
   *
   *     assertEquals(Arrays.asList(2, 4, 6), result);
   *   }
   * }
   * }
* * @param input the stream to filter * @param filter the filter predicate * @param the type of item emitted * @return the input stream filtered */ public static TransformablePublisher filter(Publisher input, Predicate filter) { return streamMap(input, out -> new WriteStream() { @Override public void item(T item) { try { if (filter.apply(item)) { out.item(item); } } catch (Throwable throwable) { out.error(throwable); } } @Override public void error(Throwable throwable) { out.error(throwable); } @Override public void complete() { out.complete(); } }); } /** * Allows transforming a stream into an entirely different stream. *

* While the {@link #map(Publisher, Function)} method support transforming individual items, this method supports transforming the stream as a whole. * This is necessary when the transformation causes a different number of items to be emitted than the original stream. *

{@code
   * import org.reactivestreams.Publisher;
   * import ratpack.stream.Streams;
   * import ratpack.stream.TransformablePublisher;
   * import ratpack.stream.WriteStream;
   * import ratpack.test.exec.ExecHarness;
   *
   * import java.util.Arrays;
   * import java.util.List;
   *
   * import static org.junit.Assert.*;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     List result = ExecHarness.yieldSingle(execControl -> {
   *       Publisher chars = Streams.publish(Arrays.asList("a", "b", "c"));
   *       TransformablePublisher mapped = Streams.streamMap(chars, out ->
   *         new WriteStream() {
   *           public void item(String item) {
   *             out.item(item);
   *             out.item(item.toUpperCase());
   *           }
   *
   *           public void error(Throwable error) {
   *             out.error(error);
   *           }
   *
   *           public void complete() {
   *             out.complete();
   *           }
   *         }
   *       );
   *       return mapped.toList();
   *     }).getValue();
   *
   *     assertEquals(Arrays.asList("a", "A", "b", "B", "c", "C"), result);
   *   }
   * }
   * }
*

* The {@code mapper} function receives a {@link WriteStream} for emitting items and returns a {@link WriteStream} that will be written to by the upstream publisher. * Calling {@link WriteStream#complete()} or {@link WriteStream#error(Throwable)} on the received write stream will {@link org.reactivestreams.Subscription#cancel() cancel} the upstream subscription and it is guaranteed that no more items will be emitted from the upstream. * If the complete/error signals from upstream don't need to be intercepted, the {@link WriteStream#itemMap(Action)} can be used on the write stream given to the mapping function to create the return write stream. *

* The returned stream is {@link #buffer buffered}, which allows the stream transformation to emit more items downstream than what was received from the upstream. * Currently, the upstream subscription will always receive a single {@link org.reactivestreams.Subscription#request(long) request} for {@link Long#MAX_VALUE}, effectively asking upstream to emit as fast as it can. * Future versions may propagate backpressure more intelligently. * * @param input the stream to map * @param mapper the mapping function * @param the type of item received * @param the type of item produced * @return the input stream transformed */ public static TransformablePublisher streamMap(Publisher input, Function, ? extends WriteStream> mapper) { /* Implementation note: we need a smarter buffering strategy here. We'll need to translate demand in a smart way as the arity of the outgoing stream may be different. If the downstream requests, say 5 items, which the mapper then receives and translates into only 3 items we'll need to ask for more from the upstream to meet the downstream demand. We get around this now by always opening up the firehose to upstream, but this means we are nullifying back pressure and buffering in memory. https://github.com/ratpack/ratpack/issues/515 */ return new StreamMapPublisher<>(input, mapper).buffer(); } /** * Returns a publisher that publishes items from the given input publisher after transforming each item via the given, promise returning, function. *

* The returned publisher does not perform any flow control on the data stream. *

* If the given transformation errors, or if the returned promise fails, the exception will be forwarded to the subscriber and the subscription to the input stream will be cancelled. * * @param input the stream of input data * @param function the transformation * @param the type of input item * @param the type of output item * @return a publisher that applies the given transformation to each item from the input stream */ public static TransformablePublisher flatMap(Publisher input, Function> function) { return new FlatMapPublisher<>(input, function); } /** * Returns a publisher that allows the given publisher to emit as fast as it can, while applying flow control downstream. *

* When the return publisher is subscribed to, a subscription will be made to the given publisher with a request for {@link Long#MAX_VALUE} items. * This effectively allows the given publisher to emit each item as soon as it can. * The return publisher will manage the back pressure by holding excess items from the given publisher in memory until the downstream subscriber is ready for them. *

* This is a simple, naive, flow control mechanism. * If the given producer emits far faster than the downstream subscriber requests, the intermediate queue will grow large and consume substantial memory. * However, it is useful or adapting non-infinite publishers that cannot meaningfully respect back pressure. * * @param publisher a data source * @param the type of item * @return a publisher that applies respects back pressure, effectively throttling the given publisher */ public static TransformablePublisher buffer(Publisher publisher) { return new BufferingPublisher<>(publisher); } /** * Allows requests from the subscriber of the return publisher to be withheld from the given publisher until an externally defined moment. *

* When the return publisher is subscribed to, the given publisher will be subscribed to. * All requests made by the subscriber of the return publisher will not be forwarded to the subscription of the given publisher until the runnable given to the given action is run. * Once the runnable is run, all requests are directly forwarded to the subscription of the given publisher. *

* The return publisher supports multi subscription, creating a new subscription to the given publisher each time. * The given action will be invoke each time the return publisher is subscribed to with a distinct runnable for releasing the gate for that subscription. *

* The given action will be invoked immediately upon subscription of the return publisher. * The runnable given to this action may be invoked any time (i.e. it does not need to be run during the action). * If the action errors, the given publisher will not be subscribed to and the error will be sent to the return publisher subscriber. * * @param publisher the data source * @param valveReceiver an action that receives a runnable “valve” that when run allows request to start flowing upstream * @param the type of item emitted * @return a publisher that is logically equivalent to the given publisher as far as subscribers are concerned */ public static TransformablePublisher gate(Publisher publisher, Action valveReceiver) { return new GatedPublisher<>(publisher, valveReceiver); } /** * Executes the given function periodically, publishing the return value to the subscriber. *

* When the return publisher is subscribed to, the given function is executed immediately (via the executor) with {@code 0} as the input. * The function will then be repeatedly executed again after the given delay (with an incrementing input) until the function returns {@code null}. * That is, a return value from the function of {@code null} signals that the data stream has finished. * The function will not be executed again after returning {@code null}. *

* Each new subscription to the publisher will cause the function to be scheduled again. * Due to this, it is generally desirable to wrap the return publisher in a multicasting publisher. *

* If the function throws an exception, the error will be sent to the subscribers and no more invocations of the function will occur. *

* The returned publisher is implicitly buffered to respect back pressure via {@link #buffer(Publisher)}. * * @param executorService the executor service that will periodically execute the function * @param duration the duration of the delay (Duration.ofSeconds(1) - delay will be 1 second) * @param producer a function that produces values to emit * @param the type of item * @return a publisher that applies respects back pressure, effectively throttling the given publisher */ public static TransformablePublisher periodically(ScheduledExecutorService executorService, Duration duration, Function producer) { return new PeriodicPublisher<>(executorService, producer, duration).buffer(); } public static TransformablePublisher periodically(Registry registry, Duration duration, Function producer) { return new PeriodicPublisher<>(registry.get(ExecController.class).getExecutor(), producer, duration).buffer(); } /** * Allows listening to the events of the given publisher as they flow to subscribers. *

* When the return publisher is subscribed to, the given publisher will be subscribed to. * All events (incl. data, error and completion) emitted by the given publisher will be forwarded to the given listener before being forward to the subscriber of the return publisher. *

* If the listener errors, the upstream subscription will be cancelled (if appropriate) and the error sent downstream. * If the listener errors while listening to an error event, the listener error will be {@link Throwable#addSuppressed(Throwable) added as a surpressed exception} * to the original exception which will then be sent downstream. * * @param publisher the data source * @param listener the listener for emitted items * @param the type of item emitted * @return a publisher that is logically equivalent to the given publisher as far as subscribers are concerned */ public static TransformablePublisher wiretap(Publisher publisher, Action> listener) { return new WiretapPublisher<>(publisher, listener); } /** * Returns a publisher that will stream events emitted from the given publisher to all of its subscribers. *

* The return publisher allows the given publisher to emit as fast as it can, while applying flow control downstream to multiple subscribers. * Each subscriber can signal its own demand. If the given publisher emits far faster than the downstream subscribers request, the intermediate * queue of each subscriber will grow large and consume substantial memory. However, given this publisher is likely to be used with a periodic * publisher or a regular indefinite stream it is unlikely to be a problem. *

* When a subscriber subscribes to the return publisher then it will not receive any events that have been emitted before it subscribed. * * @param publisher a data source * @param the type of item * @return a publisher that respects back pressure for each of it's Subscribers. */ public static TransformablePublisher multicast(Publisher publisher) { return new MulticastPublisher<>(publisher); } /** * Returns a publisher that publishes each element from Collections that are produced from the given input publisher. *

* For each item the return publisher receives from the given input publisher, the return publisher will iterate over its elements and publish a * new item for each element to its downstream subscriber e.g. if the return publisher receives a Collection with 10 elements then the downstream * subscriber will receive 10 calls to its onNext method. *

* The returned publisher is implicitly buffered to respect back pressure via {@link #buffer(Publisher)}. * * @param publisher the data source * @param the type of item emitted * @return a publisher that splits collection items into new items per collection element */ public static TransformablePublisher fanOut(Publisher> publisher) { return new FanOutPublisher<>(publisher).buffer(); } /** * Returns a publisher that merges the given input publishers into a single stream of elements. *

* The returned publisher obeys the following rules: *

    *
  • * Only when all given input publishers have signalled completion will the downstream subscriber be completed. *
  • *
  • * If one of the given input publishers errors then all other publisher subscriptions are cancelled and the error is propagated to the subscriber. *
  • *
  • * If the subscription of the returned publisher is cancelled by the subscriber then all given input publisher subscriptions are cancelled. *
  • *
*

* The returned publisher is implicitly buffered to respect back pressure via {@link #buffer(org.reactivestreams.Publisher)}. * * @param publishers the data sources to merge * @param the type of item emitted * @return a publisher that emits a single stream of elements from multiple publishers */ @SuppressWarnings({"unchecked", "varargs"}) @SafeVarargs public static TransformablePublisher merge(Publisher... publishers) { return new MergingPublisher<>(publishers).buffer(); } /** * Creates a promise for the given publisher's single item. *

* The given publisher is expected to produce zero or one items. * If it produces zero, the promised value will be {@code null}. * The it produces exactly one item, the promised value will be that item. *

* If the stream produces more than one item, the promise will fail with an {@link IllegalStateException}. * As soon as a second item is received, the subscription to the given publisher will be cancelled. *

* The single item is not provided to the promise subscriber until the stream completes, to ensure that it is indeed a one element stream. * If the stream errors before sending a second item, the promise will fail with that error. * If it fails after sending a second item, that error will be ignored. * * @param publisher the publiser the convert to a promise * @param the type of promised value * @return a promise for the publisher's single item */ public static Promise toPromise(Publisher publisher) { return Promise.of(f -> publisher.subscribe(SingleElementSubscriber.to(f::accept))); } /** * Creates a promise for the given publisher's items as a List. * * @param publisher the stream to collect to a list * @param the type of item in the stream * @return a promise for the streams contents as a list */ public static Promise> toList(Publisher publisher) { return Promise.of(f -> publisher.subscribe(new CollectingSubscriber<>(f::accept, s -> s.request(Long.MAX_VALUE)))); } public static TransformablePublisher bindExec(Publisher publisher) { return DefaultExecution.stream(publisher); } }