| OLD | NEW |
| (Empty) |
| 1 // Copyright 2014 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 <stdint.h> | |
| 6 | |
| 7 #include <map> | |
| 8 | |
| 9 #include "base/bind.h" | |
| 10 #include "base/files/file_path.h" | |
| 11 #include "base/location.h" | |
| 12 #include "base/run_loop.h" | |
| 13 #include "base/single_thread_task_runner.h" | |
| 14 #include "base/strings/utf_string_conversions.h" | |
| 15 #include "base/threading/thread_task_runner_handle.h" | |
| 16 #include "net/base/completion_callback.h" | |
| 17 #include "net/base/net_errors.h" | |
| 18 #include "storage/browser/database/database_quota_client.h" | |
| 19 #include "storage/browser/database/database_tracker.h" | |
| 20 #include "storage/browser/database/database_util.h" | |
| 21 #include "storage/common/database/database_identifier.h" | |
| 22 #include "testing/gtest/include/gtest/gtest.h" | |
| 23 | |
| 24 using storage::DatabaseQuotaClient; | |
| 25 using storage::DatabaseTracker; | |
| 26 using storage::OriginInfo; | |
| 27 | |
| 28 namespace content { | |
| 29 | |
| 30 // Declared to shorten the line lengths. | |
| 31 static const storage::StorageType kTemp = storage::kStorageTypeTemporary; | |
| 32 static const storage::StorageType kPerm = storage::kStorageTypePersistent; | |
| 33 | |
| 34 // Mock tracker class the mocks up those methods of the tracker | |
| 35 // that are used by the QuotaClient. | |
| 36 class MockDatabaseTracker : public DatabaseTracker { | |
| 37 public: | |
| 38 MockDatabaseTracker() | |
| 39 : DatabaseTracker(base::FilePath(), false, NULL, NULL, NULL), | |
| 40 delete_called_count_(0), | |
| 41 async_delete_(false) {} | |
| 42 | |
| 43 bool GetOriginInfo(const std::string& origin_identifier, | |
| 44 OriginInfo* info) override { | |
| 45 std::map<GURL, MockOriginInfo>::const_iterator found = | |
| 46 mock_origin_infos_.find( | |
| 47 storage::GetOriginFromIdentifier(origin_identifier)); | |
| 48 if (found == mock_origin_infos_.end()) | |
| 49 return false; | |
| 50 *info = OriginInfo(found->second); | |
| 51 return true; | |
| 52 } | |
| 53 | |
| 54 bool GetAllOriginIdentifiers( | |
| 55 std::vector<std::string>* origins_identifiers) override { | |
| 56 std::map<GURL, MockOriginInfo>::const_iterator iter; | |
| 57 for (iter = mock_origin_infos_.begin(); | |
| 58 iter != mock_origin_infos_.end(); | |
| 59 ++iter) { | |
| 60 origins_identifiers->push_back(iter->second.GetOriginIdentifier()); | |
| 61 } | |
| 62 return true; | |
| 63 } | |
| 64 | |
| 65 bool GetAllOriginsInfo(std::vector<OriginInfo>* origins_info) override { | |
| 66 std::map<GURL, MockOriginInfo>::const_iterator iter; | |
| 67 for (iter = mock_origin_infos_.begin(); | |
| 68 iter != mock_origin_infos_.end(); | |
| 69 ++iter) { | |
| 70 origins_info->push_back(OriginInfo(iter->second)); | |
| 71 } | |
| 72 return true; | |
| 73 } | |
| 74 | |
| 75 int DeleteDataForOrigin(const std::string& origin_identifier, | |
| 76 const net::CompletionCallback& callback) override { | |
| 77 ++delete_called_count_; | |
| 78 if (async_delete()) { | |
| 79 base::ThreadTaskRunnerHandle::Get()->PostTask( | |
| 80 FROM_HERE, base::Bind(&MockDatabaseTracker::AsyncDeleteDataForOrigin, | |
| 81 this, callback)); | |
| 82 return net::ERR_IO_PENDING; | |
| 83 } | |
| 84 return net::OK; | |
| 85 } | |
| 86 | |
| 87 void AsyncDeleteDataForOrigin(const net::CompletionCallback& callback) { | |
| 88 callback.Run(net::OK); | |
| 89 } | |
| 90 | |
| 91 void AddMockDatabase(const GURL& origin, const char* name, int size) { | |
| 92 MockOriginInfo& info = mock_origin_infos_[origin]; | |
| 93 info.set_origin(storage::GetIdentifierFromOrigin(origin)); | |
| 94 info.AddMockDatabase(base::ASCIIToUTF16(name), size); | |
| 95 } | |
| 96 | |
| 97 int delete_called_count() { return delete_called_count_; } | |
| 98 bool async_delete() { return async_delete_; } | |
| 99 void set_async_delete(bool async) { async_delete_ = async; } | |
| 100 | |
| 101 protected: | |
| 102 ~MockDatabaseTracker() override {} | |
| 103 | |
| 104 private: | |
| 105 class MockOriginInfo : public OriginInfo { | |
| 106 public: | |
| 107 void set_origin(const std::string& origin_identifier) { | |
| 108 origin_identifier_ = origin_identifier; | |
| 109 } | |
| 110 | |
| 111 void AddMockDatabase(const base::string16& name, int size) { | |
| 112 EXPECT_TRUE(database_info_.find(name) == database_info_.end()); | |
| 113 database_info_[name].first = size; | |
| 114 total_size_ += size; | |
| 115 } | |
| 116 }; | |
| 117 | |
| 118 int delete_called_count_; | |
| 119 bool async_delete_; | |
| 120 std::map<GURL, MockOriginInfo> mock_origin_infos_; | |
| 121 }; | |
| 122 | |
| 123 | |
| 124 // Base class for our test fixtures. | |
| 125 class DatabaseQuotaClientTest : public testing::Test { | |
| 126 public: | |
| 127 const GURL kOriginA; | |
| 128 const GURL kOriginB; | |
| 129 const GURL kOriginOther; | |
| 130 | |
| 131 DatabaseQuotaClientTest() | |
| 132 : kOriginA("http://host"), | |
| 133 kOriginB("http://host:8000"), | |
| 134 kOriginOther("http://other"), | |
| 135 usage_(0), | |
| 136 mock_tracker_(new MockDatabaseTracker), | |
| 137 weak_factory_(this) { | |
| 138 } | |
| 139 | |
| 140 int64_t GetOriginUsage(storage::QuotaClient* client, | |
| 141 const GURL& origin, | |
| 142 storage::StorageType type) { | |
| 143 usage_ = 0; | |
| 144 client->GetOriginUsage( | |
| 145 origin, type, | |
| 146 base::Bind(&DatabaseQuotaClientTest::OnGetOriginUsageComplete, | |
| 147 weak_factory_.GetWeakPtr())); | |
| 148 base::RunLoop().RunUntilIdle(); | |
| 149 return usage_; | |
| 150 } | |
| 151 | |
| 152 const std::set<GURL>& GetOriginsForType(storage::QuotaClient* client, | |
| 153 storage::StorageType type) { | |
| 154 origins_.clear(); | |
| 155 client->GetOriginsForType( | |
| 156 type, | |
| 157 base::Bind(&DatabaseQuotaClientTest::OnGetOriginsComplete, | |
| 158 weak_factory_.GetWeakPtr())); | |
| 159 base::RunLoop().RunUntilIdle(); | |
| 160 return origins_; | |
| 161 } | |
| 162 | |
| 163 const std::set<GURL>& GetOriginsForHost(storage::QuotaClient* client, | |
| 164 storage::StorageType type, | |
| 165 const std::string& host) { | |
| 166 origins_.clear(); | |
| 167 client->GetOriginsForHost( | |
| 168 type, host, | |
| 169 base::Bind(&DatabaseQuotaClientTest::OnGetOriginsComplete, | |
| 170 weak_factory_.GetWeakPtr())); | |
| 171 base::RunLoop().RunUntilIdle(); | |
| 172 return origins_; | |
| 173 } | |
| 174 | |
| 175 bool DeleteOriginData(storage::QuotaClient* client, | |
| 176 storage::StorageType type, | |
| 177 const GURL& origin) { | |
| 178 delete_status_ = storage::kQuotaStatusUnknown; | |
| 179 client->DeleteOriginData( | |
| 180 origin, type, | |
| 181 base::Bind(&DatabaseQuotaClientTest::OnDeleteOriginDataComplete, | |
| 182 weak_factory_.GetWeakPtr())); | |
| 183 base::RunLoop().RunUntilIdle(); | |
| 184 return delete_status_ == storage::kQuotaStatusOk; | |
| 185 } | |
| 186 | |
| 187 MockDatabaseTracker* mock_tracker() { return mock_tracker_.get(); } | |
| 188 | |
| 189 | |
| 190 private: | |
| 191 void OnGetOriginUsageComplete(int64_t usage) { usage_ = usage; } | |
| 192 | |
| 193 void OnGetOriginsComplete(const std::set<GURL>& origins) { | |
| 194 origins_ = origins; | |
| 195 } | |
| 196 | |
| 197 void OnDeleteOriginDataComplete(storage::QuotaStatusCode status) { | |
| 198 delete_status_ = status; | |
| 199 } | |
| 200 | |
| 201 base::MessageLoop message_loop_; | |
| 202 int64_t usage_; | |
| 203 std::set<GURL> origins_; | |
| 204 storage::QuotaStatusCode delete_status_; | |
| 205 scoped_refptr<MockDatabaseTracker> mock_tracker_; | |
| 206 base::WeakPtrFactory<DatabaseQuotaClientTest> weak_factory_; | |
| 207 }; | |
| 208 | |
| 209 | |
| 210 TEST_F(DatabaseQuotaClientTest, GetOriginUsage) { | |
| 211 DatabaseQuotaClient client(base::ThreadTaskRunnerHandle::Get().get(), | |
| 212 mock_tracker()); | |
| 213 | |
| 214 EXPECT_EQ(0, GetOriginUsage(&client, kOriginA, kTemp)); | |
| 215 EXPECT_EQ(0, GetOriginUsage(&client, kOriginA, kPerm)); | |
| 216 | |
| 217 mock_tracker()->AddMockDatabase(kOriginA, "fooDB", 1000); | |
| 218 EXPECT_EQ(1000, GetOriginUsage(&client, kOriginA, kTemp)); | |
| 219 EXPECT_EQ(0, GetOriginUsage(&client, kOriginA, kPerm)); | |
| 220 | |
| 221 EXPECT_EQ(0, GetOriginUsage(&client, kOriginB, kPerm)); | |
| 222 EXPECT_EQ(0, GetOriginUsage(&client, kOriginB, kTemp)); | |
| 223 } | |
| 224 | |
| 225 TEST_F(DatabaseQuotaClientTest, GetOriginsForHost) { | |
| 226 DatabaseQuotaClient client(base::ThreadTaskRunnerHandle::Get().get(), | |
| 227 mock_tracker()); | |
| 228 | |
| 229 EXPECT_EQ(kOriginA.host(), kOriginB.host()); | |
| 230 EXPECT_NE(kOriginA.host(), kOriginOther.host()); | |
| 231 | |
| 232 std::set<GURL> origins = GetOriginsForHost(&client, kTemp, kOriginA.host()); | |
| 233 EXPECT_TRUE(origins.empty()); | |
| 234 | |
| 235 mock_tracker()->AddMockDatabase(kOriginA, "fooDB", 1000); | |
| 236 origins = GetOriginsForHost(&client, kTemp, kOriginA.host()); | |
| 237 EXPECT_EQ(origins.size(), 1ul); | |
| 238 EXPECT_TRUE(origins.find(kOriginA) != origins.end()); | |
| 239 | |
| 240 mock_tracker()->AddMockDatabase(kOriginB, "barDB", 1000); | |
| 241 origins = GetOriginsForHost(&client, kTemp, kOriginA.host()); | |
| 242 EXPECT_EQ(origins.size(), 2ul); | |
| 243 EXPECT_TRUE(origins.find(kOriginA) != origins.end()); | |
| 244 EXPECT_TRUE(origins.find(kOriginB) != origins.end()); | |
| 245 | |
| 246 EXPECT_TRUE(GetOriginsForHost(&client, kPerm, kOriginA.host()).empty()); | |
| 247 EXPECT_TRUE(GetOriginsForHost(&client, kTemp, kOriginOther.host()).empty()); | |
| 248 } | |
| 249 | |
| 250 TEST_F(DatabaseQuotaClientTest, GetOriginsForType) { | |
| 251 DatabaseQuotaClient client(base::ThreadTaskRunnerHandle::Get().get(), | |
| 252 mock_tracker()); | |
| 253 | |
| 254 EXPECT_TRUE(GetOriginsForType(&client, kTemp).empty()); | |
| 255 EXPECT_TRUE(GetOriginsForType(&client, kPerm).empty()); | |
| 256 | |
| 257 mock_tracker()->AddMockDatabase(kOriginA, "fooDB", 1000); | |
| 258 std::set<GURL> origins = GetOriginsForType(&client, kTemp); | |
| 259 EXPECT_EQ(origins.size(), 1ul); | |
| 260 EXPECT_TRUE(origins.find(kOriginA) != origins.end()); | |
| 261 | |
| 262 EXPECT_TRUE(GetOriginsForType(&client, kPerm).empty()); | |
| 263 } | |
| 264 | |
| 265 TEST_F(DatabaseQuotaClientTest, DeleteOriginData) { | |
| 266 DatabaseQuotaClient client(base::ThreadTaskRunnerHandle::Get().get(), | |
| 267 mock_tracker()); | |
| 268 | |
| 269 // Perm deletions are short circuited in the Client and | |
| 270 // should not reach the DatabaseTracker. | |
| 271 EXPECT_TRUE(DeleteOriginData(&client, kPerm, kOriginA)); | |
| 272 EXPECT_EQ(0, mock_tracker()->delete_called_count()); | |
| 273 | |
| 274 mock_tracker()->set_async_delete(false); | |
| 275 EXPECT_TRUE(DeleteOriginData(&client, kTemp, kOriginA)); | |
| 276 EXPECT_EQ(1, mock_tracker()->delete_called_count()); | |
| 277 | |
| 278 mock_tracker()->set_async_delete(true); | |
| 279 EXPECT_TRUE(DeleteOriginData(&client, kTemp, kOriginA)); | |
| 280 EXPECT_EQ(2, mock_tracker()->delete_called_count()); | |
| 281 } | |
| 282 | |
| 283 } // namespace content | |
| OLD | NEW |