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

org.eclipse.jetty.server.AbstractConnector Maven / Gradle / Ivy

There is a newer version: 4.15.102
Show newest version
//
// ========================================================================
// Copyright (c) 1995-2021 Mort Bay Consulting Pty Ltd and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
// which is available at https://www.apache.org/licenses/LICENSE-2.0.
//
// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
// ========================================================================
//

package org.eclipse.jetty.server;

import java.io.IOException;
import java.net.Socket;
import java.nio.ByteBuffer;
import java.nio.channels.ClosedByInterruptException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executor;
import java.util.concurrent.locks.Condition;
import java.util.stream.Collectors;

import org.eclipse.jetty.io.ArrayByteBufferPool;
import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.io.EndPoint;
import org.eclipse.jetty.io.ssl.SslConnection;
import org.eclipse.jetty.util.ProcessorUtils;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.annotation.ManagedAttribute;
import org.eclipse.jetty.util.annotation.ManagedObject;
import org.eclipse.jetty.util.component.Container;
import org.eclipse.jetty.util.component.ContainerLifeCycle;
import org.eclipse.jetty.util.component.Dumpable;
import org.eclipse.jetty.util.component.Graceful;
import org.eclipse.jetty.util.ssl.SslContextFactory;
import org.eclipse.jetty.util.thread.AutoLock;
import org.eclipse.jetty.util.thread.ScheduledExecutorScheduler;
import org.eclipse.jetty.util.thread.Scheduler;
import org.eclipse.jetty.util.thread.ThreadPoolBudget;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * 

An abstract implementation of {@link Connector} that provides a {@link ConnectionFactory} mechanism * for creating {@link org.eclipse.jetty.io.Connection} instances for various protocols (HTTP, SSL, etc).

* *

Connector Services

