Envoy provides a way for users to communicate extra information associated with a stream that is not carried in the standard HTTP(s) and HTTP/2 headers and payloads. The information is represented by a string key-value pair. For example, users can pass along RTT information associated with a stream using a key of "rtt info", and a value of "100ms". In Envoy, we call this type of information metadata. A stream can be associated with multiple metadata, and the multiple metadata are represented by a map.
Note: the metadata implementation is still in progress, and the doc is in draft version.
For ease of implementation and compatibility purposes, metadata will only be supported in HTTP/2. Metadata sent in any other protocol should result in protocol errors or be ignored.
To simplify the implementation, we don't allow metadata frames to carry the END_STREAM flag. Because metadata frames must be associated with an existing stream, users must ensure that the peer receives metadata frames before a frame carrying the END_STREAM flag.
Metadata associated with a stream can be sent before HEADERS, after HEADERS, between DATA, or after DATA. If metadata frames have to be sent last, users must put the END_STREAM flag in an empty DATA frame and send the empty DATA frame after metadata frames.
Envoy only allows up to 1M metadata to be sent per stream. If the accumulated metadata size exceeds the limit at the sender, some metadata will be dropped. If the limit is exceeded at the receiver, the connection will fail.
Envoy provides the functionality to proxy, process, and add metadata.
If not specified, all the metadata received by Envoy is proxied to the next hop unmodified. Note that we do not guarantee the same frame order will be preserved from hop by hop. That is, metadata from upstream at the beginning of a stream can be received by the downstream at the end of the stream.
If Envoy needs to take actions when a metadata frame is received, users should create a new filter.
If Envoy needs to parse metadata sent on a request from downstream to upstream, a StreamDecoderFilter should be created. The interface to override is:
FilterMetadataStatus StreamDecoderFilter::decodeMetadata(MetadataMap& metadata_map);
The metadata passed in is a map of the metadata associated with the request
stream. After metadata has been parsed, the filter can choose to add additional
metadata, remove metadata, or keep it untouched by modifying the passed in
metadata_map
directly.
If Envoy needs to parse metadata sent on a response from upstream to downstream, a StreamEncoderFilter should be created. The interface to override is:
FilterMetadataStatus StreamEncoderFilter::encodeMetadata(MetadataMap& metadata);
The metadata passed in is a map of the metadata associated with the response
stream. After metadata has been parsed, the filter can choose to add additional
metadata, remove metadata, or keep it untouched by modifying the passed in
metadata_map
directly.
Note that if the metadata in a request or a response is removed from the map after consuming, the metadata will not be passed to the next hop. An empty map means no metadata will be sent to the next hop. If the metadata is left in the map, it will be passed to the next hop.
Envoy filters can be used to add new metadata to a stream.
If users need to add new metadata for a request from downstream to upstream, a StreamDecoderFilter should be created. The StreamDecoderFilterCallbacks object that Envoy passes to the StreamDecoderFilter has an interface MetadataMapVector& StreamDecoderFilterCallbacks::addDecodedMetadata(). By calling the interface, users get a reference to a vector of metadata maps associated with the request stream. Users can insert new metadata maps to the metadata map vector, and Envoy will proxy the new metadata map to the upstream. StreamDecoderFilterCallbacks::addDecodedMetadata() can be called in the StreamDecoderFilter::decode*() methods except for decodeMetadata(): StreamDecoderFilter::decodeHeaders(), StreamDecoderFilter::decodeData(), StreamDecoderFilter::decodeTrailers(), and StreamDecoderFilter::decodeComplete().
Do not call StreamDecoderFilterCallbacks::addDecodedMetadata() in
StreamDecoderFilter::decodeMetadata(MetadataMap& metadata_map). Instead, new
metadata can be added directly to metadata\_map
.
If users need to add new metadata for a response from upstream to downstream, a StreamEncoderFilter should be created. The StreamEncoderFilterCallbacks object that Envoy passes to the StreamEncoderFilter has an interface StreamEncoderFilterCallbacks::addEncodedMetadata(MetadataMapPtr&& metadata_map_ptr). By calling the interface, users can directly pass in the metadata to be added. This function can be called in the StreamEncoderFilter::encode*() methods except for encodeMetadata(): StreamEncoderFilter::encode1xxHeaders(HeaderMap& headers), StreamEncoderFilter::encodeHeaders(HeaderMap& headers, bool end_stream), StreamEncoderFilter::encodeData(Buffer::Instance& data, bool end_stream), StreamEncoderFilter::encodeTrailers(HeaderMap& trailers), and StreamEncoderFilter::encodeComplete(). Consequently, the new metadata will be passed through all the encoding filters that follow the filter where the new metadata is added.
Do not call StreamEncoderFilterCallbacks::addEncodedMetadata() in
StreamEncoderFilter::encodeMetadata(MetadataMap& metadata_map). Instead, if
users receive metadata from upstream, new metadata can be added directly to the
input argument metadata\_map
in
StreamEncoderFilter::encodeMetadata(MetadataMap& metadata_map).
Envoy supports metadata by utilizing nghttp2 extension frames. Envoy defines a new extension frame type METADATA frame in nghttp2:
type = 0x4D
The METADATA frame uses a standard frame header, as described in the HTTP/2 spec The payload of the METADATA frame is a block of key-value pairs encoded using the HPACK Literal Header Field Never Indexed representation. Each key-value pair represents one piece of metadata.
The METADATA frame defines the following flags:
END_METADATA (0x4).
If the flag is set, it indicates that this frame ends a metadata payload.
The METADATA frame payload is not subject to HTTP/2 flow control, but the size of the payload is bounded by the maximum frame size negotiated in SETTINGS. There are no restrictions on the set of octets that may be used in keys or values.
We do not allow a METADATA frame to terminate a stream. DATA, HEADERS or RST_STREAM must be used for that purpose.
We call metadata that need to be forwarded to downstream the response metadata. Response metadata can be received from upstream or generated locally.
Response metadata is generally a hop by hop message, so Envoy doesn't need to hold response metadata locally to wait for some events or data. As a result, filters handling response metadata don't need to stop the filter iteration and wait. Instead response metadata can be forwarded through targeted filters and sequentially to the next hop as soon as they are available, no matter if the metadata are locally generated or received from upstream. The same statement is also true for metadata from downstream to upstream (request metadata). However, request metadata may need to wait for the upstream connection to be ready before going to the next hop. In this section, we focus on response metadata handling.
We first explain how response metadata gets consumed or proxied. In function FilterManager::encodeMetadata(ActiveStreamEncoderFilter* filter, MetadataMapPtr&& metadata_map_ptr), Envoy passes response metadata received from upstream to filters by calling the following filter interface:
FilterMetadatasStatus StreamEncoderFilter::encodeMetadata(MetadataMapVector& metadata_map).
Filters, by implementing the interface, can consume response metadata. After going through the filter chain, the function FilterManager::encodeMetadata(ActiveStreamEncoderFilter* filter, MetadataMapPtr&& metadata_map_ptr) immediately forwards the updated or remaining response metadata to the next hop by calling a metadata-encoding function that calls the codec's encoding:
ConnectionManagerImpl::ActiveStream::encodeMetadata(MetadataMapVector& metadata).
If no filter consumes the response metadata, the response metadata is proxied to the downstream untouched.
Envoy can also add new response metadata through filters's encoding interfaces
(See section Inserting metadata for detailed interfaces).
Filters can add new metadata by calling
StreamEncoderFilterCallbacks::addEncodedMetadata(MetadataMapPtr&&
metadata_map_ptr), which triggers
FilterManager::encodeMetadata(ActiveStreamEncoderFilter* filter,
MetadataMapPtr&& metadata_map_ptr) to go through all the encoding filters. Or
new metadata can be added to metadata\_map
in
StreamEncoderFilter::encodeMetadata(MetadataMap& metadata_map) directly.
We first explain how request metadata gets consumed or proxied. In function FilterManager::decodeMetadata(ActiveStreamDecoderFilter* filter, MetadataMap& metadata_map), Envoy passes request metadata received from downstream to filters by calling the following filter interface:
FilterMetadataStatus StreamDecoderFilter::decodeMetadata(MetadataMap& metadata_map).
Filters, by implementing the interface, can consume or modify request metadata. If no filter touches the metadata, it is proxied to upstream unchanged.
The last filter in the filter chain is router filter. The router filter calls UpstreamRequest::encodeMetadata(const MetadataMapVector& metadata_map_vector) to pass the metadata to the codec, and the codec encodes and forwards the metadata to the upstream. If the connection to the upstream has not been established when metadata is received, the metadata is temporarily stored in UpstreamRequest::downstream_metadata_map_vector_. When the connection is ready (UpstreamRequest::onPoolReady()), the metadata is then passed to the codec and forwarded to the upstream.
Envoy can also add new request metadata through filters's decoding interfaces (See section Inserting metadata for detailed interfaces). Filters can add new metadata to FilterManager::request_metadata_map_vector_ by calling StreamDecoderFilterCallbacks::addDecodedMetadata(). After calling each filter's decoding function from StreamDecoderFilter, Envoy checks if new metadata is added to FilterManager::request_metadata_map_vector_ via FilterManager::processNewlyAddedMetadata(). If so, then Envoy calls FilterManager::decodeMetadata(ActiveStreamEncoderFilter* filter, MetadataMapPtr&& metadata_map) to go through all the filters.
Note that because METADATA frames do not carry END_STREAM, if new metadata is added to a headers only request, Envoy moves END_STREAM from HEADERS to an empty DATA frame that is sent after the new metadata. In addition, Envoy drains metadata in the router filter before any other types of frames except HEADERS to make sure END_STREAM is handled correctly.