Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2015 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 #include "base/files/file_util.h" | |
| 6 #include "base/memory/scoped_ptr.h" | |
| 7 #include "base/path_service.h" | |
| 8 #include "net/base/io_buffer.h" | |
| 9 #include "net/filter/brotli_filter.h" | |
| 10 #include "net/filter/mock_filter_context.h" | |
| 11 #include "testing/gtest/include/gtest/gtest.h" | |
| 12 #include "testing/platform_test.h" | |
| 13 | |
| 14 namespace { | |
| 15 const int kDefaultBufferSize = 4096; | |
| 16 const int kSmallBufferSize = 128; | |
| 17 } // namespace | |
| 18 | |
| 19 namespace net { | |
| 20 | |
| 21 // These tests use the path service, which uses autoreleased objects on the | |
| 22 // Mac, so this needs to be a PlatformTest. | |
| 23 class BrotliUnitTest : public PlatformTest { | |
| 24 protected: | |
| 25 void SetUp() override { | |
| 26 PlatformTest::SetUp(); | |
| 27 | |
| 28 // Get the path of data directory. | |
| 29 base::FilePath data_dir; | |
| 30 PathService::Get(base::DIR_SOURCE_ROOT, &data_dir); | |
| 31 data_dir = data_dir.AppendASCII("net"); | |
| 32 data_dir = data_dir.AppendASCII("data"); | |
| 33 data_dir = data_dir.AppendASCII("filter_unittests"); | |
| 34 | |
| 35 // Read data from the original file into buffer. | |
| 36 base::FilePath file_path; | |
| 37 file_path = data_dir.AppendASCII("google.txt"); | |
| 38 ASSERT_TRUE(base::ReadFileToString(file_path, &source_buffer_)); | |
| 39 | |
| 40 // Read data from the encoded file into buffer. | |
| 41 base::FilePath encoded_file_path; | |
| 42 encoded_file_path = data_dir.AppendASCII("google.br"); | |
| 43 ASSERT_TRUE(base::ReadFileToString(encoded_file_path, &encoded_buffer_)); | |
| 44 ASSERT_GE(kDefaultBufferSize, static_cast<int>(encoded_buffer_.size())); | |
| 45 } | |
| 46 | |
| 47 void TearDown() override { PlatformTest::TearDown(); } | |
| 48 | |
| 49 // Use filter to decode compressed data, and compare the decoded result with | |
| 50 // the orginal data. | |
| 51 // Parameters: |source| and |source_len| are original data and its size. | |
| 52 // |encoded_source| and |encoded_source_len| are compressed data and its size. | |
| 53 // |output_buffer_size| specifies the size of buffer to read out data from | |
| 54 // filter. | |
| 55 void DecodeAndCompareWithFilter(Filter* filter, | |
| 56 const char* source, | |
| 57 int source_len, | |
| 58 const char* encoded_source, | |
| 59 int encoded_source_len, | |
| 60 int output_buffer_size) { | |
| 61 // Make sure we have enough space to hold the decoding output. | |
| 62 ASSERT_GE(kDefaultBufferSize, source_len); | |
| 63 ASSERT_GE(kDefaultBufferSize, output_buffer_size); | |
| 64 | |
| 65 char decode_buffer[kDefaultBufferSize]; | |
| 66 char* decode_next = decode_buffer; | |
| 67 int decode_avail_size = kDefaultBufferSize; | |
| 68 | |
| 69 const char* encode_next = encoded_source; | |
| 70 int encode_avail_size = encoded_source_len; | |
| 71 | |
| 72 int code = Filter::FILTER_OK; | |
| 73 while (code != Filter::FILTER_DONE) { | |
| 74 int encode_data_len = | |
| 75 std::min(encode_avail_size, filter->stream_buffer_size()); | |
| 76 memcpy(filter->stream_buffer()->data(), encode_next, encode_data_len); | |
| 77 filter->FlushStreamBuffer(encode_data_len); | |
| 78 encode_next += encode_data_len; | |
| 79 encode_avail_size -= encode_data_len; | |
| 80 | |
| 81 while (true) { | |
| 82 int decode_data_len = std::min(decode_avail_size, output_buffer_size); | |
| 83 | |
| 84 code = filter->ReadData(decode_next, &decode_data_len); | |
| 85 decode_next += decode_data_len; | |
| 86 decode_avail_size -= decode_data_len; | |
| 87 | |
| 88 ASSERT_NE(Filter::FILTER_ERROR, code); | |
| 89 | |
| 90 if (code == Filter::FILTER_NEED_MORE_DATA || | |
| 91 code == Filter::FILTER_DONE) { | |
| 92 break; | |
| 93 } | |
| 94 } | |
| 95 } | |
| 96 | |
| 97 // Compare the decoding result with source data | |
| 98 int decode_total_data_len = kDefaultBufferSize - decode_avail_size; | |
| 99 EXPECT_EQ(source_len, decode_total_data_len); | |
| 100 EXPECT_EQ(memcmp(source, decode_buffer, source_len), 0); | |
| 101 } | |
| 102 | |
| 103 // Unsafe function to use filter to decode compressed data. | |
| 104 // Parameters: Source and source_len are compressed data and its size. | |
|
xunjieli
2015/12/04 15:21:19
nit: need to update variables in this comment bloc
eustas
2015/12/04 16:14:31
Done.
| |
| 105 // Dest is the buffer for decoding results. Upon entry, *dest_len is the size | |
| 106 // of the dest buffer. Upon exit, *dest_len is the number of chars written | |
| 107 // into the buffer. | |
| 108 int DecodeAllWithFilter(Filter* filter, | |
| 109 const char* source, | |
| 110 int source_len, | |
| 111 char* dest, | |
| 112 int* dest_len) { | |
| 113 memcpy(filter->stream_buffer()->data(), source, source_len); | |
| 114 filter->FlushStreamBuffer(source_len); | |
| 115 return filter->ReadData(dest, dest_len); | |
| 116 } | |
| 117 | |
| 118 void InitFilter() { | |
| 119 std::vector<Filter::FilterType> filter_types; | |
| 120 filter_types.push_back(Filter::FILTER_TYPE_BROTLI); | |
| 121 filter_.reset(Filter::Factory(filter_types, filter_context_)); | |
| 122 ASSERT_TRUE(filter_.get()); | |
| 123 ASSERT_LE(kDefaultBufferSize, filter_->stream_buffer_size()); | |
| 124 } | |
| 125 | |
| 126 void InitFilterWithBufferSize(int buffer_size) { | |
| 127 std::vector<Filter::FilterType> filter_types; | |
| 128 filter_types.push_back(Filter::FILTER_TYPE_BROTLI); | |
| 129 filter_.reset( | |
| 130 Filter::FactoryForTests(filter_types, filter_context_, buffer_size)); | |
| 131 ASSERT_TRUE(filter_.get()); | |
| 132 } | |
| 133 | |
| 134 const char* source_buffer() const { return source_buffer_.data(); } | |
| 135 int source_len() const { return static_cast<int>(source_buffer_.size()); } | |
| 136 | |
| 137 const char* encoded_buffer() const { return encoded_buffer_.data(); } | |
| 138 int encoded_len() const { return static_cast<int>(encoded_buffer_.size()); } | |
| 139 | |
| 140 scoped_ptr<Filter> filter_; | |
| 141 | |
| 142 std::string source_buffer_; | |
| 143 std::string encoded_buffer_; | |
|
xunjieli
2015/12/04 15:21:19
nit: These two variables should be private, since
eustas
2015/12/04 16:14:31
Surely. Done.
| |
| 144 | |
| 145 private: | |
| 146 MockFilterContext filter_context_; | |
| 147 }; | |
| 148 | |
| 149 // Basic scenario: decoding gzip data with big enough buffer. | |
| 150 TEST_F(BrotliUnitTest, DecodeBrotli) { | |
| 151 InitFilter(); | |
| 152 memcpy(filter_->stream_buffer()->data(), encoded_buffer(), encoded_len()); | |
| 153 filter_->FlushStreamBuffer(encoded_len()); | |
| 154 | |
| 155 char decode_buffer[kDefaultBufferSize]; | |
| 156 int decode_size = kDefaultBufferSize; | |
| 157 filter_->ReadData(decode_buffer, &decode_size); | |
| 158 | |
| 159 // Compare the decoding result with source data | |
| 160 EXPECT_EQ(source_len(), decode_size); | |
| 161 EXPECT_EQ(memcmp(source_buffer(), decode_buffer, source_len()), 0); | |
| 162 } | |
| 163 | |
| 164 // Tests we can call filter repeatedly to get all the data decoded. | |
| 165 // To do that, we create a filter with a small buffer that can not hold all | |
| 166 // the input data. | |
| 167 TEST_F(BrotliUnitTest, DecodeWithSmallBuffer) { | |
| 168 InitFilterWithBufferSize(kSmallBufferSize); | |
| 169 EXPECT_EQ(kSmallBufferSize, filter_->stream_buffer_size()); | |
| 170 DecodeAndCompareWithFilter(filter_.get(), source_buffer(), source_len(), | |
| 171 encoded_buffer(), encoded_len(), | |
| 172 kDefaultBufferSize); | |
| 173 } | |
| 174 | |
| 175 // Tests we can still decode with just 1 byte buffer in the filter. | |
| 176 // The purpose of this test: sometimes the filter will consume input without | |
| 177 // generating output. Verify filter can handle it correctly. | |
| 178 TEST_F(BrotliUnitTest, DecodeWithOneByteBuffer) { | |
| 179 InitFilterWithBufferSize(1); | |
| 180 EXPECT_EQ(1, filter_->stream_buffer_size()); | |
| 181 DecodeAndCompareWithFilter(filter_.get(), source_buffer(), source_len(), | |
| 182 encoded_buffer(), encoded_len(), | |
| 183 kDefaultBufferSize); | |
| 184 } | |
| 185 | |
| 186 // Tests we can decode when caller has small buffer to read out from filter. | |
| 187 TEST_F(BrotliUnitTest, DecodeWithSmallOutputBuffer) { | |
| 188 InitFilter(); | |
| 189 DecodeAndCompareWithFilter(filter_.get(), source_buffer(), source_len(), | |
| 190 encoded_buffer(), encoded_len(), kSmallBufferSize); | |
| 191 } | |
| 192 | |
| 193 // Tests we can still decode with just 1 byte buffer in the filter and just 1 | |
| 194 // byte buffer in the caller. | |
| 195 TEST_F(BrotliUnitTest, DecodeWithOneByteInputAndOutputBuffer) { | |
| 196 InitFilterWithBufferSize(1); | |
| 197 EXPECT_EQ(1, filter_->stream_buffer_size()); | |
| 198 DecodeAndCompareWithFilter(filter_.get(), source_buffer(), source_len(), | |
| 199 encoded_buffer(), encoded_len(), 1); | |
| 200 } | |
| 201 | |
| 202 // Decoding deflate stream with corrupted data. | |
| 203 TEST_F(BrotliUnitTest, DecodeCorruptedData) { | |
| 204 char corrupt_data[kDefaultBufferSize]; | |
| 205 int corrupt_data_len = encoded_len(); | |
| 206 memcpy(corrupt_data, encoded_buffer(), encoded_len()); | |
| 207 | |
| 208 int pos = corrupt_data_len / 2; | |
| 209 corrupt_data[pos] = !corrupt_data[pos]; | |
| 210 | |
| 211 // Decode the corrupted data with filter | |
| 212 InitFilter(); | |
| 213 char corrupt_decode_buffer[kDefaultBufferSize]; | |
| 214 int corrupt_decode_size = kDefaultBufferSize; | |
| 215 | |
| 216 int code = DecodeAllWithFilter(filter_.get(), corrupt_data, corrupt_data_len, | |
| 217 corrupt_decode_buffer, &corrupt_decode_size); | |
| 218 | |
| 219 // Expect failures | |
| 220 EXPECT_EQ(Filter::FILTER_ERROR, code); | |
| 221 } | |
| 222 | |
| 223 // Decoding deflate stream with missing data. | |
| 224 TEST_F(BrotliUnitTest, DecodeMissingData) { | |
| 225 char corrupt_data[kDefaultBufferSize]; | |
| 226 int corrupt_data_len = encoded_len(); | |
| 227 memcpy(corrupt_data, encoded_buffer(), encoded_len()); | |
| 228 | |
| 229 int pos = corrupt_data_len / 2; | |
| 230 int len = corrupt_data_len - pos - 1; | |
| 231 memmove(&corrupt_data[pos], &corrupt_data[pos + 1], len); | |
| 232 --corrupt_data_len; | |
| 233 | |
| 234 // Decode the corrupted data with filter | |
| 235 InitFilter(); | |
| 236 char corrupt_decode_buffer[kDefaultBufferSize]; | |
| 237 int corrupt_decode_size = kDefaultBufferSize; | |
| 238 | |
| 239 int code = DecodeAllWithFilter(filter_.get(), corrupt_data, corrupt_data_len, | |
| 240 corrupt_decode_buffer, &corrupt_decode_size); | |
| 241 | |
| 242 // Expect failures | |
| 243 EXPECT_EQ(Filter::FILTER_ERROR, code); | |
| 244 } | |
| 245 | |
| 246 } // namespace net | |
| OLD | NEW |