* The abstract connector manages the dependent services needed by all specific connector instances: *
    *
  • The {@link Executor} service is used to run all active tasks needed by this connector such as accepting connections * or handle HTTP requests. The default is to use the {@link Server#getThreadPool()} as an executor. *
  • *
  • The {@link Scheduler} service is used to monitor the idle timeouts of all connections and is also made available * to the connections to time such things as asynchronous request timeouts. The default is to use a new * {@link ScheduledExecutorScheduler} instance. *
  • *
  • The {@link ByteBufferPool} service is made available to all connections to be used to acquire and release * {@link ByteBuffer} instances from a pool. The default is to use a new {@link ArrayByteBufferPool} instance. *
  • *
* These services are managed as aggregate beans by the {@link ContainerLifeCycle} super class and * may either be managed or unmanaged beans. * *

Connection Factories

* The connector keeps a collection of {@link ConnectionFactory} instances, each of which are known by their * protocol name. The protocol name may be a real protocol (e.g. "http/1.1" or "h2") or it may be a private name * that represents a special connection factory. For example, the name "SSL-http/1.1" is used for * an {@link SslConnectionFactory} that has been instantiated with the {@link HttpConnectionFactory} as it's * next protocol. * *

Configuring Connection Factories

* The collection of available {@link ConnectionFactory} may be constructor injected or modified with the * methods {@link #addConnectionFactory(ConnectionFactory)}, {@link #removeConnectionFactory(String)} and * {@link #setConnectionFactories(Collection)}. Only a single {@link ConnectionFactory} instance may be configured * per protocol name, so if two factories with the same {@link ConnectionFactory#getProtocol()} are set, then * the second will replace the first. *

* The protocol factory used for newly accepted connections is specified by * the method {@link #setDefaultProtocol(String)} or defaults to the protocol of the first configured factory. *

* Each Connection factory type is responsible for the configuration of the protocols that it accepts. Thus to * configure the HTTP protocol, you pass a {@link HttpConfiguration} instance to the {@link HttpConnectionFactory} * (or other factories that can also provide HTTP Semantics). Similarly the {@link SslConnectionFactory} is * configured by passing it a {@link SslContextFactory} and a next protocol name. * *

Connection Factory Operation

* {@link ConnectionFactory}s may simply create a {@link org.eclipse.jetty.io.Connection} instance to support a specific * protocol. For example, the {@link HttpConnectionFactory} will create a {@link HttpConnection} instance * that can handle http/1.1, http/1.0 and http/0.9. *

* {@link ConnectionFactory}s may also create a chain of {@link org.eclipse.jetty.io.Connection} instances, using other {@link ConnectionFactory} instances. * For example, the {@link SslConnectionFactory} is configured with a next protocol name, so that once it has accepted * a connection and created an {@link SslConnection}, it then used the next {@link ConnectionFactory} from the * connector using the {@link #getConnectionFactory(String)} method, to create a {@link org.eclipse.jetty.io.Connection} instance that * will handle the unencrypted bytes from the {@link SslConnection}. If the next protocol is "http/1.1", then the * {@link SslConnectionFactory} will have a protocol name of "SSL-http/1.1" and lookup "http/1.1" for the protocol * to run over the SSL connection. *

* {@link ConnectionFactory}s may also create temporary {@link org.eclipse.jetty.io.Connection} instances that will exchange bytes * over the connection to determine what is the next protocol to use. For example the ALPN protocol is an extension * of SSL to allow a protocol to be specified during the SSL handshake. ALPN is used by the HTTP/2 protocol to * negotiate the protocol that the client and server will speak. Thus to accept an HTTP/2 connection, the * connector will be configured with {@link ConnectionFactory}s for "SSL-ALPN", "h2", "http/1.1" * with the default protocol being "SSL-ALPN". Thus a newly accepted connection uses "SSL-ALPN", which specifies a * SSLConnectionFactory with "ALPN" as the next protocol. Thus an SSL connection instance is created chained to an ALPN * connection instance. The ALPN connection then negotiates with the client to determined the next protocol, which * could be "h2" or the default of "http/1.1". Once the next protocol is determined, the ALPN connection * calls {@link #getConnectionFactory(String)} to create a connection instance that will replace the ALPN connection as * the connection chained to the SSL connection. *

Acceptors

* The connector will execute a number of acceptor tasks to the {@link Exception} service passed to the constructor. * The acceptor tasks run in a loop while the connector is running and repeatedly call the abstract {@link #accept(int)} method. * The implementation of the accept method must: *
    *
  1. block waiting for new connections
  2. *
  3. accept the connection (eg socket accept)
  4. *
  5. perform any configuration of the connection (eg. socket configuration)
  6. *
  7. call the {@link #getDefaultConnectionFactory()} {@link ConnectionFactory#newConnection(Connector, org.eclipse.jetty.io.EndPoint)} * method to create a new Connection instance.
  8. *
* The default number of acceptor tasks is the minimum of 1 and the number of available CPUs divided by 8. Having more acceptors may reduce * the latency for servers that see a high rate of new connections (eg HTTP/1.0 without keep-alive). Typically the default is * sufficient for modern persistent protocols (HTTP/1.1, HTTP/2 etc.) */ @ManagedObject("Abstract implementation of the Connector Interface") public abstract class AbstractConnector extends ContainerLifeCycle implements Connector, Dumpable { protected static final Logger LOG = LoggerFactory.getLogger(AbstractConnector.class); private final AutoLock _lock = new AutoLock(); private final Condition _setAccepting = _lock.newCondition(); private final Map _factories = new LinkedHashMap<>(); // Order is important on server side, so we use a LinkedHashMap private final Server _server; private final Executor _executor; private final Scheduler _scheduler; private final ByteBufferPool _byteBufferPool; private final Thread[] _acceptors; private final Set _endpoints = Collections.newSetFromMap(new ConcurrentHashMap<>()); private final Set _immutableEndPoints = Collections.unmodifiableSet(_endpoints); private Shutdown _shutdown; private HttpChannel.Listener _httpChannelListeners = HttpChannel.NOOP_LISTENER; private long _idleTimeout = 30000; private long _shutdownIdleTimeout = 1000L; private String _defaultProtocol; private ConnectionFactory _defaultConnectionFactory; private String _name; private int _acceptorPriorityDelta = -2; private boolean _accepting = true; private ThreadPoolBudget.Lease _lease; /** * @param server The server this connector will be added to. Must not be null. * @param executor An executor for this connector or null to use the servers executor * @param scheduler A scheduler for this connector or null to either a {@link Scheduler} set as a server bean or if none set, then a new {@link ScheduledExecutorScheduler} instance. * @param pool A buffer pool for this connector or null to either a {@link ByteBufferPool} set as a server bean or none set, the new {@link ArrayByteBufferPool} instance. * @param acceptors the number of acceptor threads to use, or -1 for a default value. If 0, then no acceptor threads will be launched and some other mechanism will need to be used to accept new connections. * @param factories The Connection Factories to use. */ public AbstractConnector( Server server, Executor executor, Scheduler scheduler, ByteBufferPool pool, int acceptors, ConnectionFactory... factories) { _server = server; _executor = executor != null ? executor : _server.getThreadPool(); addBean(_executor); if (executor == null) unmanage(_executor); // inherited from server if (scheduler == null) scheduler = _server.getBean(Scheduler.class); _scheduler = scheduler != null ? scheduler : new ScheduledExecutorScheduler(String.format("Connector-Scheduler-%x", hashCode()), false); addBean(_scheduler); if (pool == null) pool = _server.getBean(ByteBufferPool.class); _byteBufferPool = pool != null ? pool : new ArrayByteBufferPool(); addBean(_byteBufferPool); addEventListener(new Container.Listener() { @Override public void beanAdded(Container parent, Object bean) { if (bean instanceof HttpChannel.Listener) _httpChannelListeners = new HttpChannelListeners(getBeans(HttpChannel.Listener.class)); } @Override public void beanRemoved(Container parent, Object bean) { if (bean instanceof HttpChannel.Listener) _httpChannelListeners = new HttpChannelListeners(getBeans(HttpChannel.Listener.class)); } }); for (ConnectionFactory factory : factories) { addConnectionFactory(factory); } int cores = ProcessorUtils.availableProcessors(); if (acceptors < 0) acceptors = Math.max(1, Math.min(4, cores / 8)); if (acceptors > cores) LOG.warn("Acceptors should be <= availableProcessors: {} ", this); _acceptors = new Thread[acceptors]; } /** * Get the {@link HttpChannel.Listener}s added to the connector * as a single combined Listener. * This is equivalent to a listener that iterates over the individual * listeners returned from {@code getBeans(HttpChannel.Listener.class);}, * except that:
    *
  • The result is precomputed, so it is more efficient
  • *
  • The result is ordered by the order added.
  • *
  • The result is immutable.
  • *
* @see #getBeans(Class) * @return An unmodifiable list of EventListener beans */ public HttpChannel.Listener getHttpChannelListeners() { return _httpChannelListeners; } @Override public Server getServer() { return _server; } @Override public Executor getExecutor() { return _executor; } @Override public ByteBufferPool getByteBufferPool() { return _byteBufferPool; } @Override @ManagedAttribute("The connection idle timeout in milliseconds") public long getIdleTimeout() { return _idleTimeout; } /** *

Sets the maximum Idle time for a connection, which roughly translates to the {@link Socket#setSoTimeout(int)} * call, although with NIO implementations other mechanisms may be used to implement the timeout.

*

The max idle time is applied:

*
    *
  • When waiting for a new message to be received on a connection
  • *
  • When waiting for a new message to be sent on a connection
  • *
*

This value is interpreted as the maximum time between some progress being made on the connection. * So if a single byte is read or written, then the timeout is reset.

* * @param idleTimeout the idle timeout */ public void setIdleTimeout(long idleTimeout) { _idleTimeout = idleTimeout; if (_idleTimeout == 0) _shutdownIdleTimeout = 0; else if (_idleTimeout < _shutdownIdleTimeout) _shutdownIdleTimeout = Math.min(1000L, _idleTimeout); } public void setShutdownIdleTimeout(long idle) { _shutdownIdleTimeout = idle; } public long getShutdownIdleTimeout() { return _shutdownIdleTimeout; } /** * @return Returns the number of acceptor threads. */ @ManagedAttribute("number of acceptor threads") public int getAcceptors() { return _acceptors.length; } @Override protected void doStart() throws Exception { getConnectionFactories().stream() .filter(ConnectionFactory.Configuring.class::isInstance) .map(ConnectionFactory.Configuring.class::cast) .forEach(configuring -> configuring.configure(this)); _shutdown = new Graceful.Shutdown(this) { @Override public boolean isShutdownDone() { if (!_endpoints.isEmpty()) return false; for (Thread a : _acceptors) { if (a != null) return false; } return true; } }; if (_defaultProtocol == null) throw new IllegalStateException("No default protocol for " + this); _defaultConnectionFactory = getConnectionFactory(_defaultProtocol); if (_defaultConnectionFactory == null) throw new IllegalStateException("No protocol factory for default protocol '" + _defaultProtocol + "' in " + this); SslConnectionFactory ssl = getConnectionFactory(SslConnectionFactory.class); if (ssl != null) { String next = ssl.getNextProtocol(); ConnectionFactory cf = getConnectionFactory(next); if (cf == null) throw new IllegalStateException("No protocol factory for SSL next protocol: '" + next + "' in " + this); } _lease = ThreadPoolBudget.leaseFrom(getExecutor(), this, _acceptors.length); super.doStart(); for (int i = 0; i < _acceptors.length; i++) { Acceptor a = new Acceptor(i); addBean(a); getExecutor().execute(a); } LOG.info("Started {}", this); } protected void interruptAcceptors() { try (AutoLock lock = _lock.lock()) { for (Thread thread : _acceptors) { if (thread != null) thread.interrupt(); } } } @Override public CompletableFuture shutdown() { Shutdown shutdown = _shutdown; if (shutdown == null) return CompletableFuture.completedFuture(null); // Signal for the acceptors to stop CompletableFuture done = shutdown.shutdown(); interruptAcceptors(); // Reduce the idle timeout of existing connections for (EndPoint ep : _endpoints) ep.setIdleTimeout(getShutdownIdleTimeout()); // Return Future that waits for no acceptors and no connections. return done; } @Override public boolean isShutdown() { Shutdown shutdown = _shutdown; return shutdown == null || shutdown.isShutdown(); } @Override protected void doStop() throws Exception { if (_lease != null) _lease.close(); // Tell the acceptors we are stopping interruptAcceptors(); super.doStop(); for (Acceptor a : getBeans(Acceptor.class)) removeBean(a); _shutdown = null; LOG.info("Stopped {}", this); } public void join() throws InterruptedException { join(0); } public void join(long timeout) throws InterruptedException { try (AutoLock lock = _lock.lock()) { for (Thread thread : _acceptors) { if (thread != null) thread.join(timeout); } } } protected abstract void accept(int acceptorID) throws IOException, InterruptedException; /** * @return Is the connector accepting new connections */ public boolean isAccepting() { try (AutoLock lock = _lock.lock()) { return _accepting; } } public void setAccepting(boolean accepting) { try (AutoLock l = _lock.lock()) { _accepting = accepting; _setAccepting.signalAll(); } } @Override public ConnectionFactory getConnectionFactory(String protocol) { try (AutoLock lock = _lock.lock()) { return _factories.get(StringUtil.asciiToLowerCase(protocol)); } } @Override public T getConnectionFactory(Class factoryType) { try (AutoLock lock = _lock.lock()) { for (ConnectionFactory f : _factories.values()) { if (factoryType.isAssignableFrom(f.getClass())) return (T)f; } return null; } } public void addConnectionFactory(ConnectionFactory factory) { if (isRunning()) throw new IllegalStateException(getState()); Set toRemove = new HashSet<>(); for (String key : factory.getProtocols()) { key = StringUtil.asciiToLowerCase(key); ConnectionFactory old = _factories.remove(key); if (old != null) { if (old.getProtocol().equals(_defaultProtocol)) _defaultProtocol = null; toRemove.add(old); } _factories.put(key, factory); } // keep factories still referenced for (ConnectionFactory f : _factories.values()) { toRemove.remove(f); } // remove old factories for (ConnectionFactory old : toRemove) { removeBean(old); if (LOG.isDebugEnabled()) LOG.debug("{} removed {}", this, old); } // add new Bean addBean(factory); if (_defaultProtocol == null) _defaultProtocol = factory.getProtocol(); if (LOG.isDebugEnabled()) LOG.debug("{} added {}", this, factory); } public void addFirstConnectionFactory(ConnectionFactory factory) { if (isRunning()) throw new IllegalStateException(getState()); List existings = new ArrayList<>(_factories.values()); clearConnectionFactories(); addConnectionFactory(factory); for (ConnectionFactory existing : existings) { addConnectionFactory(existing); } } // Used from XML, do not remove. public void addIfAbsentConnectionFactory(ConnectionFactory factory) { if (isRunning()) throw new IllegalStateException(getState()); String key = StringUtil.asciiToLowerCase(factory.getProtocol()); if (!_factories.containsKey(key)) addConnectionFactory(factory); } public ConnectionFactory removeConnectionFactory(String protocol) { if (isRunning()) throw new IllegalStateException(getState()); ConnectionFactory factory = _factories.remove(StringUtil.asciiToLowerCase(protocol)); if (_factories.isEmpty()) _defaultProtocol = null; removeBean(factory); return factory; } @Override public Collection getConnectionFactories() { return _factories.values(); } public void setConnectionFactories(Collection factories) { if (isRunning()) throw new IllegalStateException(getState()); List existing = new ArrayList<>(_factories.values()); for (ConnectionFactory factory : existing) { removeConnectionFactory(factory.getProtocol()); } for (ConnectionFactory factory : factories) { if (factory != null) addConnectionFactory(factory); } } public void clearConnectionFactories() { if (isRunning()) throw new IllegalStateException(getState()); _factories.clear(); _defaultProtocol = null; } @ManagedAttribute("The priority delta to apply to acceptor threads") public int getAcceptorPriorityDelta() { return _acceptorPriorityDelta; } /** * Set the acceptor thread priority delta. *

This allows the acceptor thread to run at a different priority. * Typically this would be used to lower the priority to give preference * to handling previously accepted connections rather than accepting * new connections

* * @param acceptorPriorityDelta the acceptor priority delta */ public void setAcceptorPriorityDelta(int acceptorPriorityDelta) { int old = _acceptorPriorityDelta; _acceptorPriorityDelta = acceptorPriorityDelta; if (old != acceptorPriorityDelta && isStarted()) { for (Thread thread : _acceptors) { thread.setPriority(Math.max(Thread.MIN_PRIORITY, Math.min(Thread.MAX_PRIORITY, thread.getPriority() - old + acceptorPriorityDelta))); } } } @Override @ManagedAttribute("Protocols supported by this connector") public List getProtocols() { return new ArrayList<>(_factories.keySet()); } @ManagedAttribute("This connector's default protocol") public String getDefaultProtocol() { return _defaultProtocol; } public void setDefaultProtocol(String defaultProtocol) { _defaultProtocol = StringUtil.asciiToLowerCase(defaultProtocol); if (isRunning()) _defaultConnectionFactory = getConnectionFactory(_defaultProtocol); } @Override public ConnectionFactory getDefaultConnectionFactory() { if (isStarted()) return _defaultConnectionFactory; return getConnectionFactory(_defaultProtocol); } protected boolean handleAcceptFailure(Throwable ex) { if (isRunning()) { if (ex instanceof InterruptedException) { if (LOG.isDebugEnabled()) LOG.debug("Accept Interrupted", ex); return true; } if (ex instanceof ClosedByInterruptException) { if (LOG.isDebugEnabled()) LOG.debug("Accept Closed by Interrupt", ex); return false; } LOG.warn("Accept Failure", ex); try { // Arbitrary sleep to avoid spin looping. // Subclasses may decide for a different // sleep policy or closing the connector. Thread.sleep(1000); return true; } catch (Throwable x) { LOG.trace("IGNORED", x); } return false; } else { LOG.trace("IGNORED", ex); return false; } } private class Acceptor implements Runnable { private final int _id; private String _name; private Acceptor(int id) { _id = id; } @Override public void run() { final Thread thread = Thread.currentThread(); String name = thread.getName(); _name = String.format("%s-acceptor-%d@%x-%s", name, _id, hashCode(), AbstractConnector.this.toString()); thread.setName(_name); int priority = thread.getPriority(); if (_acceptorPriorityDelta != 0) thread.setPriority(Math.max(Thread.MIN_PRIORITY, Math.min(Thread.MAX_PRIORITY, priority + _acceptorPriorityDelta))); try (AutoLock l = _lock.lock()) { _acceptors[_id] = thread; } try { while (isRunning() && !_shutdown.isShutdown()) { try (AutoLock l = _lock.lock()) { if (!_accepting && isRunning()) { _setAccepting.await(); continue; } } catch (InterruptedException e) { continue; } try { accept(_id); } catch (Throwable x) { if (!handleAcceptFailure(x)) break; } } } finally { thread.setName(name); if (_acceptorPriorityDelta != 0) thread.setPriority(priority); try (AutoLock l = _lock.lock()) { _acceptors[_id] = null; } Shutdown shutdown = _shutdown; if (shutdown != null) shutdown.check(); } } @Override public String toString() { String name = _name; if (name == null) return String.format("acceptor-%d@%x", _id, hashCode()); return name; } } @Override public Collection getConnectedEndPoints() { return _immutableEndPoints; } protected void onEndPointOpened(EndPoint endp) { _endpoints.add(endp); } protected void onEndPointClosed(EndPoint endp) { _endpoints.remove(endp); Shutdown shutdown = _shutdown; if (shutdown != null) shutdown.check(); } @Override public Scheduler getScheduler() { return _scheduler; } @Override public String getName() { return _name; } /** * Set a connector name. A context may be configured with * virtual hosts in the form "@contextname" and will only serve * requests from the named connector, * * @param name A connector name. */ public void setName(String name) { _name = name; } @Override public String toString() { return String.format("%s@%x{%s, %s}", _name == null ? getClass().getSimpleName() : _name, hashCode(), getDefaultProtocol(), getProtocols().stream().collect(Collectors.joining(", ", "(", ")"))); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy