/* * (C) Copyright 2006-2016 Nuxeo SA (http://nuxeo.com/) and others. * * 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. * * Contributors: * bstefanescu * ataillefer */ package org.nuxeo.ecm.automation.client.jaxrs.impl; import java.io.IOException; import java.io.InputStream; import java.nio.charset.StandardCharsets; import java.nio.charset.UnsupportedCharsetException; import java.util.Map; import javax.mail.internet.MimeMultipart; import org.apache.http.Header; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; import org.apache.http.client.HttpClient; import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpPost; import org.apache.http.client.methods.HttpRequestBase; import org.apache.http.client.methods.HttpUriRequest; import org.apache.http.client.protocol.HttpClientContext; import org.apache.http.entity.StringEntity; import org.apache.http.impl.client.BasicCookieStore; import org.apache.http.params.HttpParams; import org.apache.http.protocol.BasicHttpContext; import org.apache.http.protocol.HttpContext; import org.apache.http.util.EntityUtils; import org.nuxeo.ecm.automation.client.RemoteException; import org.nuxeo.ecm.automation.client.jaxrs.spi.Connector; import org.nuxeo.ecm.automation.client.jaxrs.spi.Request; /** * Connector wrapping a {@link HttpClient} instance. * * @author <a href="mailto:bs@nuxeo.com">Bogdan Stefanescu</a> * @author <a href="mailto:ataillefer@nuxeo.com">Antoine Taillefer</a> */ public class HttpConnector implements Connector { protected final HttpClient http; /** * Timeout in milliseconds for the socket, connection manager and connection used by {@link #http}. */ protected final int httpConnectionTimeout; protected final HttpContext ctx; protected String basicAuth; public HttpConnector(HttpClient http) { this(http, 0); } /** * Allows to set a timeout for the HTTP connection to avoid infinite or quite long waiting periods if: * <ul> * <li>Nuxeo is broken or running into an infinite loop</li> * <li>the network doesn't respond at all</li> * </ul> * * @since 5.7 */ public HttpConnector(HttpClient http, int httpConnectionTimeout) { this(http, new BasicHttpContext(), httpConnectionTimeout); } public HttpConnector(HttpClient http, HttpContext ctx) { this(http, ctx, 0); } /** * @see {@link HttpConnector(HttpClient, long)} * @since 5.7 */ public HttpConnector(HttpClient http, HttpContext ctx, int httpConnectionTimeout) { ctx.setAttribute(HttpClientContext.COOKIE_STORE, new BasicCookieStore()); this.http = http; this.httpConnectionTimeout = httpConnectionTimeout; this.ctx = ctx; } @Override public Object execute(Request request) { HttpRequestBase httpRequest; if (request.getMethod() == Request.POST) { HttpPost post = new HttpPost(request.getUrl()); Object obj = request.getEntity(); if (obj != null) { HttpEntity entity; if (request.isMultiPart()) { entity = new MultipartRequestEntity((MimeMultipart) obj); } else { try { entity = new StringEntity(obj.toString(), StandardCharsets.UTF_8); } catch (UnsupportedCharsetException e) { // cannot happen throw new RuntimeException("Cannot encode into UTF-8", e); } } post.setEntity(entity); } httpRequest = post; } else { httpRequest = new HttpGet(request.getUrl()); } try { return execute(request, httpRequest); } catch (IOException e) { throw new RuntimeException("Cannot execute " + request, e); } } protected Object execute(Request request, HttpUriRequest httpReq) throws RemoteException, IOException { for (Map.Entry<String, String> entry : request.entrySet()) { httpReq.setHeader(entry.getKey(), entry.getValue()); } HttpResponse resp = executeRequestWithTimeout(httpReq); HttpEntity entity = resp.getEntity(); try { int status = resp.getStatusLine().getStatusCode(); // TODO kevin: check if it's enough regarding to entity content type String ctype = getHeaderValue(resp, "Content-Type"); String disp = getHeaderValue(resp, "Content-Disposition"); InputStream content = entity == null ? null : entity.getContent(); return request.handleResult(status, ctype, disp, content); } finally { // needed to properly release resources and return the connection to the pool EntityUtils.consume(entity); } } protected HttpResponse executeRequestWithTimeout(HttpUriRequest httpReq) throws IOException { // Set timeout for the socket, connection manager // and connection itself if (httpConnectionTimeout > 0) { HttpParams httpParams = http.getParams(); httpParams.setIntParameter("http.socket.timeout", httpConnectionTimeout); httpParams.setIntParameter("http.connection-manager.timeout", httpConnectionTimeout); httpParams.setIntParameter("http.connection.timeout", httpConnectionTimeout); } return http.execute(httpReq, ctx); } private String getHeaderValue(HttpResponse response, String key) { Header header = response.getFirstHeader(key); if (header == null) { return null; } return header.getValue(); } }