OLD | NEW |
---|---|
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "net/websockets/websocket_deflate_stream.h" | 5 #include "net/websockets/websocket_deflate_stream.h" |
6 | 6 |
7 #include <stdint.h> | 7 #include <stdint.h> |
8 #include <deque> | |
8 #include <string> | 9 #include <string> |
9 | 10 |
10 #include "base/basictypes.h" | 11 #include "base/basictypes.h" |
11 #include "base/bind.h" | 12 #include "base/bind.h" |
12 #include "base/memory/ref_counted.h" | 13 #include "base/memory/ref_counted.h" |
13 #include "base/memory/scoped_ptr.h" | 14 #include "base/memory/scoped_ptr.h" |
14 #include "base/memory/scoped_vector.h" | 15 #include "base/memory/scoped_vector.h" |
15 #include "net/base/completion_callback.h" | 16 #include "net/base/completion_callback.h" |
16 #include "net/base/io_buffer.h" | 17 #include "net/base/io_buffer.h" |
17 #include "net/base/net_errors.h" | 18 #include "net/base/net_errors.h" |
19 #include "net/websockets/websocket_deflate_predictor.h" | |
18 #include "net/websockets/websocket_deflater.h" | 20 #include "net/websockets/websocket_deflater.h" |
19 #include "net/websockets/websocket_frame.h" | 21 #include "net/websockets/websocket_frame.h" |
20 #include "net/websockets/websocket_inflater.h" | 22 #include "net/websockets/websocket_inflater.h" |
21 #include "net/websockets/websocket_stream.h" | 23 #include "net/websockets/websocket_stream.h" |
22 #include "net/websockets/websocket_test_util.h" | 24 #include "net/websockets/websocket_test_util.h" |
23 #include "testing/gmock/include/gmock/gmock.h" | 25 #include "testing/gmock/include/gmock/gmock.h" |
24 #include "testing/gtest/include/gtest/gtest.h" | 26 #include "testing/gtest/include/gtest/gtest.h" |
25 | 27 |
26 namespace net { | 28 namespace net { |
27 namespace { | 29 namespace { |
(...skipping 30 matching lines...) Expand all Loading... | |
58 } | 60 } |
59 | 61 |
60 std::string ToString(IOBuffer* buffer, size_t size) { | 62 std::string ToString(IOBuffer* buffer, size_t size) { |
61 return std::string(buffer->data(), size); | 63 return std::string(buffer->data(), size); |
62 } | 64 } |
63 | 65 |
64 std::string ToString(const scoped_refptr<IOBuffer>& buffer, size_t size) { | 66 std::string ToString(const scoped_refptr<IOBuffer>& buffer, size_t size) { |
65 return ToString(buffer.get(), size); | 67 return ToString(buffer.get(), size); |
66 } | 68 } |
67 | 69 |
68 std::string ToString(WebSocketFrame* frame) { | 70 std::string ToString(const WebSocketFrame* frame) { |
69 return frame->data ? ToString(frame->data, frame->header.payload_length) : ""; | 71 return frame->data ? ToString(frame->data, frame->header.payload_length) : ""; |
70 } | 72 } |
71 | 73 |
72 void AppendTo(ScopedVector<WebSocketFrame>* frames, | 74 void AppendTo(ScopedVector<WebSocketFrame>* frames, |
73 WebSocketFrameHeader::OpCode opcode, | 75 WebSocketFrameHeader::OpCode opcode, |
74 FrameFlag flag, | 76 FrameFlag flag, |
75 const std::string& data) { | 77 const std::string& data) { |
76 scoped_ptr<WebSocketFrame> frame(new WebSocketFrame(opcode)); | 78 scoped_ptr<WebSocketFrame> frame(new WebSocketFrame(opcode)); |
77 frame->header.final = (flag & kFinal); | 79 frame->header.final = (flag & kFinal); |
78 frame->header.reserved1 = (flag & kReserved1); | 80 frame->header.reserved1 = (flag & kReserved1); |
(...skipping 15 matching lines...) Expand all Loading... | |
94 public: | 96 public: |
95 MOCK_METHOD2(ReadFrames, int(ScopedVector<WebSocketFrame>*, | 97 MOCK_METHOD2(ReadFrames, int(ScopedVector<WebSocketFrame>*, |
96 const CompletionCallback&)); | 98 const CompletionCallback&)); |
97 MOCK_METHOD2(WriteFrames, int(ScopedVector<WebSocketFrame>*, | 99 MOCK_METHOD2(WriteFrames, int(ScopedVector<WebSocketFrame>*, |
98 const CompletionCallback&)); | 100 const CompletionCallback&)); |
99 MOCK_METHOD0(Close, void()); | 101 MOCK_METHOD0(Close, void()); |
100 MOCK_CONST_METHOD0(GetSubProtocol, std::string()); | 102 MOCK_CONST_METHOD0(GetSubProtocol, std::string()); |
101 MOCK_CONST_METHOD0(GetExtensions, std::string()); | 103 MOCK_CONST_METHOD0(GetExtensions, std::string()); |
102 }; | 104 }; |
103 | 105 |
106 // This mock class relies on some assumptions. | |
107 // - RecordProcessedDataFrame is called after the corresponding WriteFrames | |
108 // call. | |
109 // - RecordSentDataFrame is called before sending the frame. | |
110 class WebSocketDeflatePredictorMock : public WebSocketDeflatePredictor { | |
111 public: | |
112 WebSocketDeflatePredictorMock() : result_(DEFLATE) {} | |
113 virtual ~WebSocketDeflatePredictorMock() { | |
114 DCHECK(frames_to_be_processed_.empty()); | |
115 DCHECK(frames_to_be_sent_.empty()); | |
116 } | |
117 | |
118 virtual Result Predict(const ScopedVector<WebSocketFrame>& frames, | |
119 size_t frame_index) OVERRIDE { | |
120 return result_; | |
121 } | |
122 virtual void RecordProcessedDataFrame(const WebSocketFrame* frame) OVERRIDE { | |
123 DCHECK(WebSocketFrameHeader::IsKnownDataOpCode(frame->header.opcode)); | |
124 DCHECK(!frame->header.reserved1); | |
125 DCHECK(!frames_to_be_processed_.empty()); | |
126 DCHECK_EQ(frame, frames_to_be_processed_.front()); | |
127 frames_to_be_processed_.pop_front(); | |
128 } | |
129 virtual void RecordSentDataFrame(const WebSocketFrame* frame) OVERRIDE { | |
130 DCHECK(WebSocketFrameHeader::IsKnownDataOpCode(frame->header.opcode)); | |
131 frames_to_be_sent_.push_back(frame); | |
132 } | |
133 | |
134 void set_result(Result result) { result_ = result; } | |
135 void AddFrameToBeProcessed(const WebSocketFrame* frame) { | |
136 if (!WebSocketFrameHeader::IsKnownDataOpCode(frame->header.opcode)) | |
137 return; | |
138 frames_to_be_processed_.push_back(frame); | |
139 } | |
140 void AddFramesToBeProcessed(const ScopedVector<WebSocketFrame>& frames) { | |
141 for (size_t i = 0; i < frames.size(); ++i) | |
142 AddFrameToBeProcessed(frames[i]); | |
143 } | |
144 void VerifySentFrame(const WebSocketFrame* frame) { | |
145 if (!WebSocketFrameHeader::IsKnownDataOpCode(frame->header.opcode)) | |
146 return; | |
147 DCHECK(!frames_to_be_sent_.empty()); | |
148 DCHECK_EQ(frame, frames_to_be_sent_.front()); | |
149 frames_to_be_sent_.pop_front(); | |
150 } | |
151 void VerifySentFrames(const ScopedVector<WebSocketFrame>& frames) { | |
152 for (size_t i = 0; i < frames.size(); ++i) | |
153 VerifySentFrame(frames[i]); | |
154 } | |
155 // Call this method in order to disable checks in the destructor when | |
156 // WriteFrames fails. | |
157 void Clear() { | |
158 frames_to_be_processed_.clear(); | |
159 frames_to_be_sent_.clear(); | |
160 } | |
161 | |
162 private: | |
163 Result result_; | |
164 std::deque<const WebSocketFrame*> frames_to_be_processed_; | |
165 std::deque<const WebSocketFrame*> frames_to_be_sent_; | |
166 }; | |
167 | |
104 class WebSocketDeflateStreamTest : public ::testing::Test { | 168 class WebSocketDeflateStreamTest : public ::testing::Test { |
105 public: | 169 public: |
106 WebSocketDeflateStreamTest() | 170 WebSocketDeflateStreamTest() |
107 : mock_stream_(NULL) { | 171 : mock_stream_(NULL) { |
108 mock_stream_ = new testing::StrictMock<MockWebSocketStream>; | 172 mock_stream_ = new testing::StrictMock<MockWebSocketStream>; |
173 predictor_ = new WebSocketDeflatePredictorMock; | |
109 deflate_stream_.reset(new WebSocketDeflateStream( | 174 deflate_stream_.reset(new WebSocketDeflateStream( |
110 scoped_ptr<WebSocketStream>(mock_stream_), | 175 scoped_ptr<WebSocketStream>(mock_stream_), |
111 WebSocketDeflater::TAKE_OVER_CONTEXT)); | 176 WebSocketDeflater::TAKE_OVER_CONTEXT, |
177 scoped_ptr<WebSocketDeflatePredictor>(predictor_))); | |
112 } | 178 } |
113 virtual ~WebSocketDeflateStreamTest() {} | 179 virtual ~WebSocketDeflateStreamTest() {} |
114 | 180 |
115 protected: | 181 protected: |
116 scoped_ptr<WebSocketDeflateStream> deflate_stream_; | 182 scoped_ptr<WebSocketDeflateStream> deflate_stream_; |
117 // |mock_stream_| will be deleted when |deflate_stream_| is destroyed. | 183 // |mock_stream_| will be deleted when |deflate_stream_| is destroyed. |
118 MockWebSocketStream* mock_stream_; | 184 MockWebSocketStream* mock_stream_; |
185 // |predictor_| will be deleted when |deflate_stream_| is destroyed. | |
186 WebSocketDeflatePredictorMock* predictor_; | |
119 }; | 187 }; |
120 | 188 |
121 // Since WebSocketDeflater with DoNotTakeOverContext is well tested at | 189 // Since WebSocketDeflater with DoNotTakeOverContext is well tested at |
122 // websocket_deflater_test.cc, we have only one test for this configuration | 190 // websocket_deflater_test.cc, we have only a few tests for this configuration |
123 // here. | 191 // here. |
124 class WebSocketDeflateStreamWithDoNotTakeOverContextTest | 192 class WebSocketDeflateStreamWithDoNotTakeOverContextTest |
125 : public ::testing::Test { | 193 : public ::testing::Test { |
126 public: | 194 public: |
127 WebSocketDeflateStreamWithDoNotTakeOverContextTest() | 195 WebSocketDeflateStreamWithDoNotTakeOverContextTest() |
128 : mock_stream_(NULL) { | 196 : mock_stream_(NULL) { |
129 mock_stream_ = new testing::StrictMock<MockWebSocketStream>; | 197 mock_stream_ = new testing::StrictMock<MockWebSocketStream>; |
198 predictor_ = new WebSocketDeflatePredictorMock; | |
130 deflate_stream_.reset(new WebSocketDeflateStream( | 199 deflate_stream_.reset(new WebSocketDeflateStream( |
131 scoped_ptr<WebSocketStream>(mock_stream_), | 200 scoped_ptr<WebSocketStream>(mock_stream_), |
132 WebSocketDeflater::DO_NOT_TAKE_OVER_CONTEXT)); | 201 WebSocketDeflater::DO_NOT_TAKE_OVER_CONTEXT, |
202 scoped_ptr<WebSocketDeflatePredictor>(predictor_))); | |
133 } | 203 } |
134 virtual ~WebSocketDeflateStreamWithDoNotTakeOverContextTest() {} | 204 virtual ~WebSocketDeflateStreamWithDoNotTakeOverContextTest() {} |
135 | 205 |
136 protected: | 206 protected: |
137 scoped_ptr<WebSocketDeflateStream> deflate_stream_; | 207 scoped_ptr<WebSocketDeflateStream> deflate_stream_; |
138 // |mock_stream_| will be deleted when |deflate_stream_| is destroyed. | 208 // |mock_stream_| will be deleted when |deflate_stream_| is destroyed. |
139 MockWebSocketStream* mock_stream_; | 209 MockWebSocketStream* mock_stream_; |
210 // |predictor_| will be deleted when |deflate_stream_| is destroyed. | |
211 WebSocketDeflatePredictorMock* predictor_; | |
140 }; | 212 }; |
141 | 213 |
142 // ReadFrameStub is a stub for WebSocketStream::ReadFrames. | 214 // ReadFrameStub is a stub for WebSocketStream::ReadFrames. |
143 // It returns |result_| and |frames_to_output_| to the caller and | 215 // It returns |result_| and |frames_to_output_| to the caller and |
144 // saves parameters to |frames_passed_| and |callback_|. | 216 // saves parameters to |frames_passed_| and |callback_|. |
145 class ReadFramesStub { | 217 class ReadFramesStub { |
146 public: | 218 public: |
147 explicit ReadFramesStub(int result) : result_(result) {} | 219 explicit ReadFramesStub(int result) : result_(result) {} |
148 | 220 |
149 ReadFramesStub(int result, ScopedVector<WebSocketFrame>* frames_to_output) | 221 ReadFramesStub(int result, ScopedVector<WebSocketFrame>* frames_to_output) |
(...skipping 16 matching lines...) Expand all Loading... | |
166 return frames_passed_; | 238 return frames_passed_; |
167 } | 239 } |
168 | 240 |
169 private: | 241 private: |
170 int result_; | 242 int result_; |
171 CompletionCallback callback_; | 243 CompletionCallback callback_; |
172 ScopedVector<WebSocketFrame> frames_to_output_; | 244 ScopedVector<WebSocketFrame> frames_to_output_; |
173 ScopedVector<WebSocketFrame>* frames_passed_; | 245 ScopedVector<WebSocketFrame>* frames_passed_; |
174 }; | 246 }; |
175 | 247 |
176 // WriteFrameStub is a stub for WebSocketStream::WriteFrames. | 248 // WriteFramesStub is a stub for WebSocketStream::WriteFrames. |
177 // It returns |result_| and |frames_| to the caller and | 249 // It returns |result_| and |frames_| to the caller and |
178 // saves |callback| parameter to |callback_|. | 250 // saves |callback| parameter to |callback_|. |
179 class WriteFramesStub { | 251 class WriteFramesStub { |
180 public: | 252 public: |
181 explicit WriteFramesStub(int result) : result_(result) {} | 253 explicit WriteFramesStub(WebSocketDeflatePredictorMock* predictor, |
254 int result) | |
255 : result_(result), predictor_(predictor) {} | |
182 | 256 |
183 int Call(ScopedVector<WebSocketFrame>* frames, | 257 int Call(ScopedVector<WebSocketFrame>* frames, |
184 const CompletionCallback& callback) { | 258 const CompletionCallback& callback) { |
185 for (size_t i = 0; i < frames->size(); ++i) { | 259 for (size_t i = 0; i < frames->size(); ++i) { |
Adam Rice
2013/10/29 04:11:42
It looks like you could just do frames_->insert(fr
yhirano
2013/10/29 06:11:06
Done.
| |
186 frames_.push_back((*frames)[i]); | 260 frames_.push_back((*frames)[i]); |
187 } | 261 } |
188 frames->weak_clear(); | 262 frames->weak_clear(); |
189 callback_ = callback; | 263 callback_ = callback; |
264 predictor_->VerifySentFrames(frames_); | |
190 return result_; | 265 return result_; |
191 } | 266 } |
192 | 267 |
193 int result() const { return result_; } | 268 int result() const { return result_; } |
194 const CompletionCallback callback() const { return callback_; } | 269 const CompletionCallback callback() const { return callback_; } |
195 ScopedVector<WebSocketFrame>* frames() { return &frames_; } | 270 ScopedVector<WebSocketFrame>* frames() { return &frames_; } |
196 | 271 |
197 private: | 272 private: |
198 int result_; | 273 int result_; |
199 CompletionCallback callback_; | 274 CompletionCallback callback_; |
200 ScopedVector<WebSocketFrame> frames_; | 275 ScopedVector<WebSocketFrame> frames_; |
276 WebSocketDeflatePredictorMock* predictor_; | |
201 }; | 277 }; |
202 | 278 |
203 TEST_F(WebSocketDeflateStreamTest, ReadFailedImmediately) { | 279 TEST_F(WebSocketDeflateStreamTest, ReadFailedImmediately) { |
204 ScopedVector<WebSocketFrame> frames; | 280 ScopedVector<WebSocketFrame> frames; |
205 CompletionCallback callback; | 281 CompletionCallback callback; |
206 { | 282 { |
207 InSequence s; | 283 InSequence s; |
208 EXPECT_CALL(*mock_stream_, ReadFrames(&frames, _)) | 284 EXPECT_CALL(*mock_stream_, ReadFrames(&frames, _)) |
209 .WillOnce(Return(ERR_FAILED)); | 285 .WillOnce(Return(ERR_FAILED)); |
210 } | 286 } |
(...skipping 573 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
784 TEST_F(WebSocketDeflateStreamTest, WriteFailedImmediately) { | 860 TEST_F(WebSocketDeflateStreamTest, WriteFailedImmediately) { |
785 ScopedVector<WebSocketFrame> frames; | 861 ScopedVector<WebSocketFrame> frames; |
786 CompletionCallback callback; | 862 CompletionCallback callback; |
787 { | 863 { |
788 InSequence s; | 864 InSequence s; |
789 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | 865 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) |
790 .WillOnce(Return(ERR_FAILED)); | 866 .WillOnce(Return(ERR_FAILED)); |
791 } | 867 } |
792 | 868 |
793 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "hello"); | 869 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "hello"); |
870 predictor_->AddFramesToBeProcessed(frames); | |
794 EXPECT_EQ(ERR_FAILED, deflate_stream_->WriteFrames(&frames, callback)); | 871 EXPECT_EQ(ERR_FAILED, deflate_stream_->WriteFrames(&frames, callback)); |
872 predictor_->Clear(); | |
795 } | 873 } |
796 | 874 |
797 TEST_F(WebSocketDeflateStreamTest, WriteFrameImmediately) { | 875 TEST_F(WebSocketDeflateStreamTest, WriteFrameImmediately) { |
798 ScopedVector<WebSocketFrame> frames; | 876 ScopedVector<WebSocketFrame> frames; |
799 CompletionCallback callback; | 877 CompletionCallback callback; |
800 WriteFramesStub stub(OK); | 878 WriteFramesStub stub(predictor_, OK); |
801 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); | 879 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); |
880 predictor_->AddFramesToBeProcessed(frames); | |
802 { | 881 { |
803 InSequence s; | 882 InSequence s; |
804 EXPECT_CALL(*mock_stream_, WriteFrames(_, _)) | 883 EXPECT_CALL(*mock_stream_, WriteFrames(_, _)) |
805 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | 884 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); |
806 } | 885 } |
807 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | 886 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); |
808 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | 887 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); |
809 ASSERT_EQ(1u, frames_passed.size()); | 888 ASSERT_EQ(1u, frames_passed.size()); |
810 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); | 889 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); |
811 EXPECT_TRUE(frames_passed[0]->header.final); | 890 EXPECT_TRUE(frames_passed[0]->header.final); |
812 EXPECT_TRUE(frames_passed[0]->header.reserved1); | 891 EXPECT_TRUE(frames_passed[0]->header.reserved1); |
813 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), | 892 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), |
814 ToString(frames_passed[0])); | 893 ToString(frames_passed[0])); |
815 } | 894 } |
816 | 895 |
817 TEST_F(WebSocketDeflateStreamTest, WriteFrameAsync) { | 896 TEST_F(WebSocketDeflateStreamTest, WriteFrameAsync) { |
818 WriteFramesStub stub(ERR_IO_PENDING); | 897 WriteFramesStub stub(predictor_, ERR_IO_PENDING); |
819 MockCallback mock_callback, checkpoint; | 898 MockCallback mock_callback, checkpoint; |
820 CompletionCallback callback = | 899 CompletionCallback callback = |
821 base::Bind(&MockCallback::Call, base::Unretained(&mock_callback)); | 900 base::Bind(&MockCallback::Call, base::Unretained(&mock_callback)); |
822 ScopedVector<WebSocketFrame> frames; | 901 ScopedVector<WebSocketFrame> frames; |
823 { | 902 { |
824 InSequence s; | 903 InSequence s; |
825 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | 904 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) |
826 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | 905 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); |
827 EXPECT_CALL(checkpoint, Call(0)); | 906 EXPECT_CALL(checkpoint, Call(0)); |
828 EXPECT_CALL(mock_callback, Call(OK)); | 907 EXPECT_CALL(mock_callback, Call(OK)); |
829 } | 908 } |
830 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); | 909 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); |
910 predictor_->AddFramesToBeProcessed(frames); | |
831 ASSERT_EQ(ERR_IO_PENDING, deflate_stream_->WriteFrames(&frames, callback)); | 911 ASSERT_EQ(ERR_IO_PENDING, deflate_stream_->WriteFrames(&frames, callback)); |
832 | 912 |
833 checkpoint.Call(0); | 913 checkpoint.Call(0); |
834 stub.callback().Run(OK); | 914 stub.callback().Run(OK); |
835 | 915 |
836 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | 916 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); |
837 ASSERT_EQ(1u, frames_passed.size()); | 917 ASSERT_EQ(1u, frames_passed.size()); |
838 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); | 918 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); |
839 EXPECT_TRUE(frames_passed[0]->header.final); | 919 EXPECT_TRUE(frames_passed[0]->header.final); |
840 EXPECT_TRUE(frames_passed[0]->header.reserved1); | 920 EXPECT_TRUE(frames_passed[0]->header.reserved1); |
841 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), | 921 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), |
842 ToString(frames_passed[0])); | 922 ToString(frames_passed[0])); |
843 } | 923 } |
844 | 924 |
845 TEST_F(WebSocketDeflateStreamTest, WriteControlFrameBetweenDataFrames) { | 925 TEST_F(WebSocketDeflateStreamTest, WriteControlFrameBetweenDataFrames) { |
846 ScopedVector<WebSocketFrame> frames; | 926 ScopedVector<WebSocketFrame> frames; |
847 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kNoFlag, "Hel"); | 927 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kNoFlag, "Hel"); |
848 AppendTo(&frames, WebSocketFrameHeader::kOpCodePing, kFinal); | 928 AppendTo(&frames, WebSocketFrameHeader::kOpCodePing, kFinal); |
849 AppendTo(&frames, WebSocketFrameHeader::kOpCodeContinuation, kFinal, "lo"); | 929 AppendTo(&frames, WebSocketFrameHeader::kOpCodeContinuation, kFinal, "lo"); |
850 WriteFramesStub stub(OK); | 930 predictor_->AddFramesToBeProcessed(frames); |
931 WriteFramesStub stub(predictor_, OK); | |
851 CompletionCallback callback; | 932 CompletionCallback callback; |
852 | 933 |
853 { | 934 { |
854 InSequence s; | 935 InSequence s; |
855 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | 936 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) |
856 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | 937 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); |
857 } | 938 } |
858 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | 939 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); |
859 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | 940 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); |
860 ASSERT_EQ(2u, frames_passed.size()); | 941 ASSERT_EQ(2u, frames_passed.size()); |
861 EXPECT_EQ(WebSocketFrameHeader::kOpCodePing, frames_passed[0]->header.opcode); | 942 EXPECT_EQ(WebSocketFrameHeader::kOpCodePing, frames_passed[0]->header.opcode); |
862 EXPECT_TRUE(frames_passed[0]->header.final); | 943 EXPECT_TRUE(frames_passed[0]->header.final); |
863 EXPECT_FALSE(frames_passed[0]->header.reserved1); | 944 EXPECT_FALSE(frames_passed[0]->header.reserved1); |
864 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[1]->header.opcode); | 945 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[1]->header.opcode); |
865 EXPECT_TRUE(frames_passed[1]->header.final); | 946 EXPECT_TRUE(frames_passed[1]->header.final); |
866 EXPECT_TRUE(frames_passed[1]->header.reserved1); | 947 EXPECT_TRUE(frames_passed[1]->header.reserved1); |
867 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), | 948 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), |
868 ToString(frames_passed[1])); | 949 ToString(frames_passed[1])); |
869 } | 950 } |
870 | 951 |
871 TEST_F(WebSocketDeflateStreamTest, WriteEmptyMessage) { | 952 TEST_F(WebSocketDeflateStreamTest, WriteEmptyMessage) { |
872 ScopedVector<WebSocketFrame> frames; | 953 ScopedVector<WebSocketFrame> frames; |
873 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal); | 954 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal); |
874 WriteFramesStub stub(OK); | 955 predictor_->AddFramesToBeProcessed(frames); |
956 WriteFramesStub stub(predictor_, OK); | |
875 CompletionCallback callback; | 957 CompletionCallback callback; |
876 | 958 |
877 { | 959 { |
878 InSequence s; | 960 InSequence s; |
879 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | 961 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) |
880 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | 962 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); |
881 } | 963 } |
882 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | 964 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); |
883 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | 965 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); |
884 ASSERT_EQ(1u, frames_passed.size()); | 966 ASSERT_EQ(1u, frames_passed.size()); |
885 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); | 967 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); |
886 EXPECT_TRUE(frames_passed[0]->header.final); | 968 EXPECT_TRUE(frames_passed[0]->header.final); |
887 EXPECT_TRUE(frames_passed[0]->header.reserved1); | 969 EXPECT_TRUE(frames_passed[0]->header.reserved1); |
888 EXPECT_EQ(std::string("\x02\x00", 2), ToString(frames_passed[0])); | 970 EXPECT_EQ(std::string("\x02\x00", 2), ToString(frames_passed[0])); |
889 } | 971 } |
890 | 972 |
973 TEST_F(WebSocketDeflateStreamTest, WriteUncompressedMessage) { | |
974 ScopedVector<WebSocketFrame> frames; | |
975 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kNoFlag, "AAAA"); | |
976 AppendTo(&frames, WebSocketFrameHeader::kOpCodeContinuation, kFinal, "AAA"); | |
977 predictor_->AddFramesToBeProcessed(frames); | |
978 WriteFramesStub stub(predictor_, OK); | |
979 CompletionCallback callback; | |
980 | |
981 predictor_->set_result(WebSocketDeflatePredictor::DO_NOT_DEFLATE); | |
982 | |
983 { | |
984 InSequence s; | |
985 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | |
986 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | |
987 } | |
988 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | |
989 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | |
990 ASSERT_EQ(2u, frames_passed.size()); | |
991 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); | |
992 EXPECT_FALSE(frames_passed[0]->header.final); | |
993 EXPECT_FALSE(frames_passed[0]->header.reserved1); | |
994 EXPECT_EQ("AAAA", ToString(frames_passed[0])); | |
995 EXPECT_EQ(WebSocketFrameHeader::kOpCodeContinuation, | |
996 frames_passed[1]->header.opcode); | |
997 EXPECT_TRUE(frames_passed[1]->header.final); | |
998 EXPECT_FALSE(frames_passed[1]->header.reserved1); | |
999 EXPECT_EQ("AAA", ToString(frames_passed[1])); | |
1000 } | |
1001 | |
891 TEST_F(WebSocketDeflateStreamTest, LargeDeflatedFramesShouldBeSplit) { | 1002 TEST_F(WebSocketDeflateStreamTest, LargeDeflatedFramesShouldBeSplit) { |
892 WebSocketDeflater deflater(WebSocketDeflater::TAKE_OVER_CONTEXT); | 1003 WebSocketDeflater deflater(WebSocketDeflater::TAKE_OVER_CONTEXT); |
893 LinearCongruentialGenerator lcg(133); | 1004 LinearCongruentialGenerator lcg(133); |
894 WriteFramesStub stub(OK); | 1005 WriteFramesStub stub(predictor_, OK); |
895 CompletionCallback callback; | 1006 CompletionCallback callback; |
896 const size_t size = 1024; | 1007 const size_t size = 1024; |
897 | 1008 |
898 { | 1009 { |
899 InSequence s; | 1010 InSequence s; |
900 EXPECT_CALL(*mock_stream_, WriteFrames(_, _)) | 1011 EXPECT_CALL(*mock_stream_, WriteFrames(_, _)) |
901 .WillRepeatedly(Invoke(&stub, &WriteFramesStub::Call)); | 1012 .WillRepeatedly(Invoke(&stub, &WriteFramesStub::Call)); |
902 } | 1013 } |
903 ScopedVector<WebSocketFrame> total_compressed_frames; | 1014 ScopedVector<WebSocketFrame> total_compressed_frames; |
904 | 1015 |
905 deflater.Initialize(kWindowBits); | 1016 deflater.Initialize(kWindowBits); |
906 while (true) { | 1017 while (true) { |
907 bool is_final = (total_compressed_frames.size() >= 2); | 1018 bool is_final = (total_compressed_frames.size() >= 2); |
908 ScopedVector<WebSocketFrame> frames; | 1019 ScopedVector<WebSocketFrame> frames; |
909 std::string data; | 1020 std::string data; |
910 for (size_t i = 0; i < size; ++i) | 1021 for (size_t i = 0; i < size; ++i) |
911 data += static_cast<char>(lcg.Generate()); | 1022 data += static_cast<char>(lcg.Generate()); |
912 deflater.AddBytes(data.data(), data.size()); | 1023 deflater.AddBytes(data.data(), data.size()); |
913 FrameFlag flag = is_final ? kFinal : kNoFlag; | 1024 FrameFlag flag = is_final ? kFinal : kNoFlag; |
914 AppendTo(&frames, WebSocketFrameHeader::kOpCodeBinary, flag, data); | 1025 AppendTo(&frames, WebSocketFrameHeader::kOpCodeBinary, flag, data); |
1026 predictor_->AddFramesToBeProcessed(frames); | |
915 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | 1027 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); |
916 for (size_t i = 0; i < stub.frames()->size(); ++i) | 1028 for (size_t i = 0; i < stub.frames()->size(); ++i) |
917 total_compressed_frames.push_back((*stub.frames())[i]); | 1029 total_compressed_frames.push_back((*stub.frames())[i]); |
918 stub.frames()->weak_clear(); | 1030 stub.frames()->weak_clear(); |
919 if (is_final) | 1031 if (is_final) |
920 break; | 1032 break; |
921 } | 1033 } |
922 deflater.Finish(); | 1034 deflater.Finish(); |
923 std::string total_deflated; | 1035 std::string total_deflated; |
924 for (size_t i = 0; i < total_compressed_frames.size(); ++i) { | 1036 for (size_t i = 0; i < total_compressed_frames.size(); ++i) { |
(...skipping 13 matching lines...) Expand all Loading... | |
938 total_deflated += ToString(frame); | 1050 total_deflated += ToString(frame); |
939 } | 1051 } |
940 EXPECT_EQ(total_deflated, | 1052 EXPECT_EQ(total_deflated, |
941 ToString(deflater.GetOutput(deflater.CurrentOutputSize()))); | 1053 ToString(deflater.GetOutput(deflater.CurrentOutputSize()))); |
942 } | 1054 } |
943 | 1055 |
944 TEST_F(WebSocketDeflateStreamTest, WriteMultipleMessages) { | 1056 TEST_F(WebSocketDeflateStreamTest, WriteMultipleMessages) { |
945 ScopedVector<WebSocketFrame> frames; | 1057 ScopedVector<WebSocketFrame> frames; |
946 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); | 1058 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); |
947 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); | 1059 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); |
948 WriteFramesStub stub(OK); | 1060 predictor_->AddFramesToBeProcessed(frames); |
1061 WriteFramesStub stub(predictor_, OK); | |
949 CompletionCallback callback; | 1062 CompletionCallback callback; |
950 | 1063 |
951 { | 1064 { |
952 InSequence s; | 1065 InSequence s; |
953 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | 1066 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) |
954 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | 1067 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); |
955 } | 1068 } |
956 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | 1069 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); |
957 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | 1070 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); |
958 ASSERT_EQ(2u, frames_passed.size()); | 1071 ASSERT_EQ(2u, frames_passed.size()); |
959 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); | 1072 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); |
960 EXPECT_TRUE(frames_passed[0]->header.final); | 1073 EXPECT_TRUE(frames_passed[0]->header.final); |
961 EXPECT_TRUE(frames_passed[0]->header.reserved1); | 1074 EXPECT_TRUE(frames_passed[0]->header.reserved1); |
962 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), | 1075 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), |
963 ToString(frames_passed[0])); | 1076 ToString(frames_passed[0])); |
964 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[1]->header.opcode); | 1077 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[1]->header.opcode); |
965 EXPECT_TRUE(frames_passed[1]->header.final); | 1078 EXPECT_TRUE(frames_passed[1]->header.final); |
966 EXPECT_TRUE(frames_passed[1]->header.reserved1); | 1079 EXPECT_TRUE(frames_passed[1]->header.reserved1); |
967 EXPECT_EQ(std::string("\xf2\x00\x11\x00\x00", 5), ToString(frames_passed[1])); | 1080 EXPECT_EQ(std::string("\xf2\x00\x11\x00\x00", 5), ToString(frames_passed[1])); |
968 } | 1081 } |
969 | 1082 |
970 TEST_F(WebSocketDeflateStreamWithDoNotTakeOverContextTest, | 1083 TEST_F(WebSocketDeflateStreamWithDoNotTakeOverContextTest, |
971 WriteMultipleMessages) { | 1084 WriteMultipleMessages) { |
972 ScopedVector<WebSocketFrame> frames; | 1085 ScopedVector<WebSocketFrame> frames; |
973 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); | 1086 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); |
974 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); | 1087 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kFinal, "Hello"); |
975 WriteFramesStub stub(OK); | 1088 predictor_->AddFramesToBeProcessed(frames); |
1089 WriteFramesStub stub(predictor_, OK); | |
976 CompletionCallback callback; | 1090 CompletionCallback callback; |
977 | 1091 |
978 { | 1092 { |
979 InSequence s; | 1093 InSequence s; |
980 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | 1094 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) |
981 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | 1095 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); |
982 } | 1096 } |
983 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | 1097 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); |
984 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | 1098 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); |
985 ASSERT_EQ(2u, frames_passed.size()); | 1099 ASSERT_EQ(2u, frames_passed.size()); |
986 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); | 1100 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); |
987 EXPECT_TRUE(frames_passed[0]->header.final); | 1101 EXPECT_TRUE(frames_passed[0]->header.final); |
988 EXPECT_TRUE(frames_passed[0]->header.reserved1); | 1102 EXPECT_TRUE(frames_passed[0]->header.reserved1); |
989 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), | 1103 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), |
990 ToString(frames_passed[0])); | 1104 ToString(frames_passed[0])); |
991 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[1]->header.opcode); | 1105 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[1]->header.opcode); |
992 EXPECT_TRUE(frames_passed[1]->header.final); | 1106 EXPECT_TRUE(frames_passed[1]->header.final); |
993 EXPECT_TRUE(frames_passed[1]->header.reserved1); | 1107 EXPECT_TRUE(frames_passed[1]->header.reserved1); |
994 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), | 1108 EXPECT_EQ(std::string("\xf2\x48\xcd\xc9\xc9\x07\x00", 7), |
995 ToString(frames_passed[1])); | 1109 ToString(frames_passed[1])); |
996 } | 1110 } |
997 | 1111 |
1112 // In order to check the stream works correctly for multiple | |
1113 // "PossiblyCompressedMessage"s, we test various messages at one test case. | |
1114 TEST_F(WebSocketDeflateStreamWithDoNotTakeOverContextTest, | |
1115 WritePossiblyCompressMessages) { | |
1116 ScopedVector<WebSocketFrame> frames; | |
1117 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kNoFlag, "He"); | |
1118 AppendTo(&frames, WebSocketFrameHeader::kOpCodeContinuation, kFinal, "llo"); | |
1119 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kNoFlag, "AAAAAAAAAA"); | |
1120 AppendTo(&frames, WebSocketFrameHeader::kOpCodeContinuation, kFinal, "AA"); | |
1121 AppendTo(&frames, WebSocketFrameHeader::kOpCodeText, kNoFlag, "XX"); | |
1122 AppendTo(&frames, WebSocketFrameHeader::kOpCodeContinuation, kFinal, "YY"); | |
1123 predictor_->AddFramesToBeProcessed(frames); | |
1124 WriteFramesStub stub(predictor_, OK); | |
1125 CompletionCallback callback; | |
1126 predictor_->set_result(WebSocketDeflatePredictor::TRY_DEFLATE); | |
1127 | |
1128 { | |
1129 InSequence s; | |
1130 EXPECT_CALL(*mock_stream_, WriteFrames(&frames, _)) | |
1131 .WillOnce(Invoke(&stub, &WriteFramesStub::Call)); | |
1132 } | |
1133 ASSERT_EQ(OK, deflate_stream_->WriteFrames(&frames, callback)); | |
1134 const ScopedVector<WebSocketFrame>& frames_passed = *stub.frames(); | |
1135 ASSERT_EQ(5u, frames_passed.size()); | |
1136 | |
1137 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[0]->header.opcode); | |
1138 EXPECT_FALSE(frames_passed[0]->header.final); | |
1139 EXPECT_FALSE(frames_passed[0]->header.reserved1); | |
1140 EXPECT_EQ("He", ToString(frames_passed[0])); | |
1141 EXPECT_EQ(WebSocketFrameHeader::kOpCodeContinuation, | |
1142 frames_passed[1]->header.opcode); | |
1143 EXPECT_TRUE(frames_passed[1]->header.final); | |
1144 EXPECT_FALSE(frames_passed[1]->header.reserved1); | |
1145 EXPECT_EQ("llo", ToString(frames_passed[1])); | |
1146 | |
1147 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[2]->header.opcode); | |
1148 EXPECT_TRUE(frames_passed[2]->header.final); | |
1149 EXPECT_TRUE(frames_passed[2]->header.reserved1); | |
1150 EXPECT_EQ(std::string("\x72\x74\x44\x00\x00\x00", 6), | |
1151 ToString(frames_passed[2])); | |
1152 | |
1153 EXPECT_EQ(WebSocketFrameHeader::kOpCodeText, frames_passed[3]->header.opcode); | |
1154 EXPECT_FALSE(frames_passed[3]->header.final); | |
1155 EXPECT_FALSE(frames_passed[3]->header.reserved1); | |
1156 EXPECT_EQ("XX", ToString(frames_passed[3])); | |
1157 EXPECT_EQ(WebSocketFrameHeader::kOpCodeContinuation, | |
1158 frames_passed[4]->header.opcode); | |
1159 EXPECT_TRUE(frames_passed[4]->header.final); | |
1160 EXPECT_FALSE(frames_passed[4]->header.reserved1); | |
1161 EXPECT_EQ("YY", ToString(frames_passed[4])); | |
1162 } | |
1163 | |
998 } // namespace | 1164 } // namespace |
999 | 1165 |
1000 } // namespace net | 1166 } // namespace net |
OLD | NEW |