org.apache.camel.component.ahc.AhcProducer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of camel-ahc Show documentation
Show all versions of camel-ahc Show documentation
Camel Async Http Client support
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.camel.component.ahc;
import java.io.ByteArrayOutputStream;
import java.util.concurrent.ExecutorService;
import io.netty.handler.codec.http.HttpHeaders;
import org.apache.camel.AsyncCallback;
import org.apache.camel.Exchange;
import org.apache.camel.support.DefaultAsyncProducer;
import org.asynchttpclient.AsyncHandler;
import org.asynchttpclient.AsyncHttpClient;
import org.asynchttpclient.HttpResponseBodyPart;
import org.asynchttpclient.HttpResponseStatus;
import org.asynchttpclient.Request;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
*
*/
public class AhcProducer extends DefaultAsyncProducer {
private static final Logger LOG = LoggerFactory.getLogger(AhcProducer.class);
private final AsyncHttpClient client;
private ExecutorService workerPool;
private boolean shutdownPool;
public AhcProducer(AhcEndpoint endpoint) {
super(endpoint);
this.client = endpoint.getClient();
}
@Override
public AhcEndpoint getEndpoint() {
return (AhcEndpoint) super.getEndpoint();
}
@Override
protected void doStart() throws Exception {
if (workerPool == null) {
workerPool = getEndpoint().getCamelContext().getExecutorServiceManager().newCachedThreadPool(this, "AhcWorkerPool");
shutdownPool = true;
}
super.doStart();
}
@Override
protected void doStop() throws Exception {
super.doStop();
if (shutdownPool && workerPool != null) {
getEndpoint().getCamelContext().getExecutorServiceManager().shutdown(workerPool);
workerPool = null;
}
}
@Override
public boolean process(Exchange exchange, AsyncCallback callback) {
try {
// AHC supports async processing
Request request = getEndpoint().getBinding().prepareRequest(getEndpoint(), exchange);
LOG.debug("Executing request {}", request);
client.executeRequest(request,
new AhcAsyncHandler(exchange, callback, request.getUrl(), getEndpoint().getBufferSize()));
return false;
} catch (Exception e) {
exchange.setException(e);
callback.done(true);
return true;
}
}
/**
* Camel {@link AsyncHandler} to receive callbacks during the processing of the request.
*/
private final class AhcAsyncHandler implements AsyncHandler {
private final Exchange exchange;
private final AsyncCallback callback;
private final String url;
private final ByteArrayOutputStream os;
private int contentLength;
private int statusCode;
private String statusText;
private AhcAsyncHandler(Exchange exchange, AsyncCallback callback, String url, int bufferSize) {
this.exchange = exchange;
this.callback = callback;
this.url = url;
this.os = new ByteArrayOutputStream(bufferSize);
}
@Override
public void onThrowable(Throwable t) {
if (LOG.isTraceEnabled()) {
LOG.trace("{} onThrowable {}", exchange.getExchangeId(), t.getMessage(), t);
}
try {
getEndpoint().getBinding().onThrowable(getEndpoint(), exchange, t);
} catch (Exception e) {
exchange.setException(e);
} finally {
// use worker pool to continue routing to avoid blocking ahc/netty threads
workerPool.execute(callback);
}
}
@Override
public Exchange onCompleted() throws Exception {
if (LOG.isTraceEnabled()) {
LOG.trace("{} onCompleted", exchange.getExchangeId());
}
try {
getEndpoint().getBinding().onComplete(getEndpoint(), exchange, url, os, contentLength, statusCode, statusText);
} catch (Exception e) {
exchange.setException(e);
} finally {
// use worker pool to continue routing to avoid blocking ahc/netty threads
workerPool.execute(callback);
}
return exchange;
}
@Override
public String toString() {
return "AhcAsyncHandler for exchangeId: " + exchange.getExchangeId() + " -> " + url;
}
@Override
public State onBodyPartReceived(HttpResponseBodyPart bodyPart)
throws Exception {
// write body parts to stream, which we will bind to the Camel Exchange in onComplete
os.write(bodyPart.getBodyPartBytes());
if (LOG.isTraceEnabled()) {
LOG.trace("{} onBodyPartReceived {} bytes", exchange.getExchangeId(), bodyPart.length());
}
contentLength += bodyPart.length();
return State.CONTINUE;
}
@Override
public State onStatusReceived(HttpResponseStatus responseStatus)
throws Exception {
if (LOG.isTraceEnabled()) {
LOG.trace("{} onStatusReceived {}", exchange.getExchangeId(), responseStatus);
}
try {
statusCode = responseStatus.getStatusCode();
statusText = responseStatus.getStatusText();
getEndpoint().getBinding().onStatusReceived(getEndpoint(), exchange, responseStatus);
} catch (Exception e) {
exchange.setException(e);
}
return State.CONTINUE;
}
@Override
public State onHeadersReceived(HttpHeaders headers) throws Exception {
if (LOG.isTraceEnabled()) {
LOG.trace("{} onHeadersReceived {}", exchange.getExchangeId(), headers);
}
try {
getEndpoint().getBinding().onHeadersReceived(getEndpoint(), exchange, headers);
} catch (Exception e) {
exchange.setException(e);
}
return State.CONTINUE;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy