io.reactivex.rxjava3.internal.jdk8.MaybeFlattenStreamAsObservable Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rxjava Show documentation
Show all versions of rxjava Show documentation
Reactive Extensions for Java
/*
* Copyright (c) 2016-present, RxJava Contributors.
*
* 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 io.reactivex.rxjava3.internal.jdk8;
import java.util.*;
import java.util.stream.Stream;
import io.reactivex.rxjava3.annotations.*;
import io.reactivex.rxjava3.core.*;
import io.reactivex.rxjava3.core.Observable;
import io.reactivex.rxjava3.core.Observer;
import io.reactivex.rxjava3.disposables.Disposable;
import io.reactivex.rxjava3.exceptions.Exceptions;
import io.reactivex.rxjava3.functions.Function;
import io.reactivex.rxjava3.internal.disposables.DisposableHelper;
import io.reactivex.rxjava3.internal.observers.BasicIntQueueDisposable;
import io.reactivex.rxjava3.plugins.RxJavaPlugins;
/**
* Map the success value into a Java {@link Stream} and emits its values.
*
* @param the source value type
* @param the output value type
* @since 3.0.0
*/
public final class MaybeFlattenStreamAsObservable extends Observable {
final Maybe source;
final Function super T, ? extends Stream extends R>> mapper;
public MaybeFlattenStreamAsObservable(Maybe source, Function super T, ? extends Stream extends R>> mapper) {
this.source = source;
this.mapper = mapper;
}
@Override
protected void subscribeActual(@NonNull Observer super R> s) {
source.subscribe(new FlattenStreamMultiObserver<>(s, mapper));
}
static final class FlattenStreamMultiObserver
extends BasicIntQueueDisposable
implements MaybeObserver, SingleObserver {
private static final long serialVersionUID = 7363336003027148283L;
final Observer super R> downstream;
final Function super T, ? extends Stream extends R>> mapper;
Disposable upstream;
volatile Iterator extends R> iterator;
AutoCloseable close;
boolean once;
volatile boolean disposed;
boolean outputFused;
FlattenStreamMultiObserver(Observer super R> downstream, Function super T, ? extends Stream extends R>> mapper) {
this.downstream = downstream;
this.mapper = mapper;
}
@Override
public void onSubscribe(@NonNull Disposable d) {
if (DisposableHelper.validate(this.upstream, d)) {
this.upstream = d;
downstream.onSubscribe(this);
}
}
@Override
public void onSuccess(@NonNull T t) {
try {
Stream extends R> stream = Objects.requireNonNull(mapper.apply(t), "The mapper returned a null Stream");
Iterator extends R> iterator = stream.iterator();
AutoCloseable c = stream;
if (!iterator.hasNext()) {
downstream.onComplete();
close(c);
return;
}
this.iterator = iterator;
this.close = stream;
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
downstream.onError(ex);
return;
}
drain();
}
@Override
public void onError(@NonNull Throwable e) {
downstream.onError(e);
}
@Override
public void onComplete() {
downstream.onComplete();
}
@Override
public void dispose() {
disposed = true;
upstream.dispose();
if (!outputFused) {
drain();
}
}
@Override
public boolean isDisposed() {
return disposed;
}
@Override
public int requestFusion(int mode) {
if ((mode & ASYNC) != 0) {
outputFused = true;
return ASYNC;
}
return NONE;
}
@Override
public @Nullable R poll() throws Throwable {
Iterator extends R> it = iterator;
if (it != null) {
if (once) {
if (!it.hasNext()) {
clear();
return null;
}
} else {
once = true;
}
return it.next();
}
return null;
}
@Override
public boolean isEmpty() {
Iterator extends R> it = iterator;
if (it != null) {
if (!once) {
return false;
}
if (it.hasNext()) {
return false;
}
clear();
}
return true;
}
@Override
public void clear() {
iterator = null;
AutoCloseable close = this.close;
this.close = null;
close(close);
}
void close(AutoCloseable c) {
try {
if (c != null) {
c.close();
}
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
RxJavaPlugins.onError(ex);
}
}
void drain() {
if (getAndIncrement() != 0) {
return;
}
int missed = 1;
Observer super R> downstream = this.downstream;
Iterator extends R> it = iterator;
for (;;) {
if (disposed) {
clear();
} else {
if (outputFused) {
downstream.onNext(null);
downstream.onComplete();
} else {
R item;
try {
item = it.next();
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
downstream.onError(ex);
disposed = true;
continue;
}
if (disposed) {
continue;
}
downstream.onNext(item);
if (disposed) {
continue;
}
boolean has;
try {
has = it.hasNext();
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
downstream.onError(ex);
disposed = true;
continue;
}
if (disposed) {
continue;
}
if (!has) {
downstream.onComplete();
disposed = true;
}
continue;
}
}
missed = addAndGet(-missed);
if (missed == 0) {
return;
}
}
}
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy