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

io.vertx.rxjava3.impl.ReadStreamSubscriber Maven / Gradle / Ivy

package io.vertx.rxjava3.impl;

import io.reactivex.rxjava3.core.BackpressureStrategy;
import io.reactivex.rxjava3.core.Flowable;
import io.reactivex.rxjava3.core.Observable;
import io.vertx.core.Handler;
import io.vertx.core.streams.ReadStream;
import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;

import java.util.ArrayDeque;
import java.util.function.Function;

/**
 * An RxJava {@code Subscriber} that turns an {@code Observable} into a {@link ReadStream}.
 * 

* The stream implements the {@link #pause()} and {@link #resume()} operation by maintaining * a buffer of {@link #BUFFER_SIZE} elements between the {@code Observable} and the {@code ReadStream}. *

* When the subscriber is created it requests {@code 0} elements to activate the subscriber's back-pressure. * Setting the handler initially on the {@code ReadStream} triggers a request of {@link #BUFFER_SIZE} elements. * When the item buffer is half empty, new elements are requested to fill the buffer back to {@link #BUFFER_SIZE} * elements. *

* The {@link #endHandler(Handler)} is called when the {@code Observable} is completed or has failed and * no pending elements, emitted before the completion or failure, are still in the buffer, i.e the handler * is not called when the stream is paused. * * @author Julien Viet */ public class ReadStreamSubscriber implements Subscriber, ReadStream { private static final Runnable NOOP_ACTION = () -> { }; private static final Throwable DONE_SENTINEL = new Throwable(); public static final int BUFFER_SIZE = 16; public static ReadStream asReadStream(Flowable flowable, Function adapter) { return new ReadStreamSubscriber<>(adapter, flowable::subscribe); } public static ReadStream asReadStream(Observable observable, Function adapter) { return asReadStream(observable.toFlowable(BackpressureStrategy.BUFFER), adapter); } private final Function adapter; private Handler endHandler; private Handler exceptionHandler; private Handler elementHandler; private long demand = Long.MAX_VALUE; private Throwable completed; private ArrayDeque pending = new ArrayDeque<>(); private int requested = 0; private Subscription subscription; private Publisher publisher; private boolean emitting; public ReadStreamSubscriber(Function adapter, Publisher publisher) { this.adapter = adapter; this.publisher = publisher; } @Override public ReadStream handler(Handler handler) { Runnable action; synchronized (this) { elementHandler = handler; if (handler != null) { action = () -> publisher.subscribe(this); } else { Subscription s = subscription; action = s != null ? s::cancel : NOOP_ACTION; } } action.run(); serializedCheckStatus(); return this; } @Override public ReadStream pause() { synchronized (this) { demand = 0L; } return this; } @Override public ReadStream fetch(long amount) { if (amount < 0L) { throw new IllegalArgumentException("Invalid amount: " + amount); } synchronized (this) { demand += amount; if (demand < 0L) { demand = Long.MAX_VALUE; } } serializedCheckStatus(); return this; } @Override public ReadStream resume() { return fetch(Long.MAX_VALUE); } @Override public void onSubscribe(Subscription s) { synchronized (this) { subscription = s; } serializedCheckStatus(); } /** Ensure checkStatus is never called concurrently. */ private void serializedCheckStatus() { synchronized (this) { if (emitting) { return; } emitting = true; } try { checkStatus(); } finally { synchronized (this) { emitting = false; } } } private void checkStatus() { Runnable action = NOOP_ACTION; while (true) { J adapted; Handler handler; synchronized (this) { if (demand > 0L && (handler = elementHandler) != null && pending.size() > 0) { if (demand != Long.MAX_VALUE) { demand--; } requested--; R item = pending.poll(); adapted = adapter.apply(item); } else { if (completed != null) { if (pending.isEmpty()) { Handler onError; Throwable result; if (completed != DONE_SENTINEL) { onError = exceptionHandler; result = completed; exceptionHandler = null; } else { onError = null; result = null; } Handler onCompleted = endHandler; endHandler = null; action = () -> { try { if (onError != null) { onError.handle(result); } } finally { if (onCompleted != null) { onCompleted.handle(null); } } }; } } else if (elementHandler != null && requested < BUFFER_SIZE / 2) { int request = BUFFER_SIZE - requested; action = () -> subscription.request(request); requested = BUFFER_SIZE; } break; } } handler.handle(adapted); } action.run(); } @Override public ReadStream endHandler(Handler handler) { synchronized (this) { if (completed == null || pending.size() > 0) { endHandler = handler; } else { if (handler != null) { throw new IllegalStateException(); } } } return this; } @Override public ReadStream exceptionHandler(Handler handler) { synchronized (this) { if (completed == null || pending.size() > 0) { exceptionHandler = handler; } else { if (handler != null) { throw new IllegalStateException(); } } } return this; } @Override public void onComplete() { onError(DONE_SENTINEL); } @Override public void onError(Throwable e) { synchronized (this) { if (completed != null) { return; } completed = e; } serializedCheckStatus(); } @Override public void onNext(R item) { synchronized (this) { pending.add(item); } serializedCheckStatus(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy