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

io.reactivex.rxjava3.internal.operators.flowable.AbstractBackpressureThrottlingSubscriber Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2016-present, RxJava Contributors.
 *
 * 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 io.reactivex.rxjava3.internal.operators.flowable;

import io.reactivex.rxjava3.core.FlowableSubscriber;
import io.reactivex.rxjava3.internal.subscriptions.SubscriptionHelper;
import io.reactivex.rxjava3.internal.util.BackpressureHelper;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;

import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;

/**
 * Abstract base class for operators that throttle excessive updates from upstream in case if
 * downstream {@link Subscriber} is not ready to receive updates.
 *
 * @param  the upstream value type
 * @param  the downstream value type
 */
abstract class AbstractBackpressureThrottlingSubscriber extends AtomicInteger implements FlowableSubscriber, Subscription {

    private static final long serialVersionUID = -5050301752721603566L;

    final Subscriber downstream;

    Subscription upstream;

    volatile boolean done;
    Throwable error;

    volatile boolean cancelled;

    final AtomicLong requested = new AtomicLong();

    final AtomicReference current = new AtomicReference<>();

    AbstractBackpressureThrottlingSubscriber(Subscriber downstream) {
        this.downstream = downstream;
    }

    @Override
    public void onSubscribe(Subscription s) {
        if (SubscriptionHelper.validate(this.upstream, s)) {
            this.upstream = s;
            downstream.onSubscribe(this);
            s.request(Long.MAX_VALUE);
        }
    }

    @Override
    public abstract void onNext(T t);

    @Override
    public void onError(Throwable t) {
        error = t;
        done = true;
        drain();
    }

    @Override
    public void onComplete() {
        done = true;
        drain();
    }

    @Override
    public void request(long n) {
        if (SubscriptionHelper.validate(n)) {
            BackpressureHelper.add(requested, n);
            drain();
        }
    }

    @Override
    public void cancel() {
        if (!cancelled) {
            cancelled = true;
            upstream.cancel();

            if (getAndIncrement() == 0) {
                current.lazySet(null);
            }
        }
    }

    void drain() {
        if (getAndIncrement() != 0) {
            return;
        }
        final Subscriber a = downstream;
        int missed = 1;
        final AtomicLong r = requested;
        final AtomicReference q = current;

        for (;;) {
            long e = 0L;

            while (e != r.get()) {
                boolean d = done;
                R v = q.getAndSet(null);
                boolean empty = v == null;

                if (checkTerminated(d, empty, a, q)) {
                    return;
                }

                if (empty) {
                    break;
                }

                a.onNext(v);

                e++;
            }

            if (e == r.get() && checkTerminated(done, q.get() == null, a, q)) {
                return;
            }

            if (e != 0L) {
                BackpressureHelper.produced(r, e);
            }

            missed = addAndGet(-missed);
            if (missed == 0) {
                break;
            }
        }
    }

    boolean checkTerminated(boolean d, boolean empty, Subscriber a, AtomicReference q) {
        if (cancelled) {
            q.lazySet(null);
            return true;
        }

        if (d) {
            Throwable e = error;
            if (e != null) {
                q.lazySet(null);
                a.onError(e);
                return true;
            } else
            if (empty) {
                a.onComplete();
                return true;
            }
        }

        return false;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy