org.asynchttpclient.netty.request.body.NettyReactiveStreamsBody Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of async-http-client-netty4 Show documentation
Show all versions of async-http-client-netty4 Show documentation
The Async Http Client Netty 4 Provider.
/*
* Copyright (c) 2015 AsyncHttpClient Project. All rights reserved.
*
* This program is licensed to you under the Apache License Version 2.0,
* and you may not use this file except in compliance with the Apache License Version 2.0.
* You may obtain a copy of the Apache License Version 2.0 at http://www.apache.org/licenses/LICENSE-2.0.
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the Apache License Version 2.0 is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the Apache License Version 2.0 for the specific language governing permissions and limitations there under.
*/
package org.asynchttpclient.netty.request.body;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.util.NoSuchElementException;
import org.asynchttpclient.netty.NettyResponseFuture;
import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.typesafe.netty.HandlerSubscriber;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.handler.codec.http.DefaultHttpContent;
import io.netty.handler.codec.http.HttpContent;
import io.netty.handler.codec.http.LastHttpContent;
import io.netty.util.concurrent.EventExecutor;
public class NettyReactiveStreamsBody implements NettyBody {
private static final Logger LOGGER = LoggerFactory.getLogger(NettyReactiveStreamsBody.class);
private static final String NAME_IN_CHANNEL_PIPELINE = "request-body-streamer";
private final Publisher publisher;
public NettyReactiveStreamsBody(Publisher publisher) {
this.publisher = publisher;
}
@Override
public long getContentLength() {
return -1L;
}
@Override
public String getContentType() {
return null;
}
@Override
public void write(Channel channel, NettyResponseFuture> future) throws IOException {
if (future.isStreamWasAlreadyConsumed()) {
LOGGER.warn("Stream has already been consumed and cannot be reset");
} else {
future.setStreamWasAlreadyConsumed(true);
NettySubscriber subscriber = new NettySubscriber(channel, future);
channel.pipeline().addLast(NAME_IN_CHANNEL_PIPELINE, subscriber);
publisher.subscribe(new SubscriberAdapter(subscriber));
}
}
private static class SubscriberAdapter implements Subscriber {
private volatile Subscriber subscriber;
public SubscriberAdapter(Subscriber subscriber) {
this.subscriber = subscriber;
}
@Override
public void onSubscribe(Subscription s) {
subscriber.onSubscribe(s);
}
@Override
public void onNext(ByteBuffer t) {
ByteBuf buffer = Unpooled.wrappedBuffer(t.array());
HttpContent content = new DefaultHttpContent(buffer);
subscriber.onNext(content);
}
@Override
public void onError(Throwable t) {
subscriber.onError(t);
}
@Override
public void onComplete() {
subscriber.onComplete();
}
}
private static class NettySubscriber extends HandlerSubscriber {
private static final Logger LOGGER = LoggerFactory.getLogger(NettySubscriber.class);
private final Channel channel;
private final NettyResponseFuture> future;
public NettySubscriber(Channel channel, NettyResponseFuture> future) {
super(channel.eventLoop());
this.channel = channel;
this.future = future;
}
@Override
protected void complete() {
EventExecutor executor = channel.eventLoop();
executor.execute(new Runnable() {
@Override
public void run() {
channel.writeAndFlush(LastHttpContent.EMPTY_LAST_CONTENT).addListener(new ChannelFutureListener() {
@Override
public void operationComplete(ChannelFuture future) throws Exception {
removeFromPipeline();
}
});
}
});
}
@Override
protected void error(Throwable error) {
if(error == null) throw null;
removeFromPipeline();
future.abort(error);
}
private void removeFromPipeline() {
try {
channel.pipeline().remove(this);
LOGGER.debug(String.format("Removed handler %s from pipeline.", NAME_IN_CHANNEL_PIPELINE));
} catch (NoSuchElementException e) {
LOGGER.debug(String.format("Failed to remove handler %s from pipeline.", NAME_IN_CHANNEL_PIPELINE), e);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy