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

hu.akarnokd.rxjava3.joins.JoinObserver1 Maven / Gradle / Ivy

Go to download

RxJava 3.x extra sources, operators and components and ports of many 1.x companion libraries.

There is a newer version: 3.1.1
Show newest version
/*
 * Copyright 2016-2019 David Karnok
 *
 * 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 hu.akarnokd.rxjava3.joins;

import java.util.*;
import java.util.concurrent.atomic.*;

import io.reactivex.rxjava3.core.Notification;
import io.reactivex.rxjava3.core.Observable;
import io.reactivex.rxjava3.core.Observer;
import io.reactivex.rxjava3.disposables.Disposable;
import io.reactivex.rxjava3.exceptions.*;
import io.reactivex.rxjava3.functions.Consumer;
import io.reactivex.rxjava3.internal.disposables.DisposableHelper;
import io.reactivex.rxjava3.plugins.RxJavaPlugins;

/**
 * Default implementation of a join observer.
 * @param  the input element type
 */
final class JoinObserver1 implements Observer>, JoinObserver {
    private Object gate;
    private final Observable source;
    private final Consumer onError;
    private final List activePlans;
    private final Queue> queue;
    private final AtomicBoolean subscribed = new AtomicBoolean(false);
    private final InnerObserver safeObserver;

    JoinObserver1(Observable source, Consumer onError) {
        this.source = source;
        this.onError = onError;
        queue = new LinkedList>();
        activePlans = new ArrayList();
        safeObserver = new InnerObserver();
        // add this subscription so it gets unsubscribed when the parent does
//        add(safeObserver);
    }

    public Queue> queue() {
        return queue;
    }

    public void addActivePlan(ActivePlan0 activePlan) {
        activePlans.add(activePlan);
    }

    @Override
    public void subscribe(Object gate) {
        if (subscribed.compareAndSet(false, true)) {
            this.gate = gate;
            source.materialize().subscribe(this);
        } else {
            throw new IllegalStateException("Can only be subscribed to once.");
        }
    }

    @Override
    public void dequeue() {
        queue.remove();
    }

    @Override
    public void onNext(Notification args) {
        safeObserver.onNext(args);
    }

    @Override
    public void onError(Throwable e) {
        safeObserver.onError(e);
    }

    @Override
    public void onComplete() {
        safeObserver.onComplete();
    }

    void removeActivePlan(ActivePlan0 activePlan) {
        activePlans.remove(activePlan);
        if (activePlans.isEmpty()) {
            dispose();
        }
    }

    @Override
    public void onSubscribe(Disposable d) {
        safeObserver.onSubscribe(d);
    }

    @Override
    public boolean isDisposed() {
        return DisposableHelper.isDisposed(safeObserver.get());
    }

    @Override
    public void dispose() {
        DisposableHelper.dispose(safeObserver);
    }

    final class InnerObserver
    extends AtomicReference
    implements Observer> {

        private static final long serialVersionUID = -1466017793444404254L;

        @Override
        public void onSubscribe(Disposable d) {
            DisposableHelper.setOnce(this, d);
        }

        @Override
        public void onNext(Notification args) {
            synchronized (gate) {
                if (!DisposableHelper.isDisposed(get())) {
                    if (args.isOnError()) {
                        try {
                            onError.accept(args.getError());
                        } catch (Throwable ex) {
                            Exceptions.throwIfFatal(ex);
                            RxJavaPlugins.onError(new CompositeException(args.getError(), ex));
                        }
                        return;
                    }
                    queue.add(args);

                    // remark: activePlans might change while iterating
                    for (ActivePlan0 a : new ArrayList(activePlans)) {
                        try {
                            a.match();
                        } catch (Throwable ex) {
                            Exceptions.throwIfFatal(ex);
                            try {
                                onError.accept(ex);
                            } catch (Throwable ex2) {
                                Exceptions.throwIfFatal(ex2);
                                RxJavaPlugins.onError(new CompositeException(ex, ex2));
                                return;
                            }
                        }
                    }
                }
            }
        }

        @Override
        public void onError(Throwable e) {
            // not expected
        }

        @Override
        public void onComplete() {
            // not expected or ignored
        }
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy