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
3 // found in the LICENSE file.
5 #include "content/browser/loader/upload_data_stream_builder.h"
10 #include "base/files/file_path.h"
11 #include "base/files/file_util.h"
12 #include "base/message_loop/message_loop.h"
13 #include "base/message_loop/message_loop_proxy.h"
14 #include "base/run_loop.h"
15 #include "base/time/time.h"
16 #include "content/common/resource_request_body.h"
17 #include "net/base/io_buffer.h"
18 #include "net/base/net_errors.h"
19 #include "net/base/test_completion_callback.h"
20 #include "net/base/upload_bytes_element_reader.h"
21 #include "net/base/upload_data_stream.h"
22 #include "net/base/upload_file_element_reader.h"
23 #include "storage/browser/blob/blob_data_builder.h"
24 #include "storage/browser/blob/blob_storage_context.h"
25 #include "testing/gtest/include/gtest/gtest.h"
28 using storage::BlobDataBuilder
;
29 using storage::BlobDataHandle
;
30 using storage::BlobStorageContext
;
35 bool AreElementsEqual(const net::UploadElementReader
& reader
,
36 const ResourceRequestBody::Element
& element
) {
37 switch(element
.type()) {
38 case ResourceRequestBody::Element::TYPE_BYTES
: {
39 const net::UploadBytesElementReader
* bytes_reader
=
40 reader
.AsBytesReader();
41 return bytes_reader
&&
42 element
.length() == bytes_reader
->length() &&
43 std::equal(element
.bytes(), element
.bytes() + element
.length(),
44 bytes_reader
->bytes());
46 case ResourceRequestBody::Element::TYPE_FILE
: {
47 const net::UploadFileElementReader
* file_reader
= reader
.AsFileReader();
49 file_reader
->path() == element
.path() &&
50 file_reader
->range_offset() == element
.offset() &&
51 file_reader
->range_length() == element
.length() &&
52 file_reader
->expected_modification_time() ==
53 element
.expected_modification_time();
64 TEST(UploadDataStreamBuilderTest
, CreateUploadDataStreamWithoutBlob
) {
65 base::MessageLoop message_loop
;
66 scoped_refptr
<ResourceRequestBody
> request_body
= new ResourceRequestBody
;
68 const char kData
[] = "123";
69 const base::FilePath::StringType kFilePath
= FILE_PATH_LITERAL("abc");
70 const uint64 kFileOffset
= 10U;
71 const uint64 kFileLength
= 100U;
72 const base::Time kFileTime
= base::Time::FromDoubleT(999);
73 const int64 kIdentifier
= 12345;
75 request_body
->AppendBytes(kData
, arraysize(kData
) - 1);
76 request_body
->AppendFileRange(base::FilePath(kFilePath
),
77 kFileOffset
, kFileLength
, kFileTime
);
78 request_body
->set_identifier(kIdentifier
);
80 scoped_ptr
<net::UploadDataStream
> upload(UploadDataStreamBuilder::Build(
81 request_body
.get(), NULL
, NULL
, base::MessageLoopProxy::current().get()));
83 EXPECT_EQ(kIdentifier
, upload
->identifier());
84 ASSERT_TRUE(upload
->GetElementReaders());
85 ASSERT_EQ(request_body
->elements()->size(),
86 upload
->GetElementReaders()->size());
88 const net::UploadBytesElementReader
* r1
=
89 (*upload
->GetElementReaders())[0]->AsBytesReader();
91 EXPECT_EQ(kData
, std::string(r1
->bytes(), r1
->length()));
93 const net::UploadFileElementReader
* r2
=
94 (*upload
->GetElementReaders())[1]->AsFileReader();
96 EXPECT_EQ(kFilePath
, r2
->path().value());
97 EXPECT_EQ(kFileOffset
, r2
->range_offset());
98 EXPECT_EQ(kFileLength
, r2
->range_length());
99 EXPECT_EQ(kFileTime
, r2
->expected_modification_time());
102 TEST(UploadDataStreamBuilderTest
, ResolveBlobAndCreateUploadDataStream
) {
103 base::MessageLoop message_loop
;
105 // Setup blob data for testing.
106 base::Time time1
, time2
;
107 base::Time::FromString("Tue, 15 Nov 1994, 12:45:26 GMT", &time1
);
108 base::Time::FromString("Mon, 14 Nov 1994, 11:30:49 GMT", &time2
);
110 BlobStorageContext blob_storage_context
;
112 const std::string
blob_id0("id-0");
113 scoped_ptr
<BlobDataBuilder
> blob_data_builder(
114 new BlobDataBuilder(blob_id0
));
115 scoped_ptr
<BlobDataHandle
> handle1
=
116 blob_storage_context
.AddFinishedBlob(blob_data_builder
.get());
118 const std::string
blob_id1("id-1");
119 const std::string kBlobData
= "BlobData";
120 blob_data_builder
.reset(new BlobDataBuilder(blob_id1
));
121 blob_data_builder
->AppendData(kBlobData
);
122 blob_data_builder
->AppendFile(
123 base::FilePath(FILE_PATH_LITERAL("BlobFile.txt")), 0, 20, time1
);
124 scoped_ptr
<BlobDataHandle
> handle2
=
125 blob_storage_context
.AddFinishedBlob(blob_data_builder
.get());
127 // Setup upload data elements for comparison.
128 ResourceRequestBody::Element blob_element1
, blob_element2
;
129 blob_element1
.SetToBytes(kBlobData
.c_str(), kBlobData
.size());
130 blob_element2
.SetToFilePathRange(
131 base::FilePath(FILE_PATH_LITERAL("BlobFile.txt")), 0, 20, time1
);
133 ResourceRequestBody::Element upload_element1
, upload_element2
;
134 upload_element1
.SetToBytes("Hello", 5);
135 upload_element2
.SetToFilePathRange(
136 base::FilePath(FILE_PATH_LITERAL("foo1.txt")), 0, 20, time2
);
138 // Test no blob reference.
139 scoped_refptr
<ResourceRequestBody
> request_body(new ResourceRequestBody());
140 request_body
->AppendBytes(
141 upload_element1
.bytes(),
142 upload_element1
.length());
143 request_body
->AppendFileRange(
144 upload_element2
.path(),
145 upload_element2
.offset(),
146 upload_element2
.length(),
147 upload_element2
.expected_modification_time());
149 scoped_ptr
<net::UploadDataStream
> upload(
150 UploadDataStreamBuilder::Build(
152 &blob_storage_context
,
154 base::MessageLoopProxy::current().get()));
156 ASSERT_TRUE(upload
->GetElementReaders());
157 ASSERT_EQ(2U, upload
->GetElementReaders()->size());
158 EXPECT_TRUE(AreElementsEqual(
159 *(*upload
->GetElementReaders())[0], upload_element1
));
160 EXPECT_TRUE(AreElementsEqual(
161 *(*upload
->GetElementReaders())[1], upload_element2
));
163 // Test having only one blob reference that refers to empty blob data.
164 request_body
= new ResourceRequestBody();
165 request_body
->AppendBlob(blob_id0
);
167 upload
= UploadDataStreamBuilder::Build(
169 &blob_storage_context
,
171 base::MessageLoopProxy::current().get());
172 ASSERT_TRUE(upload
->GetElementReaders());
173 ASSERT_EQ(0U, upload
->GetElementReaders()->size());
175 // Test having only one blob reference.
176 request_body
= new ResourceRequestBody();
177 request_body
->AppendBlob(blob_id1
);
179 upload
= UploadDataStreamBuilder::Build(
181 &blob_storage_context
,
183 base::MessageLoopProxy::current().get());
184 ASSERT_TRUE(upload
->GetElementReaders());
185 ASSERT_EQ(2U, upload
->GetElementReaders()->size());
186 EXPECT_TRUE(AreElementsEqual(
187 *(*upload
->GetElementReaders())[0], blob_element1
));
188 EXPECT_TRUE(AreElementsEqual(
189 *(*upload
->GetElementReaders())[1], blob_element2
));
191 // Test having one blob reference at the beginning.
192 request_body
= new ResourceRequestBody();
193 request_body
->AppendBlob(blob_id1
);
194 request_body
->AppendBytes(
195 upload_element1
.bytes(),
196 upload_element1
.length());
197 request_body
->AppendFileRange(
198 upload_element2
.path(),
199 upload_element2
.offset(),
200 upload_element2
.length(),
201 upload_element2
.expected_modification_time());
203 upload
= UploadDataStreamBuilder::Build(
205 &blob_storage_context
,
207 base::MessageLoopProxy::current().get());
208 ASSERT_TRUE(upload
->GetElementReaders());
209 ASSERT_EQ(4U, upload
->GetElementReaders()->size());
210 EXPECT_TRUE(AreElementsEqual(
211 *(*upload
->GetElementReaders())[0], blob_element1
));
212 EXPECT_TRUE(AreElementsEqual(
213 *(*upload
->GetElementReaders())[1], blob_element2
));
214 EXPECT_TRUE(AreElementsEqual(
215 *(*upload
->GetElementReaders())[2], upload_element1
));
216 EXPECT_TRUE(AreElementsEqual(
217 *(*upload
->GetElementReaders())[3], upload_element2
));
219 // Test having one blob reference at the end.
220 request_body
= new ResourceRequestBody();
221 request_body
->AppendBytes(
222 upload_element1
.bytes(),
223 upload_element1
.length());
224 request_body
->AppendFileRange(
225 upload_element2
.path(),
226 upload_element2
.offset(),
227 upload_element2
.length(),
228 upload_element2
.expected_modification_time());
229 request_body
->AppendBlob(blob_id1
);
232 UploadDataStreamBuilder::Build(request_body
.get(),
233 &blob_storage_context
,
235 base::MessageLoopProxy::current().get());
236 ASSERT_TRUE(upload
->GetElementReaders());
237 ASSERT_EQ(4U, upload
->GetElementReaders()->size());
238 EXPECT_TRUE(AreElementsEqual(
239 *(*upload
->GetElementReaders())[0], upload_element1
));
240 EXPECT_TRUE(AreElementsEqual(
241 *(*upload
->GetElementReaders())[1], upload_element2
));
242 EXPECT_TRUE(AreElementsEqual(
243 *(*upload
->GetElementReaders())[2], blob_element1
));
244 EXPECT_TRUE(AreElementsEqual(
245 *(*upload
->GetElementReaders())[3], blob_element2
));
247 // Test having one blob reference in the middle.
248 request_body
= new ResourceRequestBody();
249 request_body
->AppendBytes(
250 upload_element1
.bytes(),
251 upload_element1
.length());
252 request_body
->AppendBlob(blob_id1
);
253 request_body
->AppendFileRange(
254 upload_element2
.path(),
255 upload_element2
.offset(),
256 upload_element2
.length(),
257 upload_element2
.expected_modification_time());
259 upload
= UploadDataStreamBuilder::Build(
261 &blob_storage_context
,
263 base::MessageLoopProxy::current().get());
264 ASSERT_TRUE(upload
->GetElementReaders());
265 ASSERT_EQ(4U, upload
->GetElementReaders()->size());
266 EXPECT_TRUE(AreElementsEqual(
267 *(*upload
->GetElementReaders())[0], upload_element1
));
268 EXPECT_TRUE(AreElementsEqual(
269 *(*upload
->GetElementReaders())[1], blob_element1
));
270 EXPECT_TRUE(AreElementsEqual(
271 *(*upload
->GetElementReaders())[2], blob_element2
));
272 EXPECT_TRUE(AreElementsEqual(
273 *(*upload
->GetElementReaders())[3], upload_element2
));
275 // Test having multiple blob references.
276 request_body
= new ResourceRequestBody();
277 request_body
->AppendBlob(blob_id1
);
278 request_body
->AppendBytes(
279 upload_element1
.bytes(),
280 upload_element1
.length());
281 request_body
->AppendBlob(blob_id1
);
282 request_body
->AppendBlob(blob_id1
);
283 request_body
->AppendFileRange(
284 upload_element2
.path(),
285 upload_element2
.offset(),
286 upload_element2
.length(),
287 upload_element2
.expected_modification_time());
289 upload
= UploadDataStreamBuilder::Build(
291 &blob_storage_context
,
293 base::MessageLoopProxy::current().get());
294 ASSERT_TRUE(upload
->GetElementReaders());
295 ASSERT_EQ(8U, upload
->GetElementReaders()->size());
296 EXPECT_TRUE(AreElementsEqual(
297 *(*upload
->GetElementReaders())[0], blob_element1
));
298 EXPECT_TRUE(AreElementsEqual(
299 *(*upload
->GetElementReaders())[1], blob_element2
));
300 EXPECT_TRUE(AreElementsEqual(
301 *(*upload
->GetElementReaders())[2], upload_element1
));
302 EXPECT_TRUE(AreElementsEqual(
303 *(*upload
->GetElementReaders())[3], blob_element1
));
304 EXPECT_TRUE(AreElementsEqual(
305 *(*upload
->GetElementReaders())[4], blob_element2
));
306 EXPECT_TRUE(AreElementsEqual(
307 *(*upload
->GetElementReaders())[5], blob_element1
));
308 EXPECT_TRUE(AreElementsEqual(
309 *(*upload
->GetElementReaders())[6], blob_element2
));
310 EXPECT_TRUE(AreElementsEqual(
311 *(*upload
->GetElementReaders())[7], upload_element2
));
313 // Clean up for ASAN.
314 base::RunLoop().RunUntilIdle();
317 TEST(UploadDataStreamBuilderTest
,
318 WriteUploadDataStreamWithEmptyFileBackedBlob
) {
319 base::MessageLoopForIO message_loop
;
321 base::FilePath test_blob_path
;
322 ASSERT_TRUE(base::CreateTemporaryFile(&test_blob_path
));
324 const uint64_t kZeroLength
= 0;
325 base::Time blob_time
;
326 base::Time::FromString("Tue, 15 Nov 1994, 12:45:26 GMT", &blob_time
);
327 ASSERT_TRUE(base::TouchFile(test_blob_path
, blob_time
, blob_time
));
329 BlobStorageContext blob_storage_context
;
331 // A blob created from an empty file added several times.
332 const std::string
blob_id("id-0");
333 scoped_ptr
<BlobDataBuilder
> blob_data_builder(new BlobDataBuilder(blob_id
));
334 blob_data_builder
->AppendFile(test_blob_path
, 0, kZeroLength
, blob_time
);
335 scoped_ptr
<BlobDataHandle
> handle
=
336 blob_storage_context
.AddFinishedBlob(blob_data_builder
.get());
338 ResourceRequestBody::Element blob_element
;
339 blob_element
.SetToFilePathRange(test_blob_path
, 0, kZeroLength
, blob_time
);
341 scoped_refptr
<ResourceRequestBody
> request_body(new ResourceRequestBody());
342 scoped_ptr
<net::UploadDataStream
> upload(UploadDataStreamBuilder::Build(
343 request_body
.get(), &blob_storage_context
, NULL
,
344 base::MessageLoopProxy::current().get()));
346 request_body
= new ResourceRequestBody();
347 request_body
->AppendBlob(blob_id
);
348 request_body
->AppendBlob(blob_id
);
349 request_body
->AppendBlob(blob_id
);
351 upload
= UploadDataStreamBuilder::Build(
352 request_body
.get(), &blob_storage_context
, NULL
,
353 base::MessageLoopProxy::current().get());
354 ASSERT_TRUE(upload
->GetElementReaders());
355 const auto& readers
= *upload
->GetElementReaders();
356 ASSERT_EQ(3U, readers
.size());
357 EXPECT_TRUE(AreElementsEqual(*readers
[0], blob_element
));
358 EXPECT_TRUE(AreElementsEqual(*readers
[1], blob_element
));
359 EXPECT_TRUE(AreElementsEqual(*readers
[2], blob_element
));
361 net::TestCompletionCallback init_callback
;
362 ASSERT_EQ(net::ERR_IO_PENDING
, upload
->Init(init_callback
.callback()));
363 EXPECT_EQ(net::OK
, init_callback
.WaitForResult());
365 EXPECT_EQ(kZeroLength
, upload
->size());
367 // Purposely (try to) read more than what is in the stream. If we try to
368 // read zero bytes then UploadDataStream::Read will fail a DCHECK.
369 int kBufferLength
= kZeroLength
+ 1;
370 scoped_ptr
<char[]> buffer(new char[kBufferLength
]);
371 scoped_refptr
<net::IOBuffer
> io_buffer
=
372 new net::WrappedIOBuffer(buffer
.get());
373 net::TestCompletionCallback read_callback
;
375 upload
->Read(io_buffer
.get(), kBufferLength
, read_callback
.callback());
376 EXPECT_EQ(static_cast<int>(kZeroLength
), read_callback
.GetResult(result
));
378 base::DeleteFile(test_blob_path
, false);
380 // Clean up for ASAN.
381 base::RunLoop().RunUntilIdle();
383 } // namespace content