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

hu.akarnokd.rxjava2.observables.BlockingObservable Maven / Gradle / Ivy

There is a newer version: 2.0.0-RC3
Show newest version
/**
 * Copyright 2015 David Karnok and 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 hu.akarnokd.rxjava2.observables;

import java.io.Closeable;
import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicReference;

import org.reactivestreams.*;

import hu.akarnokd.rxjava2.Observable;
import hu.akarnokd.rxjava2.Observer;
import hu.akarnokd.rxjava2.Optional;
import hu.akarnokd.rxjava2.disposables.*;
import hu.akarnokd.rxjava2.functions.Consumer;
import hu.akarnokd.rxjava2.internal.functions.Functions;
import hu.akarnokd.rxjava2.internal.operators.*;
import hu.akarnokd.rxjava2.internal.subscribers.*;
import hu.akarnokd.rxjava2.internal.util.*;
import hu.akarnokd.rxjava2.plugins.RxJavaPlugins;

public final class BlockingObservable implements Publisher, Iterable {
    final Publisher o;
    protected BlockingObservable(Publisher source) {
        this.o = source;
    }
    
    @SuppressWarnings("unchecked")
    public static  BlockingObservable from(Publisher source) {
        if (source instanceof BlockingObservable) {
            return (BlockingObservable)source;
        }
        return new BlockingObservable(source);
    }
    
    @Override
    public Iterator iterator() {
        return iterate(o);
    }
    
    public void forEach(Consumer action) {
        BlockingIterator it = iterate(o);
        while (it.hasNext()) {
            try {
                action.accept(it.next());
            } catch (Throwable e) {
                it.dispose();
                throw Exceptions.propagate(e);
            }
        }
    }
    
    static final  BlockingIterator iterate(Publisher p) {
        final BlockingQueue queue = new LinkedBlockingQueue();

        LambdaSubscriber ls = new LambdaSubscriber(
            new Consumer() {
                @Override
                public void accept(T v) {
                    queue.offer(NotificationLite.next(v));
                }
            },
            new Consumer() {
                @Override
                public void accept(Throwable e) {
                    queue.offer(NotificationLite.error(e));
                }
            },
            new Runnable() {
                @Override
                public void run() {
                    queue.offer(NotificationLite.complete());
                }
            },
            new Consumer() {
                @Override
                public void accept(Subscription s) {
                    s.request(Long.MAX_VALUE);
                }
            }
        );
        
        p.subscribe(ls);
        
        return new BlockingIterator(queue, ls);
    }
    
    static final class BlockingIterator implements Iterator, Closeable, Disposable {
        final BlockingQueue queue;
        final Disposable resource;
        
        Object last;
        
        public BlockingIterator(BlockingQueue queue, Disposable resource) {
            this.queue = queue;
            this.resource = resource;
        }
        @Override
        public boolean hasNext() {
            if (last == null) { 
                Object o = queue.poll();
                if (o == null) {
                    try {
                        o = queue.take();
                    } catch (InterruptedException ex) {
                        resource.dispose();
                        Thread.currentThread().interrupt();
                        Exceptions.propagate(ex);
                    }
                }
                last = o;
                if (NotificationLite.isError(o)) {
                    resource.dispose();
                    Throwable e = NotificationLite.getError(o);
                    Exceptions.propagate(e);
                }
                if (NotificationLite.isComplete(o)) {
                    resource.dispose();
                    return false;
                }
                return true;
            }
            Object o = last;
            if (NotificationLite.isError(o)) {
                Throwable e = NotificationLite.getError(o);
                Exceptions.propagate(e);
            }
            return !NotificationLite.isComplete(o);
        }
        
        @Override
        public T next() {
            if (hasNext()) {
                Object o = last;
                last = null;
                return NotificationLite.getValue(o);
            }
            throw new NoSuchElementException();
        }
        
        @Override
        public void remove() {
            throw new UnsupportedOperationException();
        }
        
        @Override
        public void close() {
            resource.dispose();
        }
        
        @Override
        public void dispose() {
            resource.dispose();
        }
    }

    public Optional firstOption() {
        return firstOption(o);
    }
    
    static  Optional firstOption(Publisher o) {
        final AtomicReference value = new AtomicReference();
        final AtomicReference error = new AtomicReference();
        final CountDownLatch cdl = new CountDownLatch(1);
        final MultipleAssignmentDisposable mad = new MultipleAssignmentDisposable();
        
        o.subscribe(new Subscriber() {
            Subscription s;
            @Override
            public void onSubscribe(Subscription s) {
                this.s = s;
                mad.set(Disposables.from(s));
                s.request(Long.MAX_VALUE);
            }
            
            @Override
            public void onNext(T t) {
                s.cancel();
                value.lazySet(t);
                cdl.countDown();
            }
            
            @Override
            public void onError(Throwable t) {
                error.lazySet(t);
                cdl.countDown();
            }
            
            @Override
            public void onComplete() {
                cdl.countDown();
            }
        });
        
        try {
            cdl.await();
        } catch (InterruptedException ex) {
            mad.dispose();
            Exceptions.propagate(ex);
        }
        
        Throwable e = error.get();
        if (e != null) {
            Exceptions.propagate(e);
        }
        T v = value.get();
        return v != null ? Optional.of(v) : Optional.empty();
    }
    
    public T first() {
        Optional o = firstOption();
        if (o.isPresent()) {
            return o.get();
        }
        throw new NoSuchElementException();
    }
    
    public T first(T defaultValue) {
        Optional o = firstOption();
        if (o.isPresent()) {
            return o.get();
        }
        return defaultValue;
    }
    
    public Optional lastOption() {
        return lastOption(o);
    }
    
    static  Optional lastOption(Publisher o) {
        final AtomicReference value = new AtomicReference();
        final AtomicReference error = new AtomicReference();
        final CountDownLatch cdl = new CountDownLatch(1);
        final MultipleAssignmentDisposable mad = new MultipleAssignmentDisposable();
        
        o.subscribe(new Subscriber() {
            @Override
            public void onSubscribe(Subscription s) {
                mad.set(Disposables.from(s));
                s.request(Long.MAX_VALUE);
            }
            
            @Override
            public void onNext(T t) {
                value.lazySet(t);
            }
            
            @Override
            public void onError(Throwable t) {
                error.lazySet(t);
                cdl.countDown();
            }
            
            @Override
            public void onComplete() {
                cdl.countDown();
            }
        });
        
        try {
            cdl.await();
        } catch (InterruptedException ex) {
            mad.dispose();
            Exceptions.propagate(ex);
        }
        
        Throwable e = error.get();
        if (e != null) {
            Exceptions.propagate(e);
        }
        T v = value.get();
        return v != null ? Optional.of(v) : Optional.empty();
    }
    
    public T last() {
        Optional o = lastOption();
        if (o.isPresent()) {
            return o.get();
        }
        throw new NoSuchElementException();
    }
    
    public T last(T defaultValue) {
        Optional o = lastOption();
        if (o.isPresent()) {
            return o.get();
        }
        return defaultValue;
    }
    
    public T single() {
        Optional o = firstOption(Observable.fromPublisher(this.o).single());
        if (o.isPresent()) {
            return o.get();
        }
        throw new NoSuchElementException();
    }
    
    public T single(T defaultValue) {
        Optional o = firstOption(Observable.fromPublisher(this.o).single(defaultValue));
        if (o.isPresent()) {
            return o.get();
        }
        return defaultValue;
    }
    
    public Iterable mostRecent(T initialValue) {
        return BlockingOperatorMostRecent.mostRecent(o, initialValue);
    }
    
    public Iterable next() {
        return BlockingOperatorNext.next(o);
    }
    
    public Iterable latest() {
        return BlockingOperatorLatest.latest(o);
    }
    
    public Future toFuture() {
        final CountDownLatch cdl = new CountDownLatch(1);
        final AtomicReference value = new AtomicReference();
        final AtomicReference error = new AtomicReference();
        final MultipleAssignmentDisposable mad = new MultipleAssignmentDisposable();
        
        o.subscribe(new Subscriber() {

            @Override
            public void onSubscribe(Subscription d) {
                mad.set(Disposables.from(d));
                d.request(Long.MAX_VALUE);
            }

            @Override
            public void onNext(T v) {
                value.lazySet(v);
            }

            @Override
            public void onError(Throwable e) {
                error.lazySet(e);
                cdl.countDown();
            }

            @Override
            public void onComplete() {
                cdl.countDown();
            }
            
        });
        
        return new Future() {
            
            @Override
            public boolean cancel(boolean mayInterruptIfRunning) {
                if (cdl.getCount() != 0) {
                    mad.dispose();
                    return true;
                }
                return false;
            }

            @Override
            public boolean isCancelled() {
                return mad.isDisposed();
            }

            @Override
            public boolean isDone() {
                return cdl.getCount() == 0 && !mad.isDisposed();
            }

            @Override
            public T get() throws InterruptedException, ExecutionException {
                if (cdl.getCount() != 0) {
                    cdl.await();
                }
                Throwable e = error.get();
                if (e != null) {
                    throw new ExecutionException(e);
                }
                return value.get();
            }

            @Override
            public T get(long timeout, TimeUnit unit)
                    throws InterruptedException, ExecutionException, TimeoutException {
                if (cdl.getCount() != 0) {
                    if (!cdl.await(timeout, unit)) {
                        throw new TimeoutException();
                    }
                }
                Throwable e = error.get();
                if (e != null) {
                    throw new ExecutionException(e);
                }
                return value.get();
            }
            
        };
    }
    
    private void awaitForComplete(CountDownLatch latch, Disposable subscription) {
        if (latch.getCount() == 0) {
            // Synchronous observable completes before awaiting for it.
            // Skip await so InterruptedException will never be thrown.
            return;
        }
        // block until the subscription completes and then return
        try {
            latch.await();
        } catch (InterruptedException e) {
            subscription.dispose();
            // set the interrupted flag again so callers can still get it
            // for more information see https://github.com/ReactiveX/RxJava/pull/147#issuecomment-13624780
            Thread.currentThread().interrupt();
            // using Runtime so it is not checked
            throw new RuntimeException("Interrupted while waiting for subscription to complete.", e);
        }
    }
    
    /**
     * Runs the source observable to a terminal event, ignoring any values and rethrowing any exception.
     */
    public void run() {
        final CountDownLatch cdl = new CountDownLatch(1);
        final Throwable[] error = { null };
        LambdaSubscriber ls = new LambdaSubscriber(Functions.emptyConsumer(), 
        new Consumer() {
            @Override
            public void accept(Throwable e) {
                error[0] = e;
                cdl.countDown();
            }
        }, new Runnable() {
            @Override
            public void run() {
                cdl.countDown();
            }
        }, new Consumer() {
            @Override
            public void accept(Subscription s) {
                s.request(Long.MAX_VALUE);
            }
        });
        
        o.subscribe(ls);
        
        awaitForComplete(cdl, ls);
        Throwable e = error[0];
        if (e != null) {
            if (e instanceof RuntimeException) {
                throw (RuntimeException)e;
            } else {
                throw new RuntimeException(e);
            }
        }
    }
    
    /**
     * Subscribes to the source and calls the Subscriber methods on the current thread.
     * 

* The unsubscription and backpressure is composed through. * @param subscriber the subscriber to forward events and calls to in the current thread */ @Override public void subscribe(Subscriber subscriber) { final BlockingQueue queue = new LinkedBlockingQueue(); BlockingSubscriber bs = new BlockingSubscriber(queue); o.subscribe(bs); try { for (;;) { if (bs.isCancelled()) { break; } Object o = queue.poll(); if (o == null) { if (bs.isCancelled()) { break; } o = queue.take(); } if (bs.isCancelled()) { break; } if (o == BlockingSubscriber.TERMINATED) { break; } if (NotificationLite.acceptFull(o, subscriber)) { break; } } } catch (InterruptedException e) { Thread.currentThread().interrupt(); subscriber.onError(e); } finally { bs.cancel(); } } /** * Runs the source observable to a terminal event, ignoring any values and rethrowing any exception. */ public void subscribe() { run(); } /** * Subscribes to the source and calls the given action on the current thread and rethrows any exception wrapped * into OnErrorNotImplementedException. * @param onNext the callback action for each source value */ public void subscribe(final Consumer onNext) { subscribe(onNext, RxJavaPlugins.errorConsumer(), Functions.emptyRunnable()); } /** * Subscribes to the source and calls the given actions on the current thread. * @param onNext the callback action for each source value * @param onError the callback action for an error event */ public void subscribe(final Consumer onNext, final Consumer onError) { subscribe(onNext, onError, Functions.emptyRunnable()); } /** * Subscribes to the source and calls the given actions on the current thread. * @param onNext the callback action for each source value * @param onError the callback action for an error event * @param onComplete the callback action for the completion event. */ public void subscribe(final Consumer onNext, final Consumer onError, final Runnable onComplete) { subscribe(new Observer() { @Override public void onNext(T t) { onNext.accept(t); } @Override public void onError(Throwable e) { onError.accept(e); } @Override public void onComplete() { onComplete.run(); } }); } }