hu.akarnokd.rxjava2.debug.RxJavaAssemblyTracking Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rxjava2-extensions Show documentation
Show all versions of rxjava2-extensions Show documentation
rxjava2-extensions developed by David Karnok
/*
* Copyright 2016-2017 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.rxjava2.debug;
import java.util.concurrent.Callable;
import java.util.concurrent.atomic.AtomicBoolean;
import io.reactivex.*;
import io.reactivex.flowables.ConnectableFlowable;
import io.reactivex.functions.Function;
import io.reactivex.internal.fuseable.ScalarCallable;
import io.reactivex.observables.ConnectableObservable;
import io.reactivex.parallel.ParallelFlowable;
import io.reactivex.plugins.RxJavaPlugins;
/**
* Utility class to enable and disable tracking of operator application ({@code source.map().filter()})
* by capturing the current stacktrace (warning: very expensive!), have it in a debug-time accessible
* field (when walking the references in a debugger) and append it to exceptions passing by the
* regular {@code onError}.
*/
public final class RxJavaAssemblyTracking {
/** Simply lock out concurrent state changes. */
static final AtomicBoolean lock = new AtomicBoolean();
/** Utility class. */
private RxJavaAssemblyTracking() {
throw new IllegalStateException("No instances!");
}
/**
* Enable the assembly tracking.
*/
@SuppressWarnings({ "unchecked", "rawtypes" })
public static void enable() {
if (lock.compareAndSet(false, true)) {
RxJavaPlugins.setOnFlowableAssembly(new Function() {
@Override
public Flowable apply(Flowable f) throws Exception {
if (f instanceof Callable) {
if (f instanceof ScalarCallable) {
return new FlowableOnAssemblyScalarCallable(f);
}
return new FlowableOnAssemblyCallable(f);
}
return new FlowableOnAssembly(f);
}
});
RxJavaPlugins.setOnConnectableFlowableAssembly(new Function() {
@Override
public ConnectableFlowable apply(ConnectableFlowable f) throws Exception {
return new FlowableOnAssemblyConnectable(f);
}
});
RxJavaPlugins.setOnObservableAssembly(new Function() {
@Override
public Observable apply(Observable f) throws Exception {
if (f instanceof Callable) {
if (f instanceof ScalarCallable) {
return new ObservableOnAssemblyScalarCallable(f);
}
return new ObservableOnAssemblyCallable(f);
}
return new ObservableOnAssembly(f);
}
});
RxJavaPlugins.setOnConnectableObservableAssembly(new Function() {
@Override
public ConnectableObservable apply(ConnectableObservable f) throws Exception {
return new ObservableOnAssemblyConnectable(f);
}
});
RxJavaPlugins.setOnSingleAssembly(new Function() {
@Override
public Single apply(Single f) throws Exception {
if (f instanceof Callable) {
if (f instanceof ScalarCallable) {
return new SingleOnAssemblyScalarCallable(f);
}
return new SingleOnAssemblyCallable(f);
}
return new SingleOnAssembly(f);
}
});
RxJavaPlugins.setOnCompletableAssembly(new Function() {
@Override
public Completable apply(Completable f) throws Exception {
if (f instanceof Callable) {
if (f instanceof ScalarCallable) {
return new CompletableOnAssemblyScalarCallable(f);
}
return new CompletableOnAssemblyCallable(f);
}
return new CompletableOnAssembly(f);
}
});
RxJavaPlugins.setOnMaybeAssembly(new Function() {
@Override
public Maybe apply(Maybe f) throws Exception {
if (f instanceof Callable) {
if (f instanceof ScalarCallable) {
return new MaybeOnAssemblyScalarCallable(f);
}
return new MaybeOnAssemblyCallable(f);
}
return new MaybeOnAssembly(f);
}
});
RxJavaPlugins.setOnParallelAssembly(new Function() {
@Override
public ParallelFlowable apply(ParallelFlowable t) throws Exception {
return new ParallelFlowableOnAssembly(t);
}
});
lock.set(false);
}
}
/**
* Disable the assembly tracking.
*/
public static void disable() {
if (lock.compareAndSet(false, true)) {
RxJavaPlugins.setOnCompletableAssembly(null);
RxJavaPlugins.setOnSingleAssembly(null);
RxJavaPlugins.setOnMaybeAssembly(null);
RxJavaPlugins.setOnObservableAssembly(null);
RxJavaPlugins.setOnFlowableAssembly(null);
RxJavaPlugins.setOnConnectableObservableAssembly(null);
RxJavaPlugins.setOnConnectableFlowableAssembly(null);
RxJavaPlugins.setOnParallelAssembly(null);
lock.set(false);
}
}
}