
software.amazon.awssdk.utils.async.FlatteningSubscriber Maven / Gradle / Ivy
Go to download
A single bundled dependency that includes all service and dependent JARs with third-party libraries
relocated to different namespaces.
/*
* Copyright 2010-2018 Amazon.com, Inc. or its affiliates. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License").
* You may not use this file except in compliance with the License.
* A copy of the License is located at
*
* http://aws.amazon.com/apache2.0
*
* or in the "license" file accompanying this file. This file 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 software.amazon.awssdk.utils.async;
import java.util.LinkedList;
import java.util.Queue;
import java.util.concurrent.atomic.AtomicLong;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import software.amazon.awssdk.annotations.SdkInternalApi;
@SdkInternalApi
public class FlatteningSubscriber extends DelegatingSubscriber, U> {
private final AtomicLong demand = new AtomicLong(0);
private final Object lock = new Object();
private boolean requestedNextBatch;
private Queue currentBatch;
private boolean onCompleteCalled = false;
private Subscription sourceSubscription;
public FlatteningSubscriber(Subscriber super U> subscriber) {
super(subscriber);
currentBatch = new LinkedList<>();
}
@Override
public void onSubscribe(Subscription subscription) {
sourceSubscription = subscription;
subscriber.onSubscribe(new Subscription() {
@Override
public void request(long l) {
synchronized (lock) {
demand.addAndGet(l);
if (currentBatch.isEmpty() && !requestedNextBatch) {
requestedNextBatch = true;
sourceSubscription.request(1);
} else {
fulfillDemand();
}
}
}
@Override
public void cancel() {
subscription.cancel();
}
});
}
@Override
public void onNext(Iterable nextItems) {
synchronized (lock) {
currentBatch = StreamSupport.stream(nextItems.spliterator(), false)
.collect(Collectors.toCollection(LinkedList::new));
fulfillDemand();
}
}
private void fulfillDemand() {
while (demand.decrementAndGet() > 0 && !currentBatch.isEmpty()) {
subscriber.onNext(currentBatch.poll());
}
if (onCompleteCalled && currentBatch.isEmpty()) {
subscriber.onComplete();
} else if (currentBatch.isEmpty() && demand.get() > 0) {
requestedNextBatch = true;
sourceSubscription.request(1);
}
}
@Override
public void onComplete() {
synchronized (lock) {
onCompleteCalled = true;
if (currentBatch.isEmpty()) {
subscriber.onComplete();
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy