com.aliyun.httpcomponent.httpclient.implementation.reactive.ReactiveDataConsumer Maven / Gradle / Ivy
package com.aliyun.httpcomponent.httpclient.implementation.reactive;
import com.aliyun.apache.hc.core5.http.Header;
import com.aliyun.apache.hc.core5.http.HttpStreamResetException;
import com.aliyun.apache.hc.core5.http.nio.AsyncDataConsumer;
import com.aliyun.apache.hc.core5.http.nio.CapacityChannel;
import com.aliyun.apache.hc.core5.util.Args;
import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.util.List;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;
final class ReactiveDataConsumer implements AsyncDataConsumer, Publisher {
private final AtomicLong requests = new AtomicLong(0);
private final BlockingQueue buffers = new LinkedBlockingQueue<>();
private final AtomicBoolean flushInProgress = new AtomicBoolean(false);
private final Object flushLock = new Object();
private final AtomicInteger windowScalingIncrement = new AtomicInteger(0);
private volatile boolean cancelled;
private volatile boolean completed;
private volatile Exception exception;
private volatile CapacityChannel capacityChannel;
private volatile Subscriber subscriber;
public void failed(final Exception cause) {
if (!completed) {
exception = cause;
flushToSubscriber();
}
}
@Override
public void updateCapacity(final CapacityChannel capacityChannel) throws IOException {
throwIfCancelled();
this.capacityChannel = capacityChannel;
signalCapacity(capacityChannel);
}
private void signalCapacity(final CapacityChannel channel) throws IOException {
final int increment = windowScalingIncrement.getAndSet(0);
if (increment > 0) {
channel.update(increment);
}
}
private void throwIfCancelled() throws IOException {
if (cancelled) {
throw new HttpStreamResetException("Downstream subscriber to ReactiveDataConsumer cancelled");
}
}
@Override
public void consume(final ByteBuffer byteBuffer) throws IOException {
if (completed) {
throw new IllegalStateException("Received data past end of stream");
}
throwIfCancelled();
final byte[] copy = new byte[byteBuffer.remaining()];
byteBuffer.get(copy);
buffers.add(ByteBuffer.wrap(copy));
flushToSubscriber();
}
@Override
public void streamEnd(final List trailers) {
completed = true;
flushToSubscriber();
}
@Override
public void releaseResources() {
this.capacityChannel = null;
}
private void flushToSubscriber() {
synchronized (flushLock) {
final Subscriber s = subscriber;
if (flushInProgress.getAndSet(true)) {
return;
}
try {
if (s == null) {
return;
}
if (exception != null) {
subscriber = null;
s.onError(exception);
return;
}
ByteBuffer next;
while (requests.get() > 0 && ((next = buffers.poll()) != null)) {
final int bytesFreed = next.remaining();
s.onNext(next);
requests.decrementAndGet();
windowScalingIncrement.addAndGet(bytesFreed);
}
final CapacityChannel localChannel = capacityChannel;
if (localChannel != null) {
try {
signalCapacity(localChannel);
} catch (final IOException e) {
exception = e;
s.onError(e);
return;
}
}
if (completed && buffers.isEmpty()) {
subscriber = null;
s.onComplete();
}
} finally {
flushInProgress.set(false);
}
}
}
@Override
public void subscribe(final Subscriber subscriber) {
this.subscriber = Args.notNull(subscriber, "subscriber");
subscriber.onSubscribe(new Subscription() {
@Override
public void request(final long increment) {
if (increment <= 0) {
failed(new IllegalArgumentException("The number of elements requested must be strictly positive"));
return;
}
requests.addAndGet(increment);
flushToSubscriber();
}
@Override
public void cancel() {
ReactiveDataConsumer.this.cancelled = true;
ReactiveDataConsumer.this.subscriber = null;
}
});
}
}