io.reactivex.rxjava3.internal.jdk8.ObservableFromStream Maven / Gradle / Ivy
/*
* 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.Observable;
import io.reactivex.rxjava3.core.Observer;
import io.reactivex.rxjava3.exceptions.Exceptions;
import io.reactivex.rxjava3.internal.disposables.EmptyDisposable;
import io.reactivex.rxjava3.operators.QueueDisposable;
import io.reactivex.rxjava3.plugins.RxJavaPlugins;
/**
* Wraps a {@link Stream} and emits its values as an {@link Observable} sequence.
* @param the element type of the Stream
* @since 3.0.0
*/
public final class ObservableFromStream extends Observable {
final Stream stream;
public ObservableFromStream(Stream stream) {
this.stream = stream;
}
@Override
protected void subscribeActual(Observer super T> observer) {
subscribeStream(observer, stream);
}
/**
* Subscribes to the Stream.
* @param the element type of the flow
* @param observer the observer to drive
* @param stream the sequence to consume
*/
public static void subscribeStream(Observer super T> observer, Stream stream) {
Iterator iterator;
try {
iterator = stream.iterator();
if (!iterator.hasNext()) {
EmptyDisposable.complete(observer);
closeSafely(stream);
return;
}
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
EmptyDisposable.error(ex, observer);
closeSafely(stream);
return;
}
StreamDisposable disposable = new StreamDisposable<>(observer, iterator, stream);
observer.onSubscribe(disposable);
disposable.run();
}
static void closeSafely(AutoCloseable c) {
try {
c.close();
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
RxJavaPlugins.onError(ex);
}
}
static final class StreamDisposable implements QueueDisposable {
final Observer super T> downstream;
Iterator iterator;
AutoCloseable closeable;
volatile boolean disposed;
boolean once;
boolean outputFused;
StreamDisposable(Observer super T> downstream, Iterator iterator, AutoCloseable closeable) {
this.downstream = downstream;
this.iterator = iterator;
this.closeable = closeable;
}
@Override
public void dispose() {
disposed = true;
run();
}
@Override
public boolean isDisposed() {
return disposed;
}
@Override
public int requestFusion(int mode) {
if ((mode & SYNC) != 0) {
outputFused = true;
return SYNC;
}
return NONE;
}
@Override
public boolean offer(@NonNull T value) {
throw new UnsupportedOperationException();
}
@Override
public boolean offer(@NonNull T v1, @NonNull T v2) {
throw new UnsupportedOperationException();
}
@Nullable
@Override
public T poll() {
if (iterator == null) {
return null;
}
if (!once) {
once = true;
} else {
if (!iterator.hasNext()) {
clear();
return null;
}
}
return Objects.requireNonNull(iterator.next(), "The Stream's Iterator.next() returned a null value");
}
@Override
public boolean isEmpty() {
Iterator it = iterator;
if (it != null) {
if (!once || it.hasNext()) {
return false;
}
clear();
}
return true;
}
@Override
public void clear() {
iterator = null;
AutoCloseable c = closeable;
closeable = null;
if (c != null) {
closeSafely(c);
}
}
public void run() {
if (outputFused) {
return;
}
Iterator iterator = this.iterator;
Observer super T> downstream = this.downstream;
for (;;) {
if (disposed) {
clear();
break;
}
T next;
try {
next = Objects.requireNonNull(iterator.next(), "The Stream's Iterator.next returned a null value");
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
downstream.onError(ex);
disposed = true;
continue;
}
if (disposed) {
continue;
}
downstream.onNext(next);
if (disposed) {
continue;
}
try {
if (iterator.hasNext()) {
continue;
}
} catch (Throwable ex) {
Exceptions.throwIfFatal(ex);
downstream.onError(ex);
disposed = true;
continue;
}
downstream.onComplete();
disposed = true;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy