org.eclipse.jetty.server.HttpChannel Maven / Gradle / Ivy
//
// ========================================================================
// Copyright (c) 1995 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.util.concurrent.TimeoutException;
import java.util.function.Consumer;
import java.util.function.Predicate;
import org.eclipse.jetty.http.ComplianceViolation;
import org.eclipse.jetty.http.MetaData;
import org.eclipse.jetty.server.internal.HttpChannelState;
import org.eclipse.jetty.util.thread.Invocable;
/**
* Represents the state of an HTTP request/response cycle.
* {@code HttpChannel} links the lower (closer to the network) layer {@link HttpStream}
* with the upper (application code) layer {@link Handler}.
* An {@code HttpChannel} instance may be used for many HTTP request/response cycles
* from the same connection; however, only a single cycle may be active at any time.
* Default implementations of this interface may be created via {@link DefaultFactory}.
*/
public interface HttpChannel extends Invocable
{
/**
* @return the {@code ConnectionMetaData} associated with this channel.
*/
ConnectionMetaData getConnectionMetaData();
/**
* Set the {@link HttpStream} to associate to this channel..
* @param httpStream the {@link HttpStream} to associate to this channel.
*/
void setHttpStream(HttpStream httpStream);
/**
* @return whether the request has been passed to the root {@link Handler}.
*/
boolean isRequestHandled();
/**
* {@link HttpStream} invokes this method when the metadata of an HTTP
* request (method, URI and headers, but not content) has been parsed.
* The returned {@code Runnable} invokes the root {@link Handler}.
*
* @param metaData the HTTP request metadata.
* @return a {@code Runnable} that invokes the root {@link Handler}.
*/
Runnable onRequest(MetaData.Request metaData);
/**
* Returns the {@link Request} object, if available.
* The {@code Request} object is only available after a call to
* {@link #onRequest(MetaData.Request)} has been made.
*
* @return the {@code Request} object, or null if the {@code Request} object
* is not yet available.
*/
Request getRequest();
/**
* {@link HttpStream} invokes this method when more HTTP request content is available.
*
* @return the last {@code Runnable} passed to {@link Request#demand(Runnable)},
* or {@code null} if there is no demand for content.
*/
Runnable onContentAvailable();
/**
* Notifies this {@code HttpChannel} that an idle timeout happened.
*
* @param idleTimeout the timeout.
* @return a {@code Runnable} that performs the timeout action, or {@code null}
* if no action need be performed by the calling thread
* @see Request#addIdleTimeoutListener(Predicate)
*/
Runnable onIdleTimeout(TimeoutException idleTimeout);
/**
* Notifies this {@code HttpChannel} that an asynchronous failure happened.
* Typical failure examples could be protocol failures (for example, invalid request bytes).
*
* @param failure the failure cause.
* @return a {@code Runnable} that performs the failure action, or {@code null}
* if no failure action needs be performed by the calling thread
* @see Request#addFailureListener(Consumer)
*/
Runnable onFailure(Throwable failure);
/**
* Notifies this {@code HttpChannel} that an asynchronous notification was received indicating
* a remote failure happened.
* Typical failure examples could be HTTP/2 resets.
*
* @param failure the failure cause.
* @return a {@code Runnable} that performs the failure action, or {@code null}
* if no failure action needs be performed by the calling thread
* @see Request#addFailureListener(Consumer)
*/
Runnable onRemoteFailure(Throwable failure);
/**
* Notifies this {@code HttpChannel} that an asynchronous close happened.
*
* @return a {@code Runnable} that performs the close action, or {@code null}
* if no close action needs be performed by the calling thread
*/
default Runnable onClose()
{
return null;
}
/**
* Recycle the HttpChannel, so that a new cycle of calling {@link #setHttpStream(HttpStream)},
* {@link #onRequest(MetaData.Request)} etc. may be performed on the channel.
* @see #initialize()
*/
void recycle();
/**
* Initialize the HttpChannel when a new cycle of request handling begins.
* @see #recycle()
*/
void initialize();
/**
* @return the active {@link ComplianceViolation.Listener}
*/
ComplianceViolation.Listener getComplianceViolationListener();
/**
* @param request attempt to resolve the HttpChannel from the provided request
* @return the HttpChannel if found
* @throws IllegalStateException if unable to find HttpChannel
*/
static HttpChannel from(Request request)
{
if (Request.unWrap(request).getComponents() instanceof HttpChannel httpChannel)
return httpChannel;
throw new IllegalStateException("Unable to find HttpChannel from " + request);
}
/**
* A factory for {@link HttpChannel} instances.
*
* @see DefaultFactory
*/
interface Factory
{
/**
* @param connectionMetaData the {@code ConnectionMetaData} associated with the channel.
* @return a new {@link HttpChannel} instance.
*/
HttpChannel newHttpChannel(ConnectionMetaData connectionMetaData);
}
/**
* The factory that creates default implementations of {@link HttpChannel}.
*/
class DefaultFactory implements Factory
{
@Override
public HttpChannel newHttpChannel(ConnectionMetaData connectionMetaData)
{
return new HttpChannelState(connectionMetaData);
}
}
}