/*
* Copyright 2017-2020 original authors
*
* 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
*
* https://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 io.micronaut.reactive.rxjava2;
import io.micronaut.core.annotation.Internal;
import io.micronaut.scheduling.instrument.InvocationInstrumenter;
import io.reactivex.Completable;
import io.reactivex.CompletableObserver;
import io.reactivex.Flowable;
import io.reactivex.FlowableSubscriber;
import io.reactivex.Maybe;
import io.reactivex.MaybeObserver;
import io.reactivex.Observable;
import io.reactivex.Observer;
import io.reactivex.Single;
import io.reactivex.SingleObserver;
import io.reactivex.flowables.ConnectableFlowable;
import io.reactivex.internal.fuseable.HasUpstreamCompletableSource;
import io.reactivex.internal.fuseable.HasUpstreamMaybeSource;
import io.reactivex.internal.fuseable.HasUpstreamObservableSource;
import io.reactivex.internal.fuseable.HasUpstreamPublisher;
import io.reactivex.internal.fuseable.HasUpstreamSingleSource;
import io.reactivex.observables.ConnectableObservable;
import io.reactivex.parallel.ParallelFlowable;
import org.reactivestreams.Subscriber;
Inspired by code in Brave. Provides general instrumentation abstraction for RxJava2.
See https://github.com/openzipkin/brave/tree/master/context/rxjava2/src/main/java/brave/context/rxjava2/internal.
Author: graemerocher Since: 1.1
/**
* Inspired by code in Brave. Provides general instrumentation abstraction for RxJava2.
* See https://github.com/openzipkin/brave/tree/master/context/rxjava2/src/main/java/brave/context/rxjava2/internal.
*
* @author graemerocher
* @since 1.1
*/
@Internal
final class RxInstrumentedWrappers {
Wrap a subscriber.
Params: - downstream – The downstream subscriber
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a subscriber.
*
* @param downstream The downstream subscriber
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> Subscriber<T> wrap(Subscriber<T> downstream, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(downstream)) {
return downstream;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter != null) {
if (downstream instanceof FlowableSubscriber) {
return new RxInstrumentedFlowableSubscriber<>(downstream, instrumenterFactory);
} else {
return new RxInstrumentedSubscriber<>(downstream, instrumenterFactory);
}
} else {
return downstream;
}
}
Wrap a completable source.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Returns: The wrapped subscriber
/**
* Wrap a completable source.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @return The wrapped subscriber
*/
static Completable wrap(Completable source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter == null) {
return source;
}
return new RxInstrumentedCompletable(source, instrumenter);
}
Wrap a maybe.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a maybe.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> Maybe<T> wrap(Maybe<T> source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter == null) {
return source;
}
return new RxInstrumentedMaybe<>(source, instrumenter);
}
Wrap a single.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a single.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> Single<T> wrap(Single<T> source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter == null) {
return source;
}
return new RxInstrumentedSingle<>(source, instrumenter);
}
Wrap a observable.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a observable.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> Observable<T> wrap(Observable<T> source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter == null) {
return source;
}
return new RxInstrumentedObservable<>(source, instrumenter);
}
Wrap a connectable observable.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a connectable observable.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> ConnectableObservable<T> wrap(ConnectableObservable<T> source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter == null) {
return source;
}
return new RxInstrumentedConnectableObservable<>(source, instrumenter);
}
Wrap a flowable.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a flowable.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> Flowable<T> wrap(Flowable<T> source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter == null) {
return source;
}
return new RxInstrumentedFlowable<>(source, instrumenter);
}
Wrap a flowable.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a flowable.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> ConnectableFlowable<T> wrap(ConnectableFlowable<T> source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter == null) {
return source;
}
return new RxInstrumentedConnectableFlowable<>(source, instrumenter);
}
Wrap a parallel flowable.
Params: - source – The source
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a parallel flowable.
*
* @param source The source
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> ParallelFlowable<T> wrap(ParallelFlowable<T> source, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(source)) {
return source;
}
return new RxInstrumentedParallelFlowable<>(source, instrumenterFactory);
}
Wrap a observer.
Params: - downstream – The downstream observer
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a observer.
*
* @param downstream The downstream observer
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> Observer<T> wrap(Observer<T> downstream, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(downstream)) {
return downstream;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter != null) {
return new RxInstrumentedObserver<>(downstream, instrumenterFactory);
}
return downstream;
}
Wrap a observer.
Params: - downstream – The downstream observer
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a observer.
*
* @param downstream The downstream observer
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> SingleObserver<T> wrap(SingleObserver<T> downstream, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(downstream)) {
return downstream;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter != null) {
return new RxInstrumentedSingleObserver<>(downstream, instrumenterFactory);
}
return downstream;
}
Wrap a observer.
Params: - downstream – The downstream observer
- instrumenterFactory – The instrumenterFactory
Type parameters: - <T> – The type
Returns: The wrapped subscriber
/**
* Wrap a observer.
*
* @param downstream The downstream observer
* @param instrumenterFactory The instrumenterFactory
* @param <T> The type
* @return The wrapped subscriber
*/
static <T> MaybeObserver<T> wrap(MaybeObserver<T> downstream, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(downstream)) {
return downstream;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter != null) {
return new RxInstrumentedMaybeObserver<>(downstream, instrumenterFactory);
}
return downstream;
}
Wrap a observer.
Params: - downstream – The downstream observer
- instrumenterFactory – The instrumenterFactory
Returns: The wrapped subscriber
/**
* Wrap a observer.
*
* @param downstream The downstream observer
* @param instrumenterFactory The instrumenterFactory
* @return The wrapped subscriber
*/
static CompletableObserver wrap(CompletableObserver downstream, RxInstrumenterFactory instrumenterFactory) {
if (skipWrap(downstream)) {
return downstream;
}
final InvocationInstrumenter instrumenter = instrumenterFactory.create();
if (instrumenter != null) {
return new RxInstrumentedCompletableObserver(downstream, instrumenterFactory);
}
return downstream;
}
Check if wrap should be skipped for the source object.
Params: - source – The source
Returns: true if the source should be wrapped
/**
* Check if wrap should be skipped for the source object.
*
* @param source The source
* @return true if the source should be wrapped
*/
private static boolean skipWrap(Object source) {
if (source instanceof RxInstrumentedComponent) {
return true;
}
if (source instanceof HasUpstreamObservableSource) {
return skipWrap(((HasUpstreamObservableSource) source).source());
}
if (source instanceof HasUpstreamMaybeSource) {
return skipWrap(((HasUpstreamMaybeSource) source).source());
}
if (source instanceof HasUpstreamCompletableSource) {
return skipWrap(((HasUpstreamCompletableSource) source).source());
}
if (source instanceof HasUpstreamPublisher) {
return skipWrap(((HasUpstreamPublisher) source).source());
}
if (source instanceof HasUpstreamSingleSource) {
return skipWrap(((HasUpstreamSingleSource<Object>) source).source());
}
return false;
}
}