QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 1 | // 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 | |
| 86 | namespace spdy { |
| 87 | namespace test { |
| 88 | |
| 89 | // Non-lossy representation of a SETTINGS frame payload. |
| 90 | typedef 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. |
| 96 | typedef std::pair<SpdyString, SpdyString> StringPair; |
| 97 | typedef std::vector<StringPair> StringPairVector; |
| 98 | |
| 99 | // Forward decl. |
| 100 | class 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. |
| 109 | class 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 | |
| 158 | class 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. |
| 186 | struct 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. |
| 219 | class 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_ |