blob: f00ac6109a1947467607dea083993425ab598723 [file] [log] [blame]
#ifndef QUICHE_HTTP2_ADAPTER_HTTP2_VISITOR_INTERFACE_H_
#define QUICHE_HTTP2_ADAPTER_HTTP2_VISITOR_INTERFACE_H_
#include <cstdint>
#include <vector>
#include "absl/strings/string_view.h"
#include "quiche/http2/adapter/http2_protocol.h"
#include "quiche/common/platform/api/quiche_export.h"
#include "quiche/common/platform/api/quiche_logging.h"
namespace http2 {
namespace adapter {
// Http2VisitorInterface contains callbacks for receiving HTTP/2-level events. A
// processor like NghttpAdapter parses HTTP/2 frames and invokes the callbacks
// on an instance of this interface. Prefer a void return type for these
// callbacks, instead setting output parameters as needed.
//
// Example sequences of calls/events:
// GET:
// - OnBeginHeadersForStream()
// - OnHeaderForStream()
// - OnEndHeadersForStream()
// - OnEndStream()
//
// POST:
// - OnBeginHeadersForStream()
// - OnHeaderForStream()
// - OnEndHeadersForStream()
// - OnBeginDataForStream()
// - OnDataForStream()
// - OnEndStream()
//
// Request canceled mid-stream, e.g, with error code CANCEL:
// - OnBeginHeadersForStream()
// - OnHeaderForStream()
// - OnEndHeadersForStream()
// - OnRstStream()
// - OnCloseStream()
//
// Request closed mid-stream, e.g., with error code NO_ERROR:
// - OnBeginHeadersForStream()
// - OnHeaderForStream()
// - OnEndHeadersForStream()
// - OnRstStream()
// - OnCloseStream()
//
// More details are at RFC 7540 (go/http2spec).
class QUICHE_EXPORT Http2VisitorInterface {
public:
Http2VisitorInterface(const Http2VisitorInterface&) = delete;
Http2VisitorInterface& operator=(const Http2VisitorInterface&) = delete;
virtual ~Http2VisitorInterface() = default;
enum : int64_t {
kSendBlocked = 0,
kSendError = -1,
};
// Called when there are serialized frames to send. Should return how many
// bytes were actually sent. May return kSendBlocked or kSendError.
virtual int64_t OnReadyToSend(absl::string_view serialized) = 0;
struct DataFrameHeaderInfo {
int64_t payload_length;
bool end_data;
bool end_stream; // If true, also implies end_data.
};
// Called when the codec is ready to construct a DATA frame header. The
// implementation should return the number of bytes ready to send, and whether
// it's the end of the message body data. If the implementation returns 0
// bytes and also `end_data` is false, then the stream is deferred until
// resumed by the application. `payload_length` must be at most `max_length`.
// A `payload_length` of -1 indicates that this stream has encountered an
// unrecoverable error.
virtual DataFrameHeaderInfo OnReadyToSendDataForStream(
Http2StreamId /*stream_id*/, size_t /*max_length*/) {
QUICHE_LOG(FATAL) << "Not implemented";
return DataFrameHeaderInfo{};
}
// Called when the codec is ready to send a DATA frame. The implementation
// should send the `frame_header` and specified number of payload bytes.
// Returning false indicates an unrecoverable error.
virtual bool SendDataFrame(Http2StreamId /*stream_id*/,
absl::string_view /*frame_header*/,
size_t /*payload_bytes*/) {
QUICHE_LOG(FATAL) << "Not implemented.";
return false;
}
// Called when a connection-level error has occurred.
enum class ConnectionError {
// The peer sent an invalid connection preface.
kInvalidConnectionPreface,
// The visitor encountered an error sending bytes to the peer.
kSendError,
// There was an error reading and framing bytes from the peer.
kParseError,
// The visitor considered a received header to be a connection error.
kHeaderError,
// The peer attempted to open a stream with an invalid stream ID.
kInvalidNewStreamId,
// The peer sent a frame that is invalid on an idle stream (before HEADERS).
kWrongFrameSequence,
// The peer sent an invalid PUSH_PROMISE frame.
kInvalidPushPromise,
// The peer exceeded the max concurrent streams limit.
kExceededMaxConcurrentStreams,
// The peer caused a flow control error.
kFlowControlError,
// The peer sent a GOAWAY with an invalid last-stream-ID field.
kInvalidGoAwayLastStreamId,
// The peer sent an invalid SETTINGS value.
kInvalidSetting,
};
virtual void OnConnectionError(ConnectionError error) = 0;
// Called when the header for a frame is received. Returns false if a fatal
// error has occurred.
virtual bool OnFrameHeader(Http2StreamId /*stream_id*/, size_t /*length*/,
uint8_t /*type*/, uint8_t /*flags*/) {
return true;
}
// Called when a non-ack SETTINGS frame is received.
virtual void OnSettingsStart() = 0;
// Called for each SETTINGS id-value pair.
virtual void OnSetting(Http2Setting setting) = 0;
// Called at the end of a non-ack SETTINGS frame.
virtual void OnSettingsEnd() = 0;
// Called when a SETTINGS ack frame is received.
virtual void OnSettingsAck() = 0;
// Called when the connection receives the header block for a HEADERS frame on
// a stream but has not yet parsed individual headers. Returns false if a
// fatal error has occurred.
virtual bool OnBeginHeadersForStream(Http2StreamId stream_id) = 0;
// Called when the connection receives the header |key| and |value| for a
// stream. The HTTP/2 pseudo-headers defined in RFC 7540 Sections 8.1.2.3 and
// 8.1.2.4 are also conveyed in this callback. This method is called after
// OnBeginHeadersForStream(). May return HEADER_RST_STREAM to indicate the
// header block should be rejected. This will cause the library to queue a
// RST_STREAM frame, which will have a default error code of INTERNAL_ERROR.
// The visitor implementation may choose to queue a RST_STREAM with a
// different error code instead, which should be done before returning
// HEADER_RST_STREAM. Returning HEADER_CONNECTION_ERROR will lead to a
// non-recoverable error on the connection.
enum class OnHeaderResult {
// The header was accepted.
HEADER_OK,
// The application considers the header a connection error.
HEADER_CONNECTION_ERROR,
// The application rejects the header and requests the stream be reset.
HEADER_RST_STREAM,
// The header field is invalid and will be reset with error code
// PROTOCOL_ERROR.
HEADER_FIELD_INVALID,
// The headers are a violation of HTTP messaging semantics and will be reset
// with error code PROTOCOL_ERROR.
HEADER_HTTP_MESSAGING,
// The headers caused a compression context error.
HEADER_COMPRESSION_ERROR,
};
virtual OnHeaderResult OnHeaderForStream(Http2StreamId stream_id,
absl::string_view key,
absl::string_view value) = 0;
// Called when the connection has received the complete header block for a
// logical HEADERS frame on a stream (which may contain CONTINUATION frames,
// transparent to the user). Returns false if a fatal error has occurred.
virtual bool OnEndHeadersForStream(Http2StreamId stream_id) = 0;
// Called when the connection receives the beginning of a DATA frame. The data
// payload will be provided via subsequent calls to OnDataForStream(). Returns
// false if a fatal error has occurred.
virtual bool OnBeginDataForStream(Http2StreamId stream_id,
size_t payload_length) = 0;
// Called when the optional padding length field is parsed as part of a DATA
// frame payload. `padding_length` represents the total amount of padding for
// this frame, including the length byte itself. Returns false if a fatal
// error has occurred.
virtual bool OnDataPaddingLength(Http2StreamId stream_id,
size_t padding_length) = 0;
// Called when the connection receives some |data| (as part of a DATA frame
// payload) for a stream. Returns false if a fatal error has occurred.
virtual bool OnDataForStream(Http2StreamId stream_id,
absl::string_view data) = 0;
// Called when the peer sends the END_STREAM flag on a stream, indicating that
// the peer will not send additional headers or data for that stream.
virtual bool OnEndStream(Http2StreamId stream_id) = 0;
// Called when the connection receives a RST_STREAM for a stream. This call
// will be followed by either OnCloseStream().
virtual void OnRstStream(Http2StreamId stream_id,
Http2ErrorCode error_code) = 0;
// Called when a stream is closed. Returns false if a fatal error has
// occurred.
virtual bool OnCloseStream(Http2StreamId stream_id,
Http2ErrorCode error_code) = 0;
// Called when the connection receives a PRIORITY frame.
virtual void OnPriorityForStream(Http2StreamId stream_id,
Http2StreamId parent_stream_id, int weight,
bool exclusive) = 0;
// Called when the connection receives a PING frame.
virtual void OnPing(Http2PingId ping_id, bool is_ack) = 0;
// Called when the connection receives a PUSH_PROMISE frame. The server push
// request headers follow in calls to OnHeaderForStream() with |stream_id|.
virtual void OnPushPromiseForStream(Http2StreamId stream_id,
Http2StreamId promised_stream_id) = 0;
// Called when the connection receives a GOAWAY frame. Returns false if a
// fatal error has occurred.
virtual bool OnGoAway(Http2StreamId last_accepted_stream_id,
Http2ErrorCode error_code,
absl::string_view opaque_data) = 0;
// Called when the connection receives a WINDOW_UPDATE frame. For
// connection-level window updates, the |stream_id| will be 0.
virtual void OnWindowUpdate(Http2StreamId stream_id,
int window_increment) = 0;
// Called immediately before a frame of the given type is sent. Should return
// 0 on success.
virtual int OnBeforeFrameSent(uint8_t frame_type, Http2StreamId stream_id,
size_t length, uint8_t flags) = 0;
// Called immediately after a frame of the given type is sent. Should return 0
// on success. |error_code| is only populated for RST_STREAM and GOAWAY frame
// types.
virtual int OnFrameSent(uint8_t frame_type, Http2StreamId stream_id,
size_t length, uint8_t flags,
uint32_t error_code) = 0;
// Called when the connection receives an invalid frame. A return value of
// false will result in the connection entering an error state, with no
// further frame processing possible.
enum class InvalidFrameError {
// The frame contains a general protocol error.
kProtocol,
// The frame would have caused a new (invalid) stream to be opened.
kRefusedStream,
// The frame contains an invalid header field.
kHttpHeader,
// The frame contains a violation in HTTP messaging rules.
kHttpMessaging,
// The frame causes a flow control error.
kFlowControl,
// The frame is on an already closed stream or has an invalid stream ID.
kStreamClosed,
};
virtual bool OnInvalidFrame(Http2StreamId stream_id,
InvalidFrameError error) = 0;
// Called when the connection receives the beginning of a METADATA frame
// (which may itself be the middle of a logical metadata block). The metadata
// payload will be provided via subsequent calls to OnMetadataForStream().
// TODO(birenroy): Consider removing this unnecessary method.
virtual void OnBeginMetadataForStream(Http2StreamId stream_id,
size_t payload_length) = 0;
// Called when the connection receives |metadata| as part of a METADATA frame
// payload for a stream. Returns false if a fatal error has occurred.
virtual bool OnMetadataForStream(Http2StreamId stream_id,
absl::string_view metadata) = 0;
// Called when the connection has finished receiving a logical metadata block
// for a stream. Note that there may be multiple metadata blocks for a stream.
// Returns false if there was an error unpacking the metadata payload.
virtual bool OnMetadataEndForStream(Http2StreamId stream_id) = 0;
// Called when the connection is ready to send a metadata payload for a
// stream. Should return the number of payload bytes copied to |dest|, or a
// negative integer to indicate an error, as well as a boolean indicating
// whether the metadata payload has been completely copied.
virtual std::pair<int64_t, bool> PackMetadataForStream(
Http2StreamId /*stream_id*/, uint8_t* /*dest*/, size_t /*dest_len*/) {
QUICHE_LOG(DFATAL) << "Not implemented";
return {-1, false};
}
// Invoked with an error message from the application.
virtual void OnErrorDebug(absl::string_view message) = 0;
protected:
Http2VisitorInterface() = default;
};
} // namespace adapter
} // namespace http2
#endif // QUICHE_HTTP2_ADAPTER_HTTP2_VISITOR_INTERFACE_H_