/** * 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.internal.operators; import java.util.Arrays; import java.util.concurrent.atomic.AtomicBoolean; import rx.Observable; import rx.Observable.OnSubscribe; import rx.Subscriber; import rx.Subscription; import rx.exceptions.CompositeException; import rx.functions.Action0; import rx.functions.Action1; import rx.functions.Func0; import rx.functions.Func1; /** * Constructs an observable sequence that depends on a resource object. */ public final class OnSubscribeUsing<T, Resource> implements OnSubscribe<T> { private final Func0<Resource> resourceFactory; private final Func1<? super Resource, ? extends Observable<? extends T>> observableFactory; private final Action1<? super Resource> dispose; private final boolean disposeEagerly; public OnSubscribeUsing(Func0<Resource> resourceFactory, Func1<? super Resource, ? extends Observable<? extends T>> observableFactory, Action1<? super Resource> dispose, boolean disposeEagerly) { this.resourceFactory = resourceFactory; this.observableFactory = observableFactory; this.dispose = dispose; this.disposeEagerly = disposeEagerly; } @Override public void call(Subscriber<? super T> subscriber) { try { // create the resource final Resource resource = resourceFactory.call(); // create an action/subscription that disposes only once final DisposeAction<Resource> disposeOnceOnly = new DisposeAction<Resource>(dispose, resource); // dispose on unsubscription subscriber.add(disposeOnceOnly); // create the observable final Observable<? extends T> source = observableFactory // create the observable .call(resource); final Observable<? extends T> observable; // supplement with on termination disposal if requested if (disposeEagerly) observable = source // dispose on completion or error .doOnTerminate(disposeOnceOnly); else observable = source; try { // start observable.unsafeSubscribe(subscriber); } catch (Throwable e) { Throwable disposeError = disposeEagerlyIfRequested(disposeOnceOnly); if (disposeError != null) subscriber.onError(new CompositeException(Arrays.asList(e, disposeError))); else // propagate error subscriber.onError(e); } } catch (Throwable e) { // then propagate error subscriber.onError(e); } } private Throwable disposeEagerlyIfRequested(final Action0 disposeOnceOnly) { if (disposeEagerly) try { disposeOnceOnly.call(); return null; } catch (Throwable e) { return e; } else return null; } private static final class DisposeAction<Resource> extends AtomicBoolean implements Action0, Subscription { private static final long serialVersionUID = 4262875056400218316L; private Action1<? super Resource> dispose; private Resource resource; private DisposeAction(Action1<? super Resource> dispose, Resource resource) { this.dispose = dispose; this.resource = resource; lazySet(false); // StoreStore barrier } @Override public void call() { if (compareAndSet(false, true)) { try { dispose.call(resource); } finally { resource = null; dispose = null; } } } @Override public boolean isUnsubscribed() { return get(); } @Override public void unsubscribe() { call(); } } }