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

io.reactivex.observable.internal.observers.BasicFuseableObserver 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.observable.internal.observers;

import io.reactivex.common.*;
import io.reactivex.common.exceptions.Exceptions;
import io.reactivex.common.internal.disposables.DisposableHelper;
import io.reactivex.observable.Observer;
import io.reactivex.observable.extensions.QueueDisposable;

/**
 * Base class for a fuseable intermediate observer.
 * @param  the upstream value type
 * @param  the downstream value type
 */
public abstract class BasicFuseableObserver implements Observer, QueueDisposable {

    /** The downstream subscriber. */
    protected final Observer actual;

    /** The upstream subscription. */
    protected Disposable s;

    /** The upstream's QueueDisposable if not null. */
    protected QueueDisposable qs;

    /** Flag indicating no further onXXX event should be accepted. */
    protected boolean done;

    /** Holds the established fusion mode of the upstream. */
    protected int sourceMode;

    /**
     * Construct a BasicFuseableObserver by wrapping the given subscriber.
     * @param actual the subscriber, not null (not verified)
     */
    public BasicFuseableObserver(Observer actual) {
        this.actual = actual;
    }

    // final: fixed protocol steps to support fuseable and non-fuseable upstream
    @SuppressWarnings("unchecked")
    @Override
    public final void onSubscribe(Disposable s) {
        if (DisposableHelper.validate(this.s, s)) {

            this.s = s;
            if (s instanceof QueueDisposable) {
                this.qs = (QueueDisposable)s;
            }

            if (beforeDownstream()) {

                actual.onSubscribe(this);

                afterDownstream();
            }

        }
    }

    /**
     * Override this to perform actions before the call {@code actual.onSubscribe(this)} happens.
     * @return true if onSubscribe should continue with the call
     */
    protected boolean beforeDownstream() {
        return true;
    }

    /**
     * Override this to perform actions after the call to {@code actual.onSubscribe(this)} happened.
     */
    protected void afterDownstream() {
        // default no-op
    }

    // -----------------------------------
    // Convenience and state-aware methods
    // -----------------------------------

    @Override
    public void onError(Throwable t) {
        if (done) {
            RxJavaCommonPlugins.onError(t);
            return;
        }
        done = true;
        actual.onError(t);
    }

    /**
     * Rethrows the throwable if it is a fatal exception or calls {@link #onError(Throwable)}.
     * @param t the throwable to rethrow or signal to the actual subscriber
     */
    protected final void fail(Throwable t) {
        Exceptions.throwIfFatal(t);
        s.dispose();
        onError(t);
    }

    @Override
    public void onComplete() {
        if (done) {
            return;
        }
        done = true;
        actual.onComplete();
    }

    /**
     * Calls the upstream's QueueDisposable.requestFusion with the mode and
     * saves the established mode in {@link #sourceMode} if that mode doesn't
     * have the {@link QueueDisposable#BOUNDARY} flag set.
     * 

* If the upstream doesn't support fusion ({@link #qs} is null), the method * returns {@link QueueDisposable#NONE}. * @param mode the fusion mode requested * @return the established fusion mode */ protected final int transitiveBoundaryFusion(int mode) { QueueDisposable qs = this.qs; if (qs != null) { if ((mode & BOUNDARY) == 0) { int m = qs.requestFusion(mode); if (m != NONE) { sourceMode = m; } return m; } } return NONE; } // -------------------------------------------------------------- // Default implementation of the RS and QS protocol (can be overridden) // -------------------------------------------------------------- @Override public void dispose() { s.dispose(); } @Override public boolean isDisposed() { return s.isDisposed(); } @Override public boolean isEmpty() { return qs.isEmpty(); } @Override public void clear() { qs.clear(); } // ----------------------------------------------------------- // The rest of the Queue interface methods shouldn't be called // ----------------------------------------------------------- @Override public final boolean offer(R e) { throw new UnsupportedOperationException("Should not be called!"); } @Override public final boolean offer(R v1, R v2) { throw new UnsupportedOperationException("Should not be called!"); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy