All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.predic8.membrane.core.transport.http.HttpTransport Maven / Gradle / Ivy

There is a newer version: 5.6.0
Show newest version
/* Copyright 2009, 2012 predic8 GmbH, www.predic8.com

   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. */

package com.predic8.membrane.core.transport.http;

import java.io.IOException;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.SynchronousQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

import com.google.common.base.Objects;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.predic8.membrane.annot.MCAttribute;
import com.predic8.membrane.annot.MCElement;
import com.predic8.membrane.core.Router;
import com.predic8.membrane.core.model.IPortChangeListener;
import com.predic8.membrane.core.transport.Transport;
import com.predic8.membrane.core.transport.ssl.SSLProvider;

/**
 * @description 

* The transport receives messages from clients and invokes interceptors in the request and response flow. * The interceptors that are engaged with the transport are global and are invoked for each message flowing * through the router. *

*/ @MCElement(name="transport") public class HttpTransport extends Transport { private static Logger log = LoggerFactory.getLogger(HttpTransport.class.getName()); public static final String SOURCE_HOSTNAME = "com.predic8.membrane.transport.http.source.Hostname"; public static final String HEADER_HOST = "com.predic8.membrane.transport.http.header.Host"; public static final String SOURCE_IP = "com.predic8.membrane.transport.http.source.Ip"; private int socketTimeout = 30000; private int forceSocketCloseOnHotDeployAfter = 30000; private boolean tcpNoDelay = true; private final Map> portListenerMapping = new HashMap>(); private final List> stillRunning = new ArrayList>(); private ThreadPoolExecutor executorService = new ThreadPoolExecutor(20, Integer.MAX_VALUE, 60L, TimeUnit.SECONDS, new SynchronousQueue(), new HttpServerThreadFactory()); @Override public void init(Router router) throws Exception { super.init(router); } /** * Closes the corresponding server port. Note that connections might still be open and exchanges still running after * this method completes. */ public synchronized void closePort(IpPort p) throws IOException { Map mih = portListenerMapping.get(p.getPort()); if (mih == null || mih.isEmpty()) { return; } HttpEndpointListener plt = mih.get(p); if (plt == null) return; log.info("Closing server port: " + p); plt.closePort(); try { plt.join(); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } mih.remove(p); if (mih.isEmpty()) { portListenerMapping.remove(p.getPort()); } stillRunning.add(new WeakReference(plt)); for (IPortChangeListener listener : menuListeners) { listener.removePort(p.getPort()); } } @Override public synchronized void closeAll(boolean waitForCompletion) throws IOException { log.debug("Closing all network server sockets."); List all = new ArrayList(); for (Map v : portListenerMapping.values()) { all.addAll(v.keySet()); } for (IpPort ipPort : all) { // don't iterate thru portListenerMapping !!! closePort(ipPort); } log.debug("Closing all stream pumps."); getRouter().getStatistics().getStreamPumpStats().closeAllStreamPumps(); if (waitForCompletion) { long now = System.currentTimeMillis(); log.debug("Waiting for running exchanges to finish."); executorService.shutdown(); try { while (true) { boolean onlyIdle = System.currentTimeMillis() - now <= forceSocketCloseOnHotDeployAfter; closeConnections(onlyIdle); if (executorService.awaitTermination(5, TimeUnit.SECONDS)) break; log.warn("Still waiting for running exchanges to finish. (Set to a lower value to forcibly close connections more quickly."); } } catch (InterruptedException e) { Thread.currentThread().interrupt(); } } } private void closeConnections(boolean onlyIdle) throws IOException { ArrayList> remove = new ArrayList>(); for (WeakReference whel : stillRunning) { HttpEndpointListener hel = whel.get(); if (hel == null) remove.add(whel); else if (hel.closeConnections(onlyIdle)) remove.add(whel); } for (WeakReference whel : remove) stillRunning.remove(whel); } /** * @param port * @throws IOException */ @Override public synchronized void openPort(String ip, int port, SSLProvider sslProvider) throws IOException { if (port == -1) throw new RuntimeException("The port-attribute is missing (probably on a element)."); Map mih = portListenerMapping.get(port); if (mih == null) { mih = new HashMap(); portListenerMapping.put(port, mih); } IpPort p = new IpPort(ip, port); HttpEndpointListener hel = mih.get(p); if (hel != null) { // already listen on the same "ip:port" if (Objects.equal(sslProvider, hel.getSslProvider())) { return; // O.K. both use the equivalent ssl provider } throw new RuntimeException("Lister thread on " + p.toShortString() + " should use the same SSL config"); } if ((ip == null && !mih.isEmpty()) // '*:port' vs 'XXX:port' || (ip != null && mih.containsKey(new IpPort(null, port))) // 'XXX:port' vs '*:port' ) { throw new RuntimeException(createDiffInterfacesErrorMsg(p,mih)); } HttpEndpointListener portListenerThread = new HttpEndpointListener(p, this, sslProvider); mih.put(p, portListenerThread); portListenerThread.start(); for (IPortChangeListener listener : menuListeners) { listener.addPort(port); } } @Override public String getOpenBackendConnections(int port) { for (IpPort ipPort : portListenerMapping.get(port).keySet()) if (ipPort.port == port) return Integer.toString(portListenerMapping.get(port).get(ipPort).getNumberOfOpenConnections()); return "N/A"; } private static String createDiffInterfacesErrorMsg(IpPort p, Map mih) { final StringBuilder sb = new StringBuilder("Conflict with listening on the same net interfaces [") .append(p.toShortString()).append(", "); for (IpPort ip : mih.keySet()) { sb.append(ip.toShortString()).append(", "); } return sb.replace(sb.length() - 2, sb.length(), "]").toString(); } public int getCoreThreadPoolSize() { return executorService.getCorePoolSize(); } /** * @description

Membrane uses a thread pool to allocate threads to incomming clients connections. The core thread pool size is the minimum number of threads that are created in advance to serve client requests.

* @default 20 * @example 5 */ @MCAttribute public void setCoreThreadPoolSize(int corePoolSize) { executorService.setCorePoolSize(corePoolSize); } public int getMaxThreadPoolSize() { return executorService.getMaximumPoolSize(); } /** * @description Maximum number of threads to handle incoming connections. (Membrane uses 1 thread per incoming connection.) * @default no limit * @example 300 */ @MCAttribute public void setMaxThreadPoolSize(int value) { executorService.setMaximumPoolSize(value); } public ExecutorService getExecutorService() { return executorService; } public int getSocketTimeout() { return socketTimeout; } /** * @description Socket timout in ms. * @default 30000 */ @MCAttribute public void setSocketTimeout(int timeout) { this.socketTimeout = timeout; } public boolean isTcpNoDelay() { return tcpNoDelay; } /** * @description Whether to use the "TCP no delay" option. (=A TCP/IP packet should be constructed as soon as any * data has been written to the network buffer. With "TCP no delay" set to false, the network hardware * waits a short period of time wether the software will write more data. When the packet constructed * from the data in the buffer would exceed the MTU in size, the packet is always constructed and sent * immediately.) * @default true */ @MCAttribute public void setTcpNoDelay(boolean tcpNoDelay) { this.tcpNoDelay = tcpNoDelay; } @Override public boolean isOpeningPorts() { return true; } public int getForceSocketCloseOnHotDeployAfter() { return forceSocketCloseOnHotDeployAfter; } /** * @description When proxies.xml is changed and <router hotDeploy="true">, the Spring Context is automatically refreshed, * which restarts the {@link Router} object (=Membrane Service Proxy). Before the context refresh, all open socket connections * have to be closed. Exchange objects which are still running might delay this process. Setting forceSocketCloseOnHotDeployAfter * to a non-zero number of milliseconds forces connections to be closed after this time. * @default 30000 */ @MCAttribute public void setForceSocketCloseOnHotDeployAfter(int forceSocketCloseOnHotDeployAfter) { this.forceSocketCloseOnHotDeployAfter = forceSocketCloseOnHotDeployAfter; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy