Skip to content

1.x: Completable class to support valueless event composition + tests #3444

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 1 commit into from
Dec 7, 2015
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2,181 changes: 2,181 additions & 0 deletions src/main/java/rx/Completable.java

Large diffs are not rendered by default.

152 changes: 152 additions & 0 deletions src/main/java/rx/internal/operators/CompletableOnSubscribeConcat.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
/**
* 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.concurrent.atomic.*;

import rx.*;
import rx.Completable.*;
import rx.exceptions.MissingBackpressureException;
import rx.internal.util.unsafe.SpscArrayQueue;
import rx.plugins.RxJavaPlugins;
import rx.subscriptions.SerialSubscription;

public final class CompletableOnSubscribeConcat implements CompletableOnSubscribe {
final Observable<? extends Completable> sources;
final int prefetch;

public CompletableOnSubscribeConcat(Observable<? extends Completable> sources, int prefetch) {
this.sources = sources;
this.prefetch = prefetch;
}

@Override
public void call(CompletableSubscriber s) {
CompletableConcatSubscriber parent = new CompletableConcatSubscriber(s, prefetch);
s.onSubscribe(parent);
sources.subscribe(parent);
}

static final class CompletableConcatSubscriber
extends Subscriber<Completable> {
final CompletableSubscriber actual;
final int prefetch;
final SerialSubscription sr;

final SpscArrayQueue<Completable> queue;

volatile boolean done;

volatile int once;
static final AtomicIntegerFieldUpdater<CompletableConcatSubscriber> ONCE =
AtomicIntegerFieldUpdater.newUpdater(CompletableConcatSubscriber.class, "once");

final ConcatInnerSubscriber inner;

final AtomicInteger wip;

public CompletableConcatSubscriber(CompletableSubscriber actual, int prefetch) {
this.actual = actual;
this.prefetch = prefetch;
this.queue = new SpscArrayQueue<Completable>(prefetch);
this.sr = new SerialSubscription();
this.inner = new ConcatInnerSubscriber();
this.wip = new AtomicInteger();
add(sr);
request(prefetch);
}

@Override
public void onNext(Completable t) {
if (!queue.offer(t)) {
onError(new MissingBackpressureException());
return;
}
if (wip.getAndIncrement() == 0) {
next();
}
}

@Override
public void onError(Throwable t) {
if (ONCE.compareAndSet(this, 0, 1)) {
actual.onError(t);
return;
}
RxJavaPlugins.getInstance().getErrorHandler().handleError(t);
}

@Override
public void onCompleted() {
if (done) {
return;
}
done = true;
if (wip.getAndIncrement() == 0) {
next();
}
}

void innerError(Throwable e) {
unsubscribe();
onError(e);
}

void innerComplete() {
if (wip.decrementAndGet() != 0) {
next();
}
if (!done) {
request(1);
}
}

void next() {
boolean d = done;
Completable c = queue.poll();
if (c == null) {
if (d) {
if (ONCE.compareAndSet(this, 0, 1)) {
actual.onCompleted();
}
return;
}
RxJavaPlugins.getInstance().getErrorHandler().handleError(new IllegalStateException("Queue is empty?!"));
return;
}

c.subscribe(inner);
}

final class ConcatInnerSubscriber implements CompletableSubscriber {
@Override
public void onSubscribe(Subscription d) {
sr.set(d);
}

@Override
public void onError(Throwable e) {
innerError(e);
}

@Override
public void onCompleted() {
innerComplete();
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,96 @@
/**
* 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.concurrent.atomic.AtomicInteger;

import rx.*;
import rx.Completable.*;
import rx.subscriptions.SerialSubscription;

public final class CompletableOnSubscribeConcatArray implements CompletableOnSubscribe {
final Completable[] sources;

public CompletableOnSubscribeConcatArray(Completable[] sources) {
this.sources = sources;
}

@Override
public void call(CompletableSubscriber s) {
ConcatInnerSubscriber inner = new ConcatInnerSubscriber(s, sources);
s.onSubscribe(inner.sd);
inner.next();
}

static final class ConcatInnerSubscriber extends AtomicInteger implements CompletableSubscriber {
/** */
private static final long serialVersionUID = -7965400327305809232L;

final CompletableSubscriber actual;
final Completable[] sources;

int index;

final SerialSubscription sd;

public ConcatInnerSubscriber(CompletableSubscriber actual, Completable[] sources) {
this.actual = actual;
this.sources = sources;
this.sd = new SerialSubscription();
}

@Override
public void onSubscribe(Subscription d) {
sd.set(d);
}

@Override
public void onError(Throwable e) {
actual.onError(e);
}

@Override
public void onCompleted() {
next();
}

void next() {
if (sd.isUnsubscribed()) {
return;
}

if (getAndIncrement() != 0) {
return;
}

Completable[] a = sources;
do {
if (sd.isUnsubscribed()) {
return;
}

int idx = index++;
if (idx == a.length) {
actual.onCompleted();
return;
}

a[idx].subscribe(this);
} while (decrementAndGet() != 0);
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,135 @@
/**
* 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.Iterator;
import java.util.concurrent.atomic.AtomicInteger;

import rx.*;
import rx.Completable.*;
import rx.subscriptions.*;

public final class CompletableOnSubscribeConcatIterable implements CompletableOnSubscribe {
final Iterable<? extends Completable> sources;

public CompletableOnSubscribeConcatIterable(Iterable<? extends Completable> sources) {
this.sources = sources;
}

@Override
public void call(CompletableSubscriber s) {

Iterator<? extends Completable> it;

try {
it = sources.iterator();
} catch (Throwable e) {
s.onSubscribe(Subscriptions.unsubscribed());
s.onError(e);
return;
}

if (it == null) {
s.onSubscribe(Subscriptions.unsubscribed());
s.onError(new NullPointerException("The iterator returned is null"));
return;
}

ConcatInnerSubscriber inner = new ConcatInnerSubscriber(s, it);
s.onSubscribe(inner.sd);
inner.next();
}

static final class ConcatInnerSubscriber extends AtomicInteger implements CompletableSubscriber {
/** */
private static final long serialVersionUID = -7965400327305809232L;

final CompletableSubscriber actual;
final Iterator<? extends Completable> sources;

int index;

final SerialSubscription sd;

public ConcatInnerSubscriber(CompletableSubscriber actual, Iterator<? extends Completable> sources) {
this.actual = actual;
this.sources = sources;
this.sd = new SerialSubscription();
}

@Override
public void onSubscribe(Subscription d) {
sd.set(d);
}

@Override
public void onError(Throwable e) {
actual.onError(e);
}

@Override
public void onCompleted() {
next();
}

void next() {
if (sd.isUnsubscribed()) {
return;
}

if (getAndIncrement() != 0) {
return;
}

Iterator<? extends Completable> a = sources;
do {
if (sd.isUnsubscribed()) {
return;
}

boolean b;
try {
b = a.hasNext();
} catch (Throwable ex) {
actual.onError(ex);
return;
}

if (!b) {
actual.onCompleted();
return;
}

Completable c;

try {
c = a.next();
} catch (Throwable ex) {
actual.onError(ex);
return;
}

if (c == null) {
actual.onError(new NullPointerException("The completable returned is null"));
return;
}

c.subscribe(this);
} while (decrementAndGet() != 0);
}
}
}
Loading