forked from envoyproxy/envoy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
integration_stream_decoder.cc
142 lines (122 loc) · 4.24 KB
/
integration_stream_decoder.cc
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
#include "test/integration/integration_stream_decoder.h"
#include <algorithm>
#include <cstdint>
#include <memory>
#include <string>
#include <utility>
#include "envoy/buffer/buffer.h"
#include "envoy/event/dispatcher.h"
#include "envoy/http/header_map.h"
#include "source/common/common/assert.h"
#include "gmock/gmock.h"
#include "gtest/gtest.h"
using testing::AssertionFailure;
using testing::AssertionResult;
using testing::AssertionSuccess;
namespace Envoy {
IntegrationStreamDecoder::IntegrationStreamDecoder(Event::Dispatcher& dispatcher)
: dispatcher_(dispatcher) {}
void IntegrationStreamDecoder::waitFor1xxHeaders() {
if (!continue_headers_.get()) {
waiting_for_continue_headers_ = true;
dispatcher_.run(Event::Dispatcher::RunType::Block);
}
}
void IntegrationStreamDecoder::waitForHeaders() {
if (!headers_.get()) {
waiting_for_headers_ = true;
dispatcher_.run(Event::Dispatcher::RunType::Block);
}
}
void IntegrationStreamDecoder::waitForBodyData(uint64_t size) {
ASSERT(body_data_waiting_length_ == 0);
body_data_waiting_length_ = size;
body_data_waiting_length_ -=
std::min(body_data_waiting_length_, static_cast<uint64_t>(body_.size()));
if (body_data_waiting_length_ > 0) {
dispatcher_.run(Event::Dispatcher::RunType::Block);
}
}
AssertionResult IntegrationStreamDecoder::waitForEndStream(std::chrono::milliseconds timeout) {
bool timer_fired = false;
if (!saw_end_stream_) {
Event::TimerPtr timer(dispatcher_.createTimer([this, &timer_fired]() -> void {
timer_fired = true;
dispatcher_.exit();
}));
timer->enableTimer(timeout);
waiting_for_end_stream_ = true;
dispatcher_.run(Event::Dispatcher::RunType::Block);
if (!saw_end_stream_) {
// HTTP/1.1 may end stream by disconnecting.
ENVOY_LOG_MISC(warn, "No explicit end stream detected.");
}
if (timer_fired) {
return AssertionFailure() << "Timed out waiting for end stream\n";
}
}
return AssertionSuccess();
}
AssertionResult IntegrationStreamDecoder::waitForReset(std::chrono::milliseconds timeout) {
if (!saw_reset_) {
// Set a timer to stop the dispatcher if the timeout has been exceeded.
Event::TimerPtr timer(dispatcher_.createTimer([this]() -> void { dispatcher_.exit(); }));
timer->enableTimer(timeout);
waiting_for_reset_ = true;
dispatcher_.run(Event::Dispatcher::RunType::Block);
// If the timer has fired, this timed out before a reset was received.
if (!timer->enabled()) {
return AssertionFailure() << "Timed out waiting for reset.";
}
}
return AssertionSuccess();
}
void IntegrationStreamDecoder::decode1xxHeaders(Http::ResponseHeaderMapPtr&& headers) {
continue_headers_ = std::move(headers);
if (waiting_for_continue_headers_) {
dispatcher_.exit();
}
}
void IntegrationStreamDecoder::decodeHeaders(Http::ResponseHeaderMapPtr&& headers,
bool end_stream) {
saw_end_stream_ = end_stream;
headers_ = std::move(headers);
if ((end_stream && waiting_for_end_stream_) || waiting_for_headers_) {
dispatcher_.exit();
}
}
void IntegrationStreamDecoder::decodeData(Buffer::Instance& data, bool end_stream) {
saw_end_stream_ = end_stream;
body_ += data.toString();
if (end_stream && waiting_for_end_stream_) {
dispatcher_.exit();
} else if (body_data_waiting_length_ > 0) {
body_data_waiting_length_ -= std::min(body_data_waiting_length_, data.length());
if (body_data_waiting_length_ == 0) {
dispatcher_.exit();
}
}
}
void IntegrationStreamDecoder::decodeTrailers(Http::ResponseTrailerMapPtr&& trailers) {
saw_end_stream_ = true;
trailers_ = std::move(trailers);
if (waiting_for_end_stream_) {
dispatcher_.exit();
}
}
void IntegrationStreamDecoder::decodeMetadata(Http::MetadataMapPtr&& metadata_map) {
metadata_maps_decoded_count_++;
// Combines newly received metadata with the existing metadata.
for (const auto& metadata : *metadata_map) {
duplicated_metadata_key_count_[metadata.first]++;
metadata_map_->insert(metadata);
}
}
void IntegrationStreamDecoder::onResetStream(Http::StreamResetReason reason, absl::string_view) {
saw_reset_ = true;
reset_reason_ = reason;
if (waiting_for_reset_) {
dispatcher_.exit();
}
}
} // namespace Envoy