Supervised user whitelists: Cleanup
[chromium-blink-merge.git] / net / http / mock_http_cache.cc
blob02b74bc8a3a2e16359b3febaea28f401b5f2f855
1 // Copyright (c) 2012 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 "net/http/mock_http_cache.h"
7 #include "base/bind.h"
8 #include "base/message_loop/message_loop.h"
9 #include "net/base/completion_callback.h"
10 #include "net/base/net_errors.h"
11 #include "testing/gtest/include/gtest/gtest.h"
13 namespace net {
15 namespace {
17 // We can override the test mode for a given operation by setting this global
18 // variable.
19 int g_test_mode = 0;
21 int GetTestModeForEntry(const std::string& key) {
22 // 'key' is prefixed with an identifier if it corresponds to a cached POST.
23 // Skip past that to locate the actual URL.
25 // TODO(darin): It breaks the abstraction a bit that we assume 'key' is an
26 // URL corresponding to a registered MockTransaction. It would be good to
27 // have another way to access the test_mode.
28 GURL url;
29 if (isdigit(key[0])) {
30 size_t slash = key.find('/');
31 DCHECK(slash != std::string::npos);
32 url = GURL(key.substr(slash + 1));
33 } else {
34 url = GURL(key);
36 const MockTransaction* t = FindMockTransaction(url);
37 DCHECK(t);
38 return t->test_mode;
41 void CallbackForwader(const CompletionCallback& callback, int result) {
42 callback.Run(result);
45 } // namespace
47 //-----------------------------------------------------------------------------
49 struct MockDiskEntry::CallbackInfo {
50 scoped_refptr<MockDiskEntry> entry;
51 CompletionCallback callback;
52 int result;
55 MockDiskEntry::MockDiskEntry(const std::string& key)
56 : key_(key), doomed_(false), sparse_(false),
57 fail_requests_(false), fail_sparse_requests_(false), busy_(false),
58 delayed_(false) {
59 test_mode_ = GetTestModeForEntry(key);
62 void MockDiskEntry::Doom() {
63 doomed_ = true;
66 void MockDiskEntry::Close() {
67 Release();
70 std::string MockDiskEntry::GetKey() const {
71 return key_;
74 base::Time MockDiskEntry::GetLastUsed() const {
75 return base::Time::FromInternalValue(0);
78 base::Time MockDiskEntry::GetLastModified() const {
79 return base::Time::FromInternalValue(0);
82 int32 MockDiskEntry::GetDataSize(int index) const {
83 DCHECK(index >= 0 && index < kNumCacheEntryDataIndices);
84 return static_cast<int32>(data_[index].size());
87 int MockDiskEntry::ReadData(int index,
88 int offset,
89 IOBuffer* buf,
90 int buf_len,
91 const CompletionCallback& callback) {
92 DCHECK(index >= 0 && index < kNumCacheEntryDataIndices);
93 DCHECK(!callback.is_null());
95 if (fail_requests_)
96 return ERR_CACHE_READ_FAILURE;
98 if (offset < 0 || offset > static_cast<int>(data_[index].size()))
99 return ERR_FAILED;
100 if (static_cast<size_t>(offset) == data_[index].size())
101 return 0;
103 int num = std::min(buf_len, static_cast<int>(data_[index].size()) - offset);
104 memcpy(buf->data(), &data_[index][offset], num);
106 if (MockHttpCache::GetTestMode(test_mode_) & TEST_MODE_SYNC_CACHE_READ)
107 return num;
109 CallbackLater(callback, num);
110 return ERR_IO_PENDING;
113 int MockDiskEntry::WriteData(int index,
114 int offset,
115 IOBuffer* buf,
116 int buf_len,
117 const CompletionCallback& callback,
118 bool truncate) {
119 DCHECK(index >= 0 && index < kNumCacheEntryDataIndices);
120 DCHECK(!callback.is_null());
121 DCHECK(truncate);
123 if (fail_requests_) {
124 CallbackLater(callback, ERR_CACHE_READ_FAILURE);
125 return ERR_IO_PENDING;
128 if (offset < 0 || offset > static_cast<int>(data_[index].size()))
129 return ERR_FAILED;
131 data_[index].resize(offset + buf_len);
132 if (buf_len)
133 memcpy(&data_[index][offset], buf->data(), buf_len);
135 if (MockHttpCache::GetTestMode(test_mode_) & TEST_MODE_SYNC_CACHE_WRITE)
136 return buf_len;
138 CallbackLater(callback, buf_len);
139 return ERR_IO_PENDING;
142 int MockDiskEntry::ReadSparseData(int64 offset,
143 IOBuffer* buf,
144 int buf_len,
145 const CompletionCallback& callback) {
146 DCHECK(!callback.is_null());
147 if (fail_sparse_requests_)
148 return ERR_NOT_IMPLEMENTED;
149 if (!sparse_ || busy_)
150 return ERR_CACHE_OPERATION_NOT_SUPPORTED;
151 if (offset < 0)
152 return ERR_FAILED;
154 if (fail_requests_)
155 return ERR_CACHE_READ_FAILURE;
157 DCHECK(offset < kint32max);
158 int real_offset = static_cast<int>(offset);
159 if (!buf_len)
160 return 0;
162 int num = std::min(static_cast<int>(data_[1].size()) - real_offset,
163 buf_len);
164 memcpy(buf->data(), &data_[1][real_offset], num);
166 if (MockHttpCache::GetTestMode(test_mode_) & TEST_MODE_SYNC_CACHE_READ)
167 return num;
169 CallbackLater(callback, num);
170 busy_ = true;
171 delayed_ = false;
172 return ERR_IO_PENDING;
175 int MockDiskEntry::WriteSparseData(int64 offset,
176 IOBuffer* buf,
177 int buf_len,
178 const CompletionCallback& callback) {
179 DCHECK(!callback.is_null());
180 if (fail_sparse_requests_)
181 return ERR_NOT_IMPLEMENTED;
182 if (busy_)
183 return ERR_CACHE_OPERATION_NOT_SUPPORTED;
184 if (!sparse_) {
185 if (data_[1].size())
186 return ERR_CACHE_OPERATION_NOT_SUPPORTED;
187 sparse_ = true;
189 if (offset < 0)
190 return ERR_FAILED;
191 if (!buf_len)
192 return 0;
194 if (fail_requests_)
195 return ERR_CACHE_READ_FAILURE;
197 DCHECK(offset < kint32max);
198 int real_offset = static_cast<int>(offset);
200 if (static_cast<int>(data_[1].size()) < real_offset + buf_len)
201 data_[1].resize(real_offset + buf_len);
203 memcpy(&data_[1][real_offset], buf->data(), buf_len);
204 if (MockHttpCache::GetTestMode(test_mode_) & TEST_MODE_SYNC_CACHE_WRITE)
205 return buf_len;
207 CallbackLater(callback, buf_len);
208 return ERR_IO_PENDING;
211 int MockDiskEntry::GetAvailableRange(int64 offset,
212 int len,
213 int64* start,
214 const CompletionCallback& callback) {
215 DCHECK(!callback.is_null());
216 if (!sparse_ || busy_)
217 return ERR_CACHE_OPERATION_NOT_SUPPORTED;
218 if (offset < 0)
219 return ERR_FAILED;
221 if (fail_requests_)
222 return ERR_CACHE_READ_FAILURE;
224 *start = offset;
225 DCHECK(offset < kint32max);
226 int real_offset = static_cast<int>(offset);
227 if (static_cast<int>(data_[1].size()) < real_offset)
228 return 0;
230 int num = std::min(static_cast<int>(data_[1].size()) - real_offset, len);
231 int count = 0;
232 for (; num > 0; num--, real_offset++) {
233 if (!count) {
234 if (data_[1][real_offset]) {
235 count++;
236 *start = real_offset;
238 } else {
239 if (!data_[1][real_offset])
240 break;
241 count++;
244 if (MockHttpCache::GetTestMode(test_mode_) & TEST_MODE_SYNC_CACHE_WRITE)
245 return count;
247 CallbackLater(callback, count);
248 return ERR_IO_PENDING;
251 bool MockDiskEntry::CouldBeSparse() const {
252 if (fail_sparse_requests_)
253 return false;
254 return sparse_;
257 void MockDiskEntry::CancelSparseIO() {
258 cancel_ = true;
261 int MockDiskEntry::ReadyForSparseIO(const CompletionCallback& callback) {
262 if (fail_sparse_requests_)
263 return ERR_NOT_IMPLEMENTED;
264 if (!cancel_)
265 return OK;
267 cancel_ = false;
268 DCHECK(!callback.is_null());
269 if (MockHttpCache::GetTestMode(test_mode_) & TEST_MODE_SYNC_CACHE_READ)
270 return OK;
272 // The pending operation is already in the message loop (and hopefully
273 // already in the second pass). Just notify the caller that it finished.
274 CallbackLater(callback, 0);
275 return ERR_IO_PENDING;
278 // If |value| is true, don't deliver any completion callbacks until called
279 // again with |value| set to false. Caution: remember to enable callbacks
280 // again or all subsequent tests will fail.
281 // Static.
282 void MockDiskEntry::IgnoreCallbacks(bool value) {
283 if (ignore_callbacks_ == value)
284 return;
285 ignore_callbacks_ = value;
286 if (!value)
287 StoreAndDeliverCallbacks(false, NULL, CompletionCallback(), 0);
290 MockDiskEntry::~MockDiskEntry() {
293 // Unlike the callbacks for MockHttpTransaction, we want this one to run even
294 // if the consumer called Close on the MockDiskEntry. We achieve that by
295 // leveraging the fact that this class is reference counted.
296 void MockDiskEntry::CallbackLater(const CompletionCallback& callback,
297 int result) {
298 if (ignore_callbacks_)
299 return StoreAndDeliverCallbacks(true, this, callback, result);
300 base::MessageLoop::current()->PostTask(
301 FROM_HERE,
302 base::Bind(&MockDiskEntry::RunCallback, this, callback, result));
305 void MockDiskEntry::RunCallback(const CompletionCallback& callback,
306 int result) {
307 if (busy_) {
308 // This is kind of hacky, but controlling the behavior of just this entry
309 // from a test is sort of complicated. What we really want to do is
310 // delay the delivery of a sparse IO operation a little more so that the
311 // request start operation (async) will finish without seeing the end of
312 // this operation (already posted to the message loop)... and without
313 // just delaying for n mS (which may cause trouble with slow bots). So
314 // we re-post this operation (all async sparse IO operations will take two
315 // trips through the message loop instead of one).
316 if (!delayed_) {
317 delayed_ = true;
318 return CallbackLater(callback, result);
321 busy_ = false;
322 callback.Run(result);
325 // When |store| is true, stores the callback to be delivered later; otherwise
326 // delivers any callback previously stored.
327 // Static.
328 void MockDiskEntry::StoreAndDeliverCallbacks(bool store,
329 MockDiskEntry* entry,
330 const CompletionCallback& callback,
331 int result) {
332 static std::vector<CallbackInfo> callback_list;
333 if (store) {
334 CallbackInfo c = {entry, callback, result};
335 callback_list.push_back(c);
336 } else {
337 for (size_t i = 0; i < callback_list.size(); i++) {
338 CallbackInfo& c = callback_list[i];
339 c.entry->CallbackLater(c.callback, c.result);
341 callback_list.clear();
345 // Statics.
346 bool MockDiskEntry::cancel_ = false;
347 bool MockDiskEntry::ignore_callbacks_ = false;
349 //-----------------------------------------------------------------------------
351 MockDiskCache::MockDiskCache()
352 : open_count_(0), create_count_(0), fail_requests_(false),
353 soft_failures_(false), double_create_check_(true),
354 fail_sparse_requests_(false) {
357 MockDiskCache::~MockDiskCache() {
358 ReleaseAll();
361 CacheType MockDiskCache::GetCacheType() const {
362 return DISK_CACHE;
365 int32 MockDiskCache::GetEntryCount() const {
366 return static_cast<int32>(entries_.size());
369 int MockDiskCache::OpenEntry(const std::string& key,
370 disk_cache::Entry** entry,
371 const CompletionCallback& callback) {
372 DCHECK(!callback.is_null());
373 if (fail_requests_)
374 return ERR_CACHE_OPEN_FAILURE;
376 EntryMap::iterator it = entries_.find(key);
377 if (it == entries_.end())
378 return ERR_CACHE_OPEN_FAILURE;
380 if (it->second->is_doomed()) {
381 it->second->Release();
382 entries_.erase(it);
383 return ERR_CACHE_OPEN_FAILURE;
386 open_count_++;
388 it->second->AddRef();
389 *entry = it->second;
391 if (soft_failures_)
392 it->second->set_fail_requests();
394 if (GetTestModeForEntry(key) & TEST_MODE_SYNC_CACHE_START)
395 return OK;
397 CallbackLater(callback, OK);
398 return ERR_IO_PENDING;
401 int MockDiskCache::CreateEntry(const std::string& key,
402 disk_cache::Entry** entry,
403 const CompletionCallback& callback) {
404 DCHECK(!callback.is_null());
405 if (fail_requests_)
406 return ERR_CACHE_CREATE_FAILURE;
408 EntryMap::iterator it = entries_.find(key);
409 if (it != entries_.end()) {
410 if (!it->second->is_doomed()) {
411 if (double_create_check_)
412 NOTREACHED();
413 else
414 return ERR_CACHE_CREATE_FAILURE;
416 it->second->Release();
417 entries_.erase(it);
420 create_count_++;
422 MockDiskEntry* new_entry = new MockDiskEntry(key);
424 new_entry->AddRef();
425 entries_[key] = new_entry;
427 new_entry->AddRef();
428 *entry = new_entry;
430 if (soft_failures_)
431 new_entry->set_fail_requests();
433 if (fail_sparse_requests_)
434 new_entry->set_fail_sparse_requests();
436 if (GetTestModeForEntry(key) & TEST_MODE_SYNC_CACHE_START)
437 return OK;
439 CallbackLater(callback, OK);
440 return ERR_IO_PENDING;
443 int MockDiskCache::DoomEntry(const std::string& key,
444 const CompletionCallback& callback) {
445 DCHECK(!callback.is_null());
446 EntryMap::iterator it = entries_.find(key);
447 if (it != entries_.end()) {
448 it->second->Release();
449 entries_.erase(it);
452 if (GetTestModeForEntry(key) & TEST_MODE_SYNC_CACHE_START)
453 return OK;
455 CallbackLater(callback, OK);
456 return ERR_IO_PENDING;
459 int MockDiskCache::DoomAllEntries(const CompletionCallback& callback) {
460 return ERR_NOT_IMPLEMENTED;
463 int MockDiskCache::DoomEntriesBetween(const base::Time initial_time,
464 const base::Time end_time,
465 const CompletionCallback& callback) {
466 return ERR_NOT_IMPLEMENTED;
469 int MockDiskCache::DoomEntriesSince(const base::Time initial_time,
470 const CompletionCallback& callback) {
471 return ERR_NOT_IMPLEMENTED;
474 class MockDiskCache::NotImplementedIterator : public Iterator {
475 public:
476 int OpenNextEntry(disk_cache::Entry** next_entry,
477 const CompletionCallback& callback) override {
478 return ERR_NOT_IMPLEMENTED;
482 scoped_ptr<disk_cache::Backend::Iterator> MockDiskCache::CreateIterator() {
483 return scoped_ptr<Iterator>(new NotImplementedIterator());
486 void MockDiskCache::GetStats(
487 std::vector<std::pair<std::string, std::string> >* stats) {
490 void MockDiskCache::OnExternalCacheHit(const std::string& key) {
493 void MockDiskCache::ReleaseAll() {
494 EntryMap::iterator it = entries_.begin();
495 for (; it != entries_.end(); ++it)
496 it->second->Release();
497 entries_.clear();
500 void MockDiskCache::CallbackLater(const CompletionCallback& callback,
501 int result) {
502 base::MessageLoop::current()->PostTask(
503 FROM_HERE, base::Bind(&CallbackForwader, callback, result));
506 //-----------------------------------------------------------------------------
508 int MockBackendFactory::CreateBackend(NetLog* net_log,
509 scoped_ptr<disk_cache::Backend>* backend,
510 const CompletionCallback& callback) {
511 backend->reset(new MockDiskCache());
512 return OK;
515 //-----------------------------------------------------------------------------
517 MockHttpCache::MockHttpCache()
518 : http_cache_(new MockNetworkLayer(), NULL, new MockBackendFactory()) {
521 MockHttpCache::MockHttpCache(HttpCache::BackendFactory* disk_cache_factory)
522 : http_cache_(new MockNetworkLayer(), NULL, disk_cache_factory) {
525 disk_cache::Backend* MockHttpCache::backend() {
526 TestCompletionCallback cb;
527 disk_cache::Backend* backend;
528 int rv = http_cache_.GetBackend(&backend, cb.callback());
529 rv = cb.GetResult(rv);
530 return (rv == OK) ? backend : NULL;
533 MockDiskCache* MockHttpCache::disk_cache() {
534 return static_cast<MockDiskCache*>(backend());
537 int MockHttpCache::CreateTransaction(scoped_ptr<HttpTransaction>* trans) {
538 return http_cache_.CreateTransaction(DEFAULT_PRIORITY, trans);
541 void MockHttpCache::BypassCacheLock() {
542 http_cache_.BypassLockForTest();
545 void MockHttpCache::FailConditionalizations() {
546 http_cache_.FailConditionalizationForTest();
549 bool MockHttpCache::ReadResponseInfo(disk_cache::Entry* disk_entry,
550 HttpResponseInfo* response_info,
551 bool* response_truncated) {
552 int size = disk_entry->GetDataSize(0);
554 TestCompletionCallback cb;
555 scoped_refptr<IOBuffer> buffer(new IOBuffer(size));
556 int rv = disk_entry->ReadData(0, 0, buffer.get(), size, cb.callback());
557 rv = cb.GetResult(rv);
558 EXPECT_EQ(size, rv);
560 return HttpCache::ParseResponseInfo(buffer->data(), size, response_info,
561 response_truncated);
564 bool MockHttpCache::WriteResponseInfo(disk_cache::Entry* disk_entry,
565 const HttpResponseInfo* response_info,
566 bool skip_transient_headers,
567 bool response_truncated) {
568 Pickle pickle;
569 response_info->Persist(
570 &pickle, skip_transient_headers, response_truncated);
572 TestCompletionCallback cb;
573 scoped_refptr<WrappedIOBuffer> data(
574 new WrappedIOBuffer(reinterpret_cast<const char*>(pickle.data())));
575 int len = static_cast<int>(pickle.size());
577 int rv = disk_entry->WriteData(0, 0, data.get(), len, cb.callback(), true);
578 rv = cb.GetResult(rv);
579 return (rv == len);
582 bool MockHttpCache::OpenBackendEntry(const std::string& key,
583 disk_cache::Entry** entry) {
584 TestCompletionCallback cb;
585 int rv = backend()->OpenEntry(key, entry, cb.callback());
586 return (cb.GetResult(rv) == OK);
589 bool MockHttpCache::CreateBackendEntry(const std::string& key,
590 disk_cache::Entry** entry,
591 NetLog* net_log) {
592 TestCompletionCallback cb;
593 int rv = backend()->CreateEntry(key, entry, cb.callback());
594 return (cb.GetResult(rv) == OK);
597 // Static.
598 int MockHttpCache::GetTestMode(int test_mode) {
599 if (!g_test_mode)
600 return test_mode;
602 return g_test_mode;
605 // Static.
606 void MockHttpCache::SetTestMode(int test_mode) {
607 g_test_mode = test_mode;
610 //-----------------------------------------------------------------------------
612 int MockDiskCacheNoCB::CreateEntry(const std::string& key,
613 disk_cache::Entry** entry,
614 const CompletionCallback& callback) {
615 return ERR_IO_PENDING;
618 //-----------------------------------------------------------------------------
620 int MockBackendNoCbFactory::CreateBackend(
621 NetLog* net_log,
622 scoped_ptr<disk_cache::Backend>* backend,
623 const CompletionCallback& callback) {
624 backend->reset(new MockDiskCacheNoCB());
625 return OK;
628 //-----------------------------------------------------------------------------
630 MockBlockingBackendFactory::MockBlockingBackendFactory()
631 : backend_(NULL),
632 block_(true),
633 fail_(false) {
636 MockBlockingBackendFactory::~MockBlockingBackendFactory() {
639 int MockBlockingBackendFactory::CreateBackend(
640 NetLog* net_log,
641 scoped_ptr<disk_cache::Backend>* backend,
642 const CompletionCallback& callback) {
643 if (!block_) {
644 if (!fail_)
645 backend->reset(new MockDiskCache());
646 return Result();
649 backend_ = backend;
650 callback_ = callback;
651 return ERR_IO_PENDING;
654 void MockBlockingBackendFactory::FinishCreation() {
655 block_ = false;
656 if (!callback_.is_null()) {
657 if (!fail_)
658 backend_->reset(new MockDiskCache());
659 CompletionCallback cb = callback_;
660 callback_.Reset();
661 cb.Run(Result()); // This object can be deleted here.
665 } // namespace net