blob: 50d9987b811eda1d6eaeb3227d45a39b9509a3f6 [file] [log] [blame]
QUICHE team82dee2f2019-01-18 12:35:12 -05001// Copyright 2016 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#ifndef QUICHE_SPDY_CORE_SPDY_DEFRAMER_VISITOR_H_
6#define QUICHE_SPDY_CORE_SPDY_DEFRAMER_VISITOR_H_
7
8// Supports testing by converting callbacks to SpdyFramerVisitorInterface into
9// callbacks to SpdyDeframerVisitorInterface, whose arguments are generally
10// SpdyFrameIR instances. This enables a test client or test backend to operate
11// at a level between the low-level callbacks of SpdyFramerVisitorInterface and
12// the much higher level of entire messages (i.e. headers, body, trailers).
13// Where possible the converter (SpdyTestDeframer) tries to preserve information
14// that might be useful to tests (e.g. the order of headers or the amount of
15// padding); the design also aims to allow tests to be concise, ideally
16// supporting gMock style EXPECT_CALL(visitor, OnHeaders(...matchers...))
17// without too much boilerplate.
18//
19// Only supports HTTP/2 for the moment.
20//
21// Example of usage:
22//
23// SpdyFramer framer(HTTP2);
24//
25// // Need to call SpdyTestDeframer::AtFrameEnd() after processing each
26// // frame, so tell SpdyFramer to stop after each.
27// framer.set_process_single_input_frame(true);
28//
29// // Need the new OnHeader callbacks.
30// framer.set_use_new_methods_for_test(true);
31//
32// // Create your visitor, a subclass of SpdyDeframerVisitorInterface.
33// // For example, using DeframerCallbackCollector to collect frames:
34// std::vector<CollectedFrame> collected_frames;
35// auto your_visitor = SpdyMakeUnique<DeframerCallbackCollector>(
36// &collected_frames);
37//
38// // Transfer ownership of your visitor to the converter, which ensures that
39// // your visitor stays alive while the converter needs to call it.
40// auto the_deframer = SpdyTestDeframer::CreateConverter(
41// std::move(your_visitor));
42//
43// // Tell the framer to notify SpdyTestDeframer of the decoded frame
44// // details.
45// framer.set_visitor(the_deframer.get());
46//
47// // Process frames.
48// SpdyStringPiece input = ...
49// while (!input.empty() && !framer.HasError()) {
50// size_t consumed = framer.ProcessInput(input.data(), input.size());
51// input.remove_prefix(consumed);
52// if (framer.state() == SpdyFramer::SPDY_READY_FOR_FRAME) {
53// the_deframer->AtFrameEnd();
54// }
55// }
56//
57// // Make sure that the correct frames were received. For example:
58// ASSERT_EQ(collected_frames.size(), 3);
59//
60// SpdyDataIR expected1(7 /*stream_id*/, "Data Payload");
61// expected1.set_padding_len(17);
62// EXPECT_TRUE(collected_frames[0].VerifyEquals(expected1));
63//
64// // Repeat for the other frames.
65//
66// Note that you could also seed the subclass of SpdyDeframerVisitorInterface
67// with the expected frames, which it would pop-off the list as its expectations
68// are met.
69
70#include <cstdint>
71
72#include <memory>
73#include <type_traits>
74#include <utility>
75#include <vector>
76
77#include "base/logging.h"
78#include "base/macros.h"
79#include "testing/gtest/include/gtest/gtest.h"
80#include "net/third_party/quiche/src/spdy/core/http2_frame_decoder_adapter.h"
81#include "net/third_party/quiche/src/spdy/core/spdy_protocol.h"
82#include "net/third_party/quiche/src/spdy/core/spdy_protocol_test_utils.h"
83#include "net/third_party/quiche/src/spdy/core/spdy_test_utils.h"
84#include "net/third_party/quiche/src/spdy/platform/api/spdy_string.h"
85
86namespace spdy {
87namespace test {
88
89// Non-lossy representation of a SETTINGS frame payload.
90typedef std::vector<std::pair<SpdyKnownSettingsId, uint32_t>> SettingVector;
91
92// StringPairVector is used to record information lost by SpdyHeaderBlock, in
93// particular the order of each header entry, though it doesn't expose the
94// inner details of the HPACK block, such as the type of encoding selected
95// for each header entry, nor dynamic table size changes.
96typedef std::pair<SpdyString, SpdyString> StringPair;
97typedef std::vector<StringPair> StringPairVector;
98
99// Forward decl.
100class SpdyTestDeframer;
101
102// Note that this only roughly captures the frames, as padding bytes are lost,
103// continuation frames are combined with their leading HEADERS or PUSH_PROMISE,
104// the details of the HPACK encoding are lost, leaving
105// only the list of header entries (name and value strings). If really helpful,
106// we could add a SpdyRawDeframerVisitorInterface that gets the HPACK bytes,
107// and receives continuation frames. For more info we'd need to improve
108// SpdyFramerVisitorInterface.
109class SpdyDeframerVisitorInterface {
110 public:
111 virtual ~SpdyDeframerVisitorInterface() {}
112
113 // Wrap a visitor in another SpdyDeframerVisitorInterface that will
114 // DVLOG each call, and will then forward the calls to the wrapped visitor
115 // (if provided; nullptr is OK). Takes ownership of the wrapped visitor.
116 static std::unique_ptr<SpdyDeframerVisitorInterface> LogBeforeVisiting(
117 std::unique_ptr<SpdyDeframerVisitorInterface> wrapped_visitor);
118
119 virtual void OnAltSvc(std::unique_ptr<SpdyAltSvcIR> frame) {}
120 virtual void OnData(std::unique_ptr<SpdyDataIR> frame) {}
121 virtual void OnGoAway(std::unique_ptr<SpdyGoAwayIR> frame) {}
122
123 // SpdyHeadersIR and SpdyPushPromiseIR each has a SpdyHeaderBlock which
124 // significantly modifies the headers, so the actual header entries (name
125 // and value strings) are provided in a vector.
126 virtual void OnHeaders(std::unique_ptr<SpdyHeadersIR> frame,
127 std::unique_ptr<StringPairVector> headers) {}
128
129 virtual void OnPing(std::unique_ptr<SpdyPingIR> frame) {}
130 virtual void OnPingAck(std::unique_ptr<SpdyPingIR> frame);
131 virtual void OnPriority(std::unique_ptr<SpdyPriorityIR> frame) {}
132
133 // SpdyHeadersIR and SpdyPushPromiseIR each has a SpdyHeaderBlock which
134 // significantly modifies the headers, so the actual header entries (name
135 // and value strings) are provided in a vector.
136 virtual void OnPushPromise(std::unique_ptr<SpdyPushPromiseIR> frame,
137 std::unique_ptr<StringPairVector> headers) {}
138
139 virtual void OnRstStream(std::unique_ptr<SpdyRstStreamIR> frame) {}
140
141 // SpdySettingsIR has a map for settings, so loses info about the order of
142 // settings, and whether the same setting appeared more than once, so the
143 // the actual settings (parameter and value) are provided in a vector.
144 virtual void OnSettings(std::unique_ptr<SpdySettingsIR> frame,
145 std::unique_ptr<SettingVector> settings) {}
146
147 // A settings frame with an ACK has no content, but for uniformity passing
148 // a frame with the ACK flag set.
149 virtual void OnSettingsAck(std::unique_ptr<SpdySettingsIR> frame);
150
151 virtual void OnWindowUpdate(std::unique_ptr<SpdyWindowUpdateIR> frame) {}
152
153 // The SpdyFramer will not process any more data at this point.
154 virtual void OnError(http2::Http2DecoderAdapter::SpdyFramerError error,
155 SpdyTestDeframer* deframer) {}
156};
157
158class SpdyTestDeframer : public SpdyFramerVisitorInterface {
159 public:
160 ~SpdyTestDeframer() override {}
161
162 // Creates a SpdyFramerVisitorInterface that builds SpdyFrameIR concrete
163 // instances based on the callbacks it receives; when an entire frame is
164 // decoded/reconstructed it calls the passed in SpdyDeframerVisitorInterface.
165 // Transfers ownership of visitor to the new SpdyTestDeframer, which ensures
166 // that it continues to exist while the SpdyTestDeframer exists.
167 static std::unique_ptr<SpdyTestDeframer> CreateConverter(
168 std::unique_ptr<SpdyDeframerVisitorInterface> visitor);
169
170 // Call to notify the deframer that the SpdyFramer has returned after reaching
171 // the end of decoding a frame. This is used to flush info about some frame
172 // types where we don't get a clear end signal; others are flushed (i.e. the
173 // appropriate call to the SpdyDeframerVisitorInterface method is invoked)
174 // as they're decoded by SpdyFramer and it calls the deframer. See the
175 // example in the comments at the top of this file.
176 virtual bool AtFrameEnd() = 0;
177
178 protected:
179 SpdyTestDeframer() {}
180 SpdyTestDeframer(const SpdyTestDeframer&) = delete;
181 SpdyTestDeframer& operator=(const SpdyTestDeframer&) = delete;
182};
183
184// CollectedFrame holds the result of one call to SpdyDeframerVisitorInterface,
185// as recorded by DeframerCallbackCollector.
186struct CollectedFrame {
187 CollectedFrame();
188 CollectedFrame(CollectedFrame&& other);
189 ~CollectedFrame();
190 CollectedFrame& operator=(CollectedFrame&& other);
191
192 // Compare a SpdyFrameIR sub-class instance, expected_ir, against the
193 // collected SpdyFrameIR.
194 template <class T,
195 typename X =
196 typename std::enable_if<std::is_base_of<SpdyFrameIR, T>::value>>
197 ::testing::AssertionResult VerifyHasFrame(const T& expected_ir) const {
198 return VerifySpdyFrameIREquals(expected_ir, frame_ir.get());
199 }
200
201 // Compare the collected headers against a StringPairVector. Ignores
202 // this->frame_ir.
203 ::testing::AssertionResult VerifyHasHeaders(
204 const StringPairVector& expected_headers) const;
205
206 // Compare the collected settings (parameter and value pairs) against
207 // expected_settings. Ignores this->frame_ir.
208 ::testing::AssertionResult VerifyHasSettings(
209 const SettingVector& expected_settings) const;
210
211 std::unique_ptr<SpdyFrameIR> frame_ir;
212 std::unique_ptr<StringPairVector> headers;
213 std::unique_ptr<SettingVector> settings;
214 bool error_reported = false;
215};
216
217// Creates a CollectedFrame instance for each callback, storing it in the
218// vector provided to the constructor.
219class DeframerCallbackCollector : public SpdyDeframerVisitorInterface {
220 public:
221 explicit DeframerCallbackCollector(
222 std::vector<CollectedFrame>* collected_frames);
223 ~DeframerCallbackCollector() override {}
224
225 void OnAltSvc(std::unique_ptr<SpdyAltSvcIR> frame_ir) override;
226 void OnData(std::unique_ptr<SpdyDataIR> frame_ir) override;
227 void OnGoAway(std::unique_ptr<SpdyGoAwayIR> frame_ir) override;
228 void OnHeaders(std::unique_ptr<SpdyHeadersIR> frame_ir,
229 std::unique_ptr<StringPairVector> headers) override;
230 void OnPing(std::unique_ptr<SpdyPingIR> frame_ir) override;
231 void OnPingAck(std::unique_ptr<SpdyPingIR> frame_ir) override;
232 void OnPriority(std::unique_ptr<SpdyPriorityIR> frame_ir) override;
233 void OnPushPromise(std::unique_ptr<SpdyPushPromiseIR> frame_ir,
234 std::unique_ptr<StringPairVector> headers) override;
235 void OnRstStream(std::unique_ptr<SpdyRstStreamIR> frame_ir) override;
236 void OnSettings(std::unique_ptr<SpdySettingsIR> frame_ir,
237 std::unique_ptr<SettingVector> settings) override;
238 void OnSettingsAck(std::unique_ptr<SpdySettingsIR> frame_ir) override;
239 void OnWindowUpdate(std::unique_ptr<SpdyWindowUpdateIR> frame_ir) override;
240 void OnError(http2::Http2DecoderAdapter::SpdyFramerError error,
241 SpdyTestDeframer* deframer) override;
242
243 private:
244 std::vector<CollectedFrame>* collected_frames_;
245};
246
247} // namespace test
248} // namespace spdy
249
250#endif // QUICHE_SPDY_CORE_SPDY_DEFRAMER_VISITOR_H_