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

io.vertx.core.http.HttpServerRequest Maven / Gradle / Ivy

/*
 * Copyright (c) 2011-2019 Contributors to the Eclipse Foundation
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License 2.0 which is available at
 * http://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 io.vertx.core.http;

import io.netty.handler.codec.DecoderResult;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.TooLongHttpHeaderException;
import io.netty.handler.codec.http.TooLongHttpLineException;
import io.vertx.codegen.annotations.*;
import io.vertx.core.Future;
import io.vertx.core.Handler;
import io.vertx.core.MultiMap;
import io.vertx.core.buffer.Buffer;
import io.vertx.core.http.impl.HttpUtils;
import io.vertx.core.net.HostAndPort;
import io.vertx.core.net.NetSocket;
import io.vertx.core.net.SocketAddress;
import io.vertx.core.streams.ReadStream;

import javax.net.ssl.SSLSession;
import java.util.*;
import java.util.stream.Collectors;

/**
 * Represents a server-side HTTP request.
 * 

* Instances are created for each request and passed to the user via a handler. *

* Each instance of this class is associated with a corresponding {@link HttpServerResponse} instance via * {@link #response}.

* It implements {@link io.vertx.core.streams.ReadStream} so it can be used with * {@link io.vertx.core.streams.Pipe} to pipe data with flow control. *

* * @author Tim Fox */ @VertxGen public interface HttpServerRequest extends ReadStream { /** * The default invalid request handler, it uses the {@link #decoderResult()} cause and the request information * to determine the status code of the response to be sent. * *

    *
  • When the cause is an instance of {@code io.netty.handler.codec.TooLongFrameException} and the error message * starts with An HTTP line is larger than the {@code REQUEST_URI_TOO_LONG} status is sent
  • *
  • Otherwise when the cause is an instance of {@code io.netty.handler.codec.TooLongFrameException} and the error message * starts with HTTP header is larger than the {@code REQUEST_HEADER_FIELDS_TOO_LARGE} status is sent
  • *
  • Otherwise then {@code BAD_REQUEST} status is sent
  • *
*/ @GenIgnore Handler DEFAULT_INVALID_REQUEST_HANDLER = request -> { DecoderResult result = request.decoderResult(); Throwable cause = result.cause(); HttpResponseStatus status = null; if (cause instanceof TooLongHttpLineException) { status = HttpResponseStatus.REQUEST_URI_TOO_LONG; } else if (cause instanceof TooLongHttpHeaderException) { status = HttpResponseStatus.REQUEST_HEADER_FIELDS_TOO_LARGE; } if (status == null) { status = HttpResponseStatus.BAD_REQUEST; } HttpServerResponse response = request.response(); response.setStatusCode(status.code()).end(); request.connection().close(); }; @Override HttpServerRequest exceptionHandler(Handler handler); @Override HttpServerRequest handler(Handler handler); @Override HttpServerRequest pause(); @Override HttpServerRequest resume(); @Override HttpServerRequest fetch(long amount); @Override HttpServerRequest endHandler(Handler endHandler); /** * @return the HTTP version of the request */ HttpVersion version(); /** * @return the HTTP method for the request. */ HttpMethod method(); /** * @return true if this {@link io.vertx.core.net.NetSocket} is encrypted via SSL/TLS */ default boolean isSSL() { return connection().isSsl(); } /** * @return the scheme of the request */ @Nullable String scheme(); /** * @return the URI of the request. This is usually a relative URI */ String uri(); /** * @return The path part of the uri. For example /somepath/somemorepath/someresource.foo */ @Nullable String path(); /** * @return the query part of the uri. For example someparam=32&someotherparam=x */ @Nullable String query(); /** * @return the request authority. For HTTP/2 the {@literal :authority} pseudo header is returned, for HTTP/1.x the * {@literal Host} header is returned or {@code null} when no such header is present. When the authority * string does not carry a port, the {@link HostAndPort#port()} returns {@code -1} to indicate the * scheme port is prevalent. */ @Nullable HostAndPort authority(); /** * @return the total number of bytes read for the body of the request. */ long bytesRead(); /** * @return the response. Each instance of this class has an {@link HttpServerResponse} instance attached to it. This is used * to send the response back to the client. */ @CacheReturn HttpServerResponse response(); /** * @return the headers in the request. */ @CacheReturn MultiMap headers(); /** * Return the first header value with the specified name * * @param headerName the header name * @return the header value */ @Nullable default String getHeader(String headerName) { return headers().get(headerName); } /** * Return the first header value with the specified name * * @param headerName the header name * @return the header value */ @GenIgnore(GenIgnore.PERMITTED_TYPE) default String getHeader(CharSequence headerName) { return headers().get(headerName); } /** * Override the charset to use for decoding the query parameter map, when none is set, {@code UTF8} is used. * * @param charset the charset to use for decoding query params * @return a reference to this, so the API can be used fluently */ @Fluent HttpServerRequest setParamsCharset(String charset); /** * @return the charset used for decoding query parameters */ String getParamsCharset(); /** * @return the query parameters in the request */ @CacheReturn default MultiMap params() { return params(false); } /** * @param semicolonIsNormalChar whether semicolon is treated as a normal character or a query parameter separator * @return the query parameters in the request */ MultiMap params(boolean semicolonIsNormalChar); /** * Return the first param value with the specified name * * @param paramName the param name * @return the param value */ @Nullable default String getParam(String paramName) { return params().get(paramName); } /** * Return the first param value with the specified name or {@code defaultValue} when the query param is not present * * @param paramName the param name * @param defaultValue the default value, must be non-null * @return the param value or {@code defaultValue} when not present */ default String getParam(String paramName, String defaultValue) { Objects.requireNonNull(defaultValue, "defaultValue"); final String paramValue = params().get(paramName); return paramValue != null ? paramValue : defaultValue; } /** * @return the remote address for this connection, possibly {@code null} (e.g a server bound on a domain socket). * If {@code useProxyProtocol} is set to {@code true}, the address returned will be of the actual connecting client. */ @CacheReturn default SocketAddress remoteAddress() { return connection().remoteAddress(); } /** * @return the local address for this connection, possibly {@code null} (e.g a server bound on a domain socket) * If {@code useProxyProtocol} is set to {@code true}, the address returned will be of the proxy. */ @CacheReturn default SocketAddress localAddress() { return connection().localAddress(); } /** * @return SSLSession associated with the underlying socket. Returns null if connection is * not SSL. * @see javax.net.ssl.SSLSession */ @GenIgnore(GenIgnore.PERMITTED_TYPE) default SSLSession sslSession() { return connection().sslSession(); } /** * @return the absolute URI corresponding to the HTTP request */ String absoluteURI(); /** * Convenience method for receiving the entire request body in one piece. *

* This saves the user having to manually setting a data and end handler and append the chunks of the body until * the whole body received. Don't use this if your request body is large - you could potentially run out of RAM. * * @param bodyHandler This handler will be called after all the body has been received */ @Fluent default HttpServerRequest bodyHandler(@Nullable Handler bodyHandler) { body().onSuccess(bodyHandler); return this; } /** * Convenience method for receiving the entire request body in one piece. *

* This saves you having to manually set a dataHandler and an endHandler and append the chunks of the body until * the whole body received. Don't use this if your request body is large - you could potentially run out of RAM. * * @return a future completed with the body result */ Future body(); /** * Returns a future signaling when the request has been fully received successfully or failed. * * @return a future completed with the body result */ Future end(); /** * Establish a TCP tunnel with the client. * *

This must be called only for {@code CONNECT} HTTP method or for HTTP connection upgrade, before any response is sent. * *

Calling this sends a {@code 200} response for a {@code CONNECT} or a {@code 101} for a connection upgrade wit * no {@code content-length} header set and then provides the {@code NetSocket} for handling the created tunnel. * Any HTTP header set on the response before calling this method will be sent. * *

   * server.requestHandler(req -> {
   *   if (req.method() == HttpMethod.CONNECT) {
   *     // Send a 200 response to accept the connect
   *     NetSocket socket = req.netSocket();
   *     socket.handler(buff -> {
   *       socket.write(buff);
   *     });
   *   }
   *   ...
   * });
   * 
* * @return a future notified with the upgraded socket */ Future toNetSocket(); /** * Call this with true if you are expecting a multi-part body to be submitted in the request. * This must be called before the body of the request has been received * * @param expect true - if you are expecting a multi-part body * @return a reference to this, so the API can be used fluently */ @Fluent HttpServerRequest setExpectMultipart(boolean expect); /** * @return true if we are expecting a multi-part body for this request. See {@link #setExpectMultipart}. */ boolean isExpectMultipart(); /** * Set an upload handler. The handler will get notified once a new file upload was received to allow you to deal * with the file upload. * * @return a reference to this, so the API can be used fluently */ @Fluent HttpServerRequest uploadHandler(@Nullable Handler uploadHandler); /** * Returns a map of all form attributes in the request. *

* Be aware that the attributes will only be available after the whole body has been received, i.e. after * the request end handler has been called. *

* {@link #setExpectMultipart(boolean)} must be called first before trying to get the form attributes. * * @return the form attributes */ @CacheReturn MultiMap formAttributes(); /** * Return the first form attribute value with the specified name * * @param attributeName the attribute name * @return the attribute value */ @Nullable String getFormAttribute(String attributeName); /** * @return the id of the stream of this request, {@literal -1} when it is not yet determined, i.e * the request has not been yet sent or it is not supported HTTP/1.x */ @CacheReturn default int streamId() { return -1; } /** * @return whether this request can be upgraded to a WebSocket, implying it uses HTTP/1.x and presents * the correct characteristics for a proper upgrade. */ default boolean canUpgradeToWebSocket() { return HttpUtils.canUpgradeToWebSocket(this); } /** * Upgrade the connection of the current request to a WebSocket. *

* This is an alternative way of handling WebSockets and can only be used if no WebSocket handler is set on the * {@code HttpServer}, and can only be used during the upgrade request during the WebSocket handshake. * *

Both {@link #handler(Handler)} and {@link #endHandler(Handler)} will be set to get the full body of the * request that is necessary to perform the WebSocket handshake. * *

If you need to do an asynchronous upgrade, i.e not performed immediately in your request handler, * you need to {@link #pause()} the request in order to not lose HTTP events necessary to upgrade the * request. * * @return a future notified with the upgraded WebSocket */ Future toWebSocket(); /** * Has the request ended? I.e. has the entire request, including the body been read? * * @return true if ended */ boolean isEnded(); /** * Set a custom frame handler. The handler will get notified when the http stream receives an custom HTTP/2 * frame. HTTP/2 permits extension of the protocol. * * @return a reference to this, so the API can be used fluently */ @Fluent HttpServerRequest customFrameHandler(Handler handler); /** * @return the {@link HttpConnection} associated with this request */ @CacheReturn HttpConnection connection(); /** * @return the priority of the associated HTTP/2 stream for HTTP/2 otherwise {@code null} */ default StreamPriority streamPriority() { return null; } /** * Set an handler for stream priority changes *

* This is not implemented for HTTP/1.x. * * @param handler the handler to be called when stream priority changes */ @Fluent HttpServerRequest streamPriorityHandler(Handler handler); /** * @return Netty's decoder result useful for handling invalid requests with {@link HttpServer#invalidRequestHandler} */ @GenIgnore DecoderResult decoderResult(); /** * Get the cookie with the specified name. * * NOTE: this will return just the 1st {@link Cookie} that matches the given name, to get all cookies for this name * see: {@link #cookies(String)} * * @param name the cookie name * @return the cookie or {@code null} if not found. */ @Nullable Cookie getCookie(String name); /** * Get the cookie with the specified {@code }. * * @param name the cookie name * @param domain the cookie domain * @param path the cookie path * @return the cookie or {@code null} if not found. */ @Nullable Cookie getCookie(String name, String domain, String path); /** * @return the number of cookies in the cookie jar. */ default int cookieCount() { return cookies().size(); } /** * @deprecated the implementation made a wrong assumption that cookies could be identified only by their name. The RFC * states that the tuple of {@code } is the unique identifier. * * When more than one cookie has the same name, the map will hold that lost one to be parsed and any previously parsed * value will be silently overwritten. * * @return a map of all the cookies. */ @Deprecated default Map cookieMap() { return cookies() .stream() .collect(Collectors.toMap(Cookie::getName, cookie -> cookie)); } /** * Returns a read only set of parsed cookies that match the given name, or an empty set. Several cookies may share the * same name but have different keys. A cookie is unique by its {@code } tuple. * * The set entries are references to the request original set. This means that performing property changes in the * cookie objects will affect the original object too. * * NOTE: the returned {@link Set} is read-only. This means any attempt to modify (add or remove to the set), will * throw {@link UnsupportedOperationException}. * * @param name the name to be matches * @return the matching cookies or empty set */ Set cookies(String name); /** * Returns a modifiable set of parsed cookies from the {@code COOKIE} header. Several cookies may share the * same name but have different keys. A cookie is unique by its {@code } tuple. * * Request cookies are directly linked to response cookies. Any modification to a cookie object in the returned set * will mark the cookie to be included in the HTTP response. Removing a cookie from the set, will also mean that it * will be removed from the response, regardless if it was modified or not. * * @return a set with all cookies in the cookie jar. */ Set cookies(); /** * Marks this request as being routed to the given route. This is purely informational and is * being provided to metrics. * * @param route The route this request has been routed to. */ @Fluent default HttpServerRequest routed(String route) { return this; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy