-
Notifications
You must be signed in to change notification settings - Fork 4.8k
/
http_integration.h
407 lines (349 loc) · 20.1 KB
/
http_integration.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
#pragma once
#include <cstdint>
#include <memory>
#include <string>
#include "envoy/extensions/early_data/v3/default_early_data_policy_descriptor.pb.h"
#include "envoy/extensions/filters/http/router/v3/router.pb.h"
#include "envoy/extensions/filters/http/upstream_codec/v3/upstream_codec.pb.h"
#include "source/common/http/codec_client.h"
#include "source/common/network/filter_impl.h"
#include "source/extensions/early_data/default_early_data_policy.h"
#include "test/common/http/http2/http2_frame.h"
#include "test/integration/integration.h"
#include "test/integration/utility.h"
#include "test/test_common/printers.h"
#include "test/test_common/utility.h"
#ifdef ENVOY_ENABLE_QUIC
#include "quiche/quic/core/deterministic_connection_id_generator.h"
#endif
namespace Envoy {
using ::Envoy::Http::Http2::Http2Frame;
enum class Http2Impl {
Nghttp2,
Oghttp2,
};
/**
* HTTP codec client used during integration testing.
*/
class IntegrationCodecClient : public Http::CodecClientProd {
public:
IntegrationCodecClient(Event::Dispatcher& dispatcher, Random::RandomGenerator& random,
Network::ClientConnectionPtr&& conn,
Upstream::HostDescriptionConstSharedPtr host_description,
Http::CodecType type);
IntegrationCodecClient(Event::Dispatcher& dispatcher, Random::RandomGenerator& random,
Network::ClientConnectionPtr&& conn,
Upstream::HostDescriptionConstSharedPtr host_description,
Http::CodecType type, bool wait_till_connected);
IntegrationStreamDecoderPtr makeHeaderOnlyRequest(const Http::RequestHeaderMap& headers);
IntegrationStreamDecoderPtr makeRequestWithBody(const Http::RequestHeaderMap& headers,
uint64_t body_size, bool end_stream = true);
IntegrationStreamDecoderPtr makeRequestWithBody(const Http::RequestHeaderMap& headers,
const std::string& body, bool end_stream = true);
bool sawGoAway() const { return saw_goaway_; }
bool connected() const { return connected_; }
bool streamOpen() const { return !stream_gone_; }
void sendData(Http::RequestEncoder& encoder, absl::string_view data, bool end_stream);
void sendData(Http::RequestEncoder& encoder, Buffer::Instance& data, bool end_stream);
void sendData(Http::RequestEncoder& encoder, uint64_t size, bool end_stream);
void sendTrailers(Http::RequestEncoder& encoder, const Http::RequestTrailerMap& trailers);
void sendReset(Http::RequestEncoder& encoder);
// Intentionally makes a copy of metadata_map.
void sendMetadata(Http::RequestEncoder& encoder, Http::MetadataMap metadata_map);
IntegrationStreamDecoderPtr initRequest(const Http::RequestHeaderMap& headers,
bool header_only_request = false);
std::pair<Http::RequestEncoder&, IntegrationStreamDecoderPtr>
startRequest(const Http::RequestHeaderMap& headers, bool header_only_request = false);
ABSL_MUST_USE_RESULT AssertionResult
waitForDisconnect(std::chrono::milliseconds time_to_wait = TestUtility::DefaultTimeout);
Network::ClientConnection* connection() const { return connection_.get(); }
Network::ConnectionEvent lastConnectionEvent() const { return last_connection_event_; }
Network::Connection& rawConnection() { return *connection_; }
bool disconnected() { return disconnected_; }
private:
struct ConnectionCallbacks : public Network::ConnectionCallbacks {
ConnectionCallbacks(IntegrationCodecClient& parent, bool block_till_connected)
: parent_(parent), block_till_connected_(block_till_connected) {}
// Network::ConnectionCallbacks
void onEvent(Network::ConnectionEvent event) override;
void onAboveWriteBufferHighWatermark() override {}
void onBelowWriteBufferLowWatermark() override {}
IntegrationCodecClient& parent_;
bool block_till_connected_;
};
struct CodecCallbacks : public Http::ConnectionCallbacks {
CodecCallbacks(IntegrationCodecClient& parent) : parent_(parent) {}
// Http::ConnectionCallbacks
void onGoAway(Http::GoAwayErrorCode) override { parent_.saw_goaway_ = true; }
IntegrationCodecClient& parent_;
};
struct CodecClientCallbacks : public Http::CodecClientCallbacks {
CodecClientCallbacks(IntegrationCodecClient& parent) : parent_(parent) {}
// Http::CodecClientCallbacks
void onStreamDestroy() override { parent_.stream_gone_ = true; }
void onStreamReset(Http::StreamResetReason) override { parent_.stream_gone_ = true; }
IntegrationCodecClient& parent_;
};
void flushWrite();
Event::Dispatcher& dispatcher_;
ConnectionCallbacks callbacks_;
CodecCallbacks codec_callbacks_;
CodecClientCallbacks codec_client_callbacks_;
bool connected_{};
bool disconnected_{};
bool saw_goaway_{};
bool stream_gone_{};
Network::ConnectionEvent last_connection_event_;
};
using IntegrationCodecClientPtr = std::unique_ptr<IntegrationCodecClient>;
/**
* Test fixture for HTTP and HTTP/2 integration tests.
*/
class HttpIntegrationTest : public BaseIntegrationTest {
public:
HttpIntegrationTest(Http::CodecType downstream_protocol, Network::Address::IpVersion version)
: HttpIntegrationTest(
downstream_protocol, version,
ConfigHelper::httpProxyConfig(/*downstream_use_quic=*/downstream_protocol ==
Http::CodecType::HTTP3)) {}
HttpIntegrationTest(Http::CodecType downstream_protocol, Network::Address::IpVersion version,
const std::string& config);
HttpIntegrationTest(Http::CodecType downstream_protocol,
const InstanceConstSharedPtrFn& upstream_address_fn,
Network::Address::IpVersion version)
: HttpIntegrationTest(
downstream_protocol, upstream_address_fn, version,
ConfigHelper::httpProxyConfig(/*downstream_use_quic=*/downstream_protocol ==
Http::CodecType::HTTP3)) {}
HttpIntegrationTest(Http::CodecType downstream_protocol,
const InstanceConstSharedPtrFn& upstream_address_fn,
Network::Address::IpVersion version, const std::string& config);
~HttpIntegrationTest() override;
void initialize() override;
void setupHttp1ImplOverrides(Http1ParserImpl http1_implementation);
void setupHttp2ImplOverrides(Http2Impl http2_implementation);
protected:
void useAccessLog(absl::string_view format = "",
std::vector<envoy::config::core::v3::TypedExtensionConfig> formatters = {});
std::string waitForAccessLog(const std::string& filename, uint32_t entry = 0,
bool allow_excess_entries = false,
Network::ClientConnection* client_connection = nullptr) {
if (client_connection == nullptr && codec_client_) {
client_connection = codec_client_->connection();
}
return BaseIntegrationTest::waitForAccessLog(filename, entry, allow_excess_entries,
client_connection);
};
IntegrationCodecClientPtr makeHttpConnection(uint32_t port);
// Makes a http connection object without checking its connected state.
virtual IntegrationCodecClientPtr makeRawHttpConnection(
Network::ClientConnectionPtr&& conn,
absl::optional<envoy::config::core::v3::Http2ProtocolOptions> http2_options,
absl::optional<envoy::config::core::v3::HttpProtocolOptions> common_http_options =
absl::nullopt,
bool wait_till_connected = true);
// Makes a downstream network connection object based on client codec version.
Network::ClientConnectionPtr makeClientConnectionWithOptions(
uint32_t port, const Network::ConnectionSocket::OptionsSharedPtr& options) override;
// Makes a http connection object with asserting a connected state.
IntegrationCodecClientPtr makeHttpConnection(Network::ClientConnectionPtr&& conn);
// Sets downstream_protocol_ and alters the HTTP connection manager codec type in the
// config_helper_.
void setDownstreamProtocol(Http::CodecType type);
// Enable the encoding/decoding of Http1 trailers downstream
ConfigHelper::HttpModifierFunction setEnableDownstreamTrailersHttp1();
// Enable the encoding/decoding of Http1 trailers upstream
ConfigHelper::ConfigModifierFunction setEnableUpstreamTrailersHttp1();
// Enable Proxy-Status response header.
ConfigHelper::HttpModifierFunction configureProxyStatus();
// Sends |request_headers| and |request_body_size| bytes of body upstream.
// Configured upstream to send |response_headers| and |response_body_size|
// bytes of body downstream.
// Waits |time| ms for both the request to be proxied upstream and the
// response to be proxied downstream.
//
// Waits for the complete downstream response before returning.
// Requires |codec_client_| to be initialized.
IntegrationStreamDecoderPtr sendRequestAndWaitForResponse(
const Http::TestRequestHeaderMapImpl& request_headers, uint32_t request_body_size,
const Http::TestResponseHeaderMapImpl& response_headers, uint32_t response_body_size,
uint64_t upstream_index = 0, std::chrono::milliseconds timeout = TestUtility::DefaultTimeout);
struct Result {
IntegrationStreamDecoderPtr response;
absl::optional<uint64_t> upstream_index;
};
Result sendRequestAndWaitForResponse(
const Http::TestRequestHeaderMapImpl& request_headers, uint32_t request_body_size,
const Http::TestResponseHeaderMapImpl& response_headers, uint32_t response_body_size,
const std::vector<uint64_t>& upstream_indices,
std::chrono::milliseconds timeout = TestUtility::DefaultTimeout);
// Wait for the end of stream on the next upstream stream on any of the provided fake upstreams.
// Sets fake_upstream_connection_ to the connection and upstream_request_ to stream.
// In cases where the upstream that will receive the request is not deterministic, a second
// upstream index may be provided, in which case both upstreams will be checked for requests.
absl::optional<uint64_t> waitForNextUpstreamRequest(
const std::vector<uint64_t>& upstream_indices,
std::chrono::milliseconds connection_wait_timeout = TestUtility::DefaultTimeout);
void waitForNextUpstreamRequest(
uint64_t upstream_index = 0,
std::chrono::milliseconds connection_wait_timeout = TestUtility::DefaultTimeout);
absl::optional<uint64_t>
waitForNextUpstreamConnection(const std::vector<uint64_t>& upstream_indices,
std::chrono::milliseconds connection_wait_timeout,
FakeHttpConnectionPtr& fake_upstream_connection);
// Close |codec_client_| and |fake_upstream_connection_| cleanly.
void cleanupUpstreamAndDownstream();
// Verifies the response_headers contains the expected_headers, and response body matches given
// body string.
void verifyResponse(IntegrationStreamDecoderPtr response, const std::string& response_code,
const Http::TestResponseHeaderMapImpl& expected_headers,
const std::string& expected_body);
// Helper that sends a request to Envoy, and verifies if Envoy response headers and body size is
// the same as the expected headers map.
// Requires the "http" port has been registered.
void sendRequestAndVerifyResponse(const Http::TestRequestHeaderMapImpl& request_headers,
const int request_size,
const Http::TestResponseHeaderMapImpl& response_headers,
const int response_size, const int backend_idx,
absl::optional<const Http::TestResponseHeaderMapImpl>
expected_response_headers = absl::nullopt);
// Check for completion of upstream_request_, and a simple "200" response.
void checkSimpleRequestSuccess(uint64_t expected_request_size, uint64_t expected_response_size,
IntegrationStreamDecoder* response);
using ConnectionCreationFunction = std::function<Network::ClientConnectionPtr()>;
// Sends a simple header-only HTTP request, and waits for a response.
IntegrationStreamDecoderPtr makeHeaderOnlyRequest(ConnectionCreationFunction* create_connection,
int upstream_index,
const std::string& path = "/test/long/url",
const std::string& overwrite_authority = "");
void testRouterNotFound();
void testRouterNotFoundWithBody();
void testRouterVirtualClusters();
void testRouteStats();
void testRouterUpstreamProtocolError(const std::string&, const std::string&);
void testRouterRequestAndResponseWithBody(
uint64_t request_size, uint64_t response_size, bool big_header,
bool set_content_length_header = false, ConnectionCreationFunction* creator = nullptr,
std::chrono::milliseconds timeout = TestUtility::DefaultTimeout);
void testRouterHeaderOnlyRequestAndResponse(ConnectionCreationFunction* creator = nullptr,
int upstream_index = 0,
const std::string& path = "/test/long/url",
const std::string& overwrite_authority = "");
// Disconnect tests
void testRouterUpstreamDisconnectBeforeRequestComplete();
void
testRouterUpstreamDisconnectBeforeResponseComplete(ConnectionCreationFunction* creator = nullptr);
void testRouterDownstreamDisconnectBeforeRequestComplete(
ConnectionCreationFunction* creator = nullptr);
void testRouterDownstreamDisconnectBeforeResponseComplete(
ConnectionCreationFunction* creator = nullptr);
void testRouterUpstreamResponseBeforeRequestComplete(uint32_t status_code = 0);
void testTwoRequests(bool force_network_backup = false);
void testLargeRequestUrl(uint32_t url_size, uint32_t max_headers_size);
void testLargeRequestHeaders(uint32_t size, uint32_t count, uint32_t max_size = 60,
uint32_t max_count = 100,
std::chrono::milliseconds timeout = TestUtility::DefaultTimeout);
void testLargeResponseHeaders(uint32_t size, uint32_t count, uint32_t max_size = 60,
uint32_t max_count = 100,
std::chrono::milliseconds timeout = TestUtility::DefaultTimeout);
void testLargeRequestTrailers(uint32_t size, uint32_t max_size = 60);
void testManyRequestHeaders(std::chrono::milliseconds time = TestUtility::DefaultTimeout);
void testAddEncodedTrailers();
void testRetry();
void testRouterRetryOnResetBeforeRequestBeforeHeaders();
void testRouterRetryOnResetBeforeRequestAfterHeaders();
void testRetryHittingBufferLimit();
void testRetryAttemptCountHeader();
void testGrpcRetry();
void testEnvoyHandling1xx(bool additional_continue_from_upstream = false,
const std::string& via = "", bool disconnect_after_100 = false);
void testEnvoyProxying1xx(bool continue_before_upstream_complete = false,
bool with_encoder_filter = false,
bool with_multiple_1xx_headers = false,
absl::string_view initial_code = "100");
void simultaneousRequest(uint32_t request1_bytes, uint32_t request2_bytes,
uint32_t response1_bytes, uint32_t response2_bytes);
// HTTP/2 client tests.
void testDownstreamResetBeforeResponseComplete();
// Test that trailers are sent. request_trailers_present and
// response_trailers_present will check if the trailers are present, otherwise
// makes sure they were dropped.
void testTrailers(uint64_t request_size, uint64_t response_size, bool request_trailers_present,
bool response_trailers_present);
// Test /drain_listener from admin portal.
void testAdminDrain(Http::CodecClient::Type admin_request_type);
// Test sending and receiving large request and response bodies with autonomous upstream.
void testGiantRequestAndResponse(uint64_t request_size, uint64_t response_size,
bool set_content_length_header,
std::chrono::milliseconds timeout = 2 *
TestUtility::DefaultTimeout);
struct BytesCountExpectation {
BytesCountExpectation(int wire_bytes_sent, int wire_bytes_received, int header_bytes_sent,
int header_bytes_received)
: wire_bytes_sent_{wire_bytes_sent}, wire_bytes_received_{wire_bytes_received},
header_bytes_sent_{header_bytes_sent}, header_bytes_received_{header_bytes_received} {}
int wire_bytes_sent_;
int wire_bytes_received_;
int header_bytes_sent_;
int header_bytes_received_;
};
void expectUpstreamBytesSentAndReceived(BytesCountExpectation h1_expectation,
BytesCountExpectation h2_expectation,
BytesCountExpectation h3_expectation, const int id = 0);
void expectDownstreamBytesSentAndReceived(BytesCountExpectation h1_expectation,
BytesCountExpectation h2_expectation,
BytesCountExpectation h3_expectation, const int id = 0);
Http::CodecClient::Type downstreamProtocol() const { return downstream_protocol_; }
std::string downstreamProtocolStatsRoot() const;
// Return the upstream protocol part of the stats root.
std::string upstreamProtocolStatsRoot() const;
// Prefix listener stat with IP:port, including IP version dependent loopback address.
std::string listenerStatPrefix(const std::string& stat_name);
Network::UpstreamTransportSocketFactoryPtr quic_transport_socket_factory_;
// Must outlive |codec_client_| because it may not close connection till the end of its life
// scope.
std::unique_ptr<Http::PersistentQuicInfo> quic_connection_persistent_info_;
// The client making requests to Envoy.
IntegrationCodecClientPtr codec_client_;
// A placeholder for the first upstream connection.
FakeHttpConnectionPtr fake_upstream_connection_;
// A placeholder for the first request received at upstream.
FakeStreamPtr upstream_request_;
// A pointer to the request encoder, if used.
Http::RequestEncoder* request_encoder_{nullptr};
// The response headers sent by sendRequestAndWaitForResponse() by default.
Http::TestResponseHeaderMapImpl default_response_headers_{{":status", "200"}};
Http::TestRequestHeaderMapImpl default_request_headers_{{":method", "GET"},
{":path", "/test/long/url"},
{":scheme", "http"},
{":authority", "sni.lyft.com"}};
// The codec type for the client-to-Envoy connection
Http::CodecType downstream_protocol_{Http::CodecType::HTTP1};
std::string access_log_name_;
testing::NiceMock<Random::MockRandomGenerator> random_;
Quic::QuicStatNames quic_stat_names_;
std::string san_to_match_{"spiffe://lyft.com/backend-team"};
// Set this to true when sending malformed requests to avoid test client codec rejecting it.
// This flag is only valid when UHV build flag is enabled.
bool disable_client_header_validation_{false};
#ifdef ENVOY_ENABLE_QUIC
quic::DeterministicConnectionIdGenerator connection_id_generator_{
quic::kQuicDefaultConnectionIdLength};
#endif
};
// Helper class for integration tests using raw HTTP/2 frames
class Http2RawFrameIntegrationTest : public HttpIntegrationTest {
public:
Http2RawFrameIntegrationTest(Network::Address::IpVersion version)
: HttpIntegrationTest(Http::CodecType::HTTP2, version) {}
protected:
void startHttp2Session();
Http2Frame readFrame();
void sendFrame(const Http2Frame& frame);
virtual void beginSession();
IntegrationTcpClientPtr tcp_client_;
};
absl::string_view upstreamToString(Http::CodecType type);
absl::string_view downstreamToString(Http::CodecType type);
absl::string_view http2ImplementationToString(Http2Impl impl);
} // namespace Envoy