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

rx.subjects.SubjectSubscriptionManager Maven / Gradle / Ivy

There is a newer version: 0.20.7
Show newest version
/**
 * Copyright 2014 Netflix, Inc.
 * 
 * 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 rx.subjects;

import java.util.Arrays;
import java.util.Collection;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicReference;

import rx.Observable.OnSubscribe;
import rx.Observer;
import rx.Subscriber;
import rx.Subscription;
import rx.functions.Action0;
import rx.functions.Action1;
import rx.operators.SafeObservableSubscription;
import rx.subscriptions.Subscriptions;

/* package */class SubjectSubscriptionManager {

    private AtomicReference> state = new AtomicReference>(new State());

    /**
     * 
     * @param onSubscribe
     *            Always runs at the beginning of 'subscribe' regardless of terminal state.
     * @param onTerminated
     *            Only runs if Subject is in terminal state and the Observer ends up not being registered.
     * @param onUnsubscribe called after the child subscription is removed from the state
     * @return
     */
    public OnSubscribe getOnSubscribeFunc(final Action1> onSubscribe, 
            final Action1> onTerminated,
            final Action1> onUnsubscribe) {
        return new OnSubscribe() {
            @Override
            public void call(Subscriber actualOperator) {
                final SubjectObserver observer = new SubjectObserver(actualOperator);
                // invoke onSubscribe logic 
                if (onSubscribe != null) {
                    onSubscribe.call(observer);
                }

                State current;
                State newState = null;
                boolean addedObserver = false;
                do {
                    current = state.get();
                    if (current.terminated) {
                        // we are terminated so don't need to do anything
                        addedObserver = false;
                        // break out and don't try to modify state
                        newState = current;
                        // wait for termination to complete if 
                        try {
                            current.terminationLatch.await();
                        } catch (InterruptedException e) {
                            Thread.currentThread().interrupt();
                            throw new RuntimeException("Interrupted waiting for termination.", e);
                        }
                        break;
                    } else {
                        final SafeObservableSubscription subscription = new SafeObservableSubscription();
                        actualOperator.add(subscription); // add to parent if the Subject itself is unsubscribed
                        addedObserver = true;
                        subscription.wrap(Subscriptions.create(new Action0() {

                            @Override
                            public void call() {
                                State current;
                                State newState;
                                do {
                                    current = state.get();
                                    // on unsubscribe remove it from the map of outbound observers to notify
                                    newState = current.removeObserver(subscription);
                                } while (!state.compareAndSet(current, newState));
                                if (onUnsubscribe != null) {
                                    onUnsubscribe.call(observer);
                                }
                            }
                        }));

                        // on subscribe add it to the map of outbound observers to notify
                        newState = current.addObserver(subscription, observer);
                    }
                } while (!state.compareAndSet(current, newState));

                /**
                 * Whatever happened above, if we are terminated we run `onTerminated`
                 */
                if (newState.terminated && !addedObserver) {
                    onTerminated.call(observer);
                }
            }

        };
    }

    @SuppressWarnings({ "unchecked", "rawtypes" })
    protected void terminate(Action1>> onTerminate) {
        State current;
        State newState = null;
        do {
            current = state.get();
            if (current.terminated) {
                // already terminated so do nothing
                return;
            } else {
                newState = current.terminate();
            }
        } while (!state.compareAndSet(current, newState));

        /*
         * if we get here then we won setting the state to terminated
         * and have a deterministic set of Observers to emit to (concurrent subscribes
         * will have failed and will try again and see we are term
         * inated)
         */
        try {
            // had to circumvent type check, we know what the array contains
            onTerminate.call((Collection) Arrays.asList(newState.observers));
        } finally {
            // mark that termination is completed
            newState.terminationLatch.countDown();
        }
    }

    /**
     * Returns the array of observers directly.
     * Don't modify the array!
     * 
     * @return the array of current observers
     */
    @SuppressWarnings("unchecked")
    public SubjectObserver[] rawSnapshot() {
        return state.get().observers;
    }

    @SuppressWarnings("rawtypes")
    protected static class State {
        final boolean terminated;
        final CountDownLatch terminationLatch;
        final Subscription[] subscriptions;
        final SubjectObserver[] observers;
        // to avoid lots of empty arrays
        final Subscription[] EMPTY_S = new Subscription[0];
        // to avoid lots of empty arrays
        final SubjectObserver[] EMPTY_O = new SubjectObserver[0];

        private State(boolean isTerminated, CountDownLatch terminationLatch,
                Subscription[] subscriptions, SubjectObserver[] observers) {
            this.terminationLatch = terminationLatch;
            this.terminated = isTerminated;
            this.subscriptions = subscriptions;
            this.observers = observers;
        }

        State() {
            this.terminated = false;
            this.terminationLatch = null;
            this.subscriptions = EMPTY_S;
            this.observers = EMPTY_O;
        }

        public State terminate() {
            if (terminated) {
                throw new IllegalStateException("Already terminated.");
            }
            return new State(true, new CountDownLatch(1), subscriptions, observers);
        }

        public State addObserver(Subscription s, SubjectObserver observer) {
            int n = this.observers.length;

            Subscription[] newsubscriptions = Arrays.copyOf(this.subscriptions, n + 1);
            SubjectObserver[] newobservers = Arrays.copyOf(this.observers, n + 1);

            newsubscriptions[n] = s;
            newobservers[n] = observer;

            return createNewWith(newsubscriptions, newobservers);
        }

        private State createNewWith(Subscription[] newsubscriptions, SubjectObserver[] newobservers) {
            return new State(terminated, terminationLatch, newsubscriptions, newobservers);
        }

        public State removeObserver(Subscription s) {
            // we are empty, nothing to remove
            if (this.observers.length == 0) {
                return this;
            }
            int n = Math.max(this.observers.length - 1, 1);
            int copied = 0;
            Subscription[] newsubscriptions = Arrays.copyOf(this.subscriptions, n);
            SubjectObserver[] newobservers = Arrays.copyOf(this.observers, n);

            for (int i = 0; i < this.subscriptions.length; i++) {
                Subscription s0 = this.subscriptions[i];
                if (s0 != s) {
                    if (copied == n) {
                        // if s was not found till the end of the iteration
                        // we return ourselves since no modification should
                        // have happened
                        return this;
                    }
                    newsubscriptions[copied] = s0;
                    newobservers[copied] = this.observers[i];
                    copied++;
                }
            }

            if (copied == 0) {
                return createNewWith(EMPTY_S, EMPTY_O);
            }
            // if somehow copied less than expected, truncate the arrays
            // if s is unique, this should never happen
            if (copied < n) {
                return createNewWith(Arrays.copyOf(newsubscriptions, copied), Arrays.copyOf(newobservers, copied));
            }
            return createNewWith(newsubscriptions, newobservers);
        }
    }

    protected static class SubjectObserver implements Observer {

        private final Observer actual;
        protected volatile boolean caughtUp = false;

        SubjectObserver(Observer actual) {
            this.actual = actual;
        }

        @Override
        public void onCompleted() {
            this.actual.onCompleted();
        }

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

        @Override
        public void onNext(T v) {
            this.actual.onNext(v);
        }

    }

}