1 // Copyright (c) 2011 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 <map>
6
7 #include "base/bind.h"
8 #include "base/file_util.h"
9 #include "base/files/file_path.h"
10 #include "base/files/scoped_temp_dir.h"
11 #include "base/message_loop/message_loop.h"
12 #include "base/test/test_simple_task_runner.h"
13 #include "base/threading/thread.h"
14 #include "content/browser/browser_thread_impl.h"
15 #include "content/browser/indexed_db/indexed_db_context_impl.h"
16 #include "content/browser/indexed_db/indexed_db_quota_client.h"
17 #include "content/public/browser/storage_partition.h"
18 #include "content/public/test/test_browser_context.h"
19 #include "content/public/test/test_browser_thread_bundle.h"
20 #include "testing/gtest/include/gtest/gtest.h"
21 #include "webkit/browser/quota/mock_quota_manager.h"
22 #include "webkit/common/database/database_identifier.h"
23
24 // Declared to shorten the line lengths.
25 static const quota::StorageType kTemp = quota::kStorageTypeTemporary;
26 static const quota::StorageType kPerm = quota::kStorageTypePersistent;
27
28 namespace content {
29
30 // Base class for our test fixtures.
31 class IndexedDBQuotaClientTest : public testing::Test {
32 public:
33 const GURL kOriginA;
34 const GURL kOriginB;
35 const GURL kOriginOther;
36
IndexedDBQuotaClientTest()37 IndexedDBQuotaClientTest()
38 : kOriginA("http://host"),
39 kOriginB("http://host:8000"),
40 kOriginOther("http://other"),
41 usage_(0),
42 task_runner_(new base::TestSimpleTaskRunner),
43 weak_factory_(this) {
44 browser_context_.reset(new TestBrowserContext());
45
46 scoped_refptr<quota::QuotaManager> quota_manager =
47 new quota::MockQuotaManager(
48 false /*in_memory*/,
49 browser_context_->GetPath(),
50 base::MessageLoop::current()->message_loop_proxy(),
51 base::MessageLoop::current()->message_loop_proxy(),
52 browser_context_->GetSpecialStoragePolicy());
53
54 idb_context_ =
55 new IndexedDBContextImpl(browser_context_->GetPath(),
56 browser_context_->GetSpecialStoragePolicy(),
57 quota_manager->proxy(),
58 task_runner_);
59 base::MessageLoop::current()->RunUntilIdle();
60 setup_temp_dir();
61 }
62
FlushIndexedDBTaskRunner()63 void FlushIndexedDBTaskRunner() { task_runner_->RunUntilIdle(); }
64
setup_temp_dir()65 void setup_temp_dir() {
66 ASSERT_TRUE(temp_dir_.CreateUniqueTempDir());
67 base::FilePath indexeddb_dir =
68 temp_dir_.path().Append(IndexedDBContextImpl::kIndexedDBDirectory);
69 ASSERT_TRUE(base::CreateDirectory(indexeddb_dir));
70 idb_context()->set_data_path_for_testing(indexeddb_dir);
71 }
72
~IndexedDBQuotaClientTest()73 virtual ~IndexedDBQuotaClientTest() {
74 FlushIndexedDBTaskRunner();
75 idb_context_ = NULL;
76 browser_context_.reset();
77 base::MessageLoop::current()->RunUntilIdle();
78 }
79
GetOriginUsage(quota::QuotaClient * client,const GURL & origin,quota::StorageType type)80 int64 GetOriginUsage(quota::QuotaClient* client,
81 const GURL& origin,
82 quota::StorageType type) {
83 usage_ = -1;
84 client->GetOriginUsage(
85 origin,
86 type,
87 base::Bind(&IndexedDBQuotaClientTest::OnGetOriginUsageComplete,
88 weak_factory_.GetWeakPtr()));
89 FlushIndexedDBTaskRunner();
90 base::MessageLoop::current()->RunUntilIdle();
91 EXPECT_GT(usage_, -1);
92 return usage_;
93 }
94
GetOriginsForType(quota::QuotaClient * client,quota::StorageType type)95 const std::set<GURL>& GetOriginsForType(quota::QuotaClient* client,
96 quota::StorageType type) {
97 origins_.clear();
98 client->GetOriginsForType(
99 type,
100 base::Bind(&IndexedDBQuotaClientTest::OnGetOriginsComplete,
101 weak_factory_.GetWeakPtr()));
102 FlushIndexedDBTaskRunner();
103 base::MessageLoop::current()->RunUntilIdle();
104 return origins_;
105 }
106
GetOriginsForHost(quota::QuotaClient * client,quota::StorageType type,const std::string & host)107 const std::set<GURL>& GetOriginsForHost(quota::QuotaClient* client,
108 quota::StorageType type,
109 const std::string& host) {
110 origins_.clear();
111 client->GetOriginsForHost(
112 type,
113 host,
114 base::Bind(&IndexedDBQuotaClientTest::OnGetOriginsComplete,
115 weak_factory_.GetWeakPtr()));
116 FlushIndexedDBTaskRunner();
117 base::MessageLoop::current()->RunUntilIdle();
118 return origins_;
119 }
120
DeleteOrigin(quota::QuotaClient * client,const GURL & origin_url)121 quota::QuotaStatusCode DeleteOrigin(quota::QuotaClient* client,
122 const GURL& origin_url) {
123 delete_status_ = quota::kQuotaStatusUnknown;
124 client->DeleteOriginData(
125 origin_url,
126 kTemp,
127 base::Bind(&IndexedDBQuotaClientTest::OnDeleteOriginComplete,
128 weak_factory_.GetWeakPtr()));
129 FlushIndexedDBTaskRunner();
130 base::MessageLoop::current()->RunUntilIdle();
131 return delete_status_;
132 }
133
idb_context()134 IndexedDBContextImpl* idb_context() { return idb_context_; }
135
SetFileSizeTo(const base::FilePath & path,int size)136 void SetFileSizeTo(const base::FilePath& path, int size) {
137 std::string junk(size, 'a');
138 ASSERT_EQ(size, file_util::WriteFile(path, junk.c_str(), size));
139 }
140
AddFakeIndexedDB(const GURL & origin,int size)141 void AddFakeIndexedDB(const GURL& origin, int size) {
142 base::FilePath file_path_origin = idb_context()->GetFilePathForTesting(
143 webkit_database::GetIdentifierFromOrigin(origin));
144 if (!base::CreateDirectory(file_path_origin)) {
145 LOG(ERROR) << "failed to base::CreateDirectory "
146 << file_path_origin.value();
147 }
148 file_path_origin = file_path_origin.Append(FILE_PATH_LITERAL("fake_file"));
149 SetFileSizeTo(file_path_origin, size);
150 idb_context()->ResetCaches();
151 }
152
153 private:
OnGetOriginUsageComplete(int64 usage)154 void OnGetOriginUsageComplete(int64 usage) { usage_ = usage; }
155
OnGetOriginsComplete(const std::set<GURL> & origins)156 void OnGetOriginsComplete(const std::set<GURL>& origins) {
157 origins_ = origins;
158 }
159
OnDeleteOriginComplete(quota::QuotaStatusCode code)160 void OnDeleteOriginComplete(quota::QuotaStatusCode code) {
161 delete_status_ = code;
162 }
163
164 base::ScopedTempDir temp_dir_;
165 int64 usage_;
166 std::set<GURL> origins_;
167 scoped_refptr<base::TestSimpleTaskRunner> task_runner_;
168 scoped_refptr<IndexedDBContextImpl> idb_context_;
169 content::TestBrowserThreadBundle thread_bundle_;
170 scoped_ptr<TestBrowserContext> browser_context_;
171 quota::QuotaStatusCode delete_status_;
172 base::WeakPtrFactory<IndexedDBQuotaClientTest> weak_factory_;
173
174 DISALLOW_COPY_AND_ASSIGN(IndexedDBQuotaClientTest);
175 };
176
TEST_F(IndexedDBQuotaClientTest,GetOriginUsage)177 TEST_F(IndexedDBQuotaClientTest, GetOriginUsage) {
178 IndexedDBQuotaClient client(idb_context());
179
180 AddFakeIndexedDB(kOriginA, 6);
181 AddFakeIndexedDB(kOriginB, 3);
182 EXPECT_EQ(6, GetOriginUsage(&client, kOriginA, kTemp));
183 EXPECT_EQ(0, GetOriginUsage(&client, kOriginA, kPerm));
184 EXPECT_EQ(3, GetOriginUsage(&client, kOriginB, kTemp));
185 EXPECT_EQ(0, GetOriginUsage(&client, kOriginB, kPerm));
186
187 AddFakeIndexedDB(kOriginA, 1000);
188 EXPECT_EQ(1000, GetOriginUsage(&client, kOriginA, kTemp));
189 EXPECT_EQ(0, GetOriginUsage(&client, kOriginA, kPerm));
190 EXPECT_EQ(3, GetOriginUsage(&client, kOriginB, kTemp));
191 EXPECT_EQ(0, GetOriginUsage(&client, kOriginB, kPerm));
192 }
193
TEST_F(IndexedDBQuotaClientTest,GetOriginsForHost)194 TEST_F(IndexedDBQuotaClientTest, GetOriginsForHost) {
195 IndexedDBQuotaClient client(idb_context());
196
197 EXPECT_EQ(kOriginA.host(), kOriginB.host());
198 EXPECT_NE(kOriginA.host(), kOriginOther.host());
199
200 std::set<GURL> origins = GetOriginsForHost(&client, kTemp, kOriginA.host());
201 EXPECT_TRUE(origins.empty());
202
203 AddFakeIndexedDB(kOriginA, 1000);
204 origins = GetOriginsForHost(&client, kTemp, kOriginA.host());
205 EXPECT_EQ(origins.size(), 1ul);
206 EXPECT_TRUE(origins.find(kOriginA) != origins.end());
207
208 AddFakeIndexedDB(kOriginB, 1000);
209 origins = GetOriginsForHost(&client, kTemp, kOriginA.host());
210 EXPECT_EQ(origins.size(), 2ul);
211 EXPECT_TRUE(origins.find(kOriginA) != origins.end());
212 EXPECT_TRUE(origins.find(kOriginB) != origins.end());
213
214 EXPECT_TRUE(GetOriginsForHost(&client, kPerm, kOriginA.host()).empty());
215 EXPECT_TRUE(GetOriginsForHost(&client, kTemp, kOriginOther.host()).empty());
216 }
217
TEST_F(IndexedDBQuotaClientTest,GetOriginsForType)218 TEST_F(IndexedDBQuotaClientTest, GetOriginsForType) {
219 IndexedDBQuotaClient client(idb_context());
220
221 EXPECT_TRUE(GetOriginsForType(&client, kTemp).empty());
222 EXPECT_TRUE(GetOriginsForType(&client, kPerm).empty());
223
224 AddFakeIndexedDB(kOriginA, 1000);
225 std::set<GURL> origins = GetOriginsForType(&client, kTemp);
226 EXPECT_EQ(origins.size(), 1ul);
227 EXPECT_TRUE(origins.find(kOriginA) != origins.end());
228
229 EXPECT_TRUE(GetOriginsForType(&client, kPerm).empty());
230 }
231
TEST_F(IndexedDBQuotaClientTest,DeleteOrigin)232 TEST_F(IndexedDBQuotaClientTest, DeleteOrigin) {
233 IndexedDBQuotaClient client(idb_context());
234
235 AddFakeIndexedDB(kOriginA, 1000);
236 AddFakeIndexedDB(kOriginB, 50);
237 EXPECT_EQ(1000, GetOriginUsage(&client, kOriginA, kTemp));
238 EXPECT_EQ(50, GetOriginUsage(&client, kOriginB, kTemp));
239
240 quota::QuotaStatusCode delete_status = DeleteOrigin(&client, kOriginA);
241 EXPECT_EQ(quota::kQuotaStatusOk, delete_status);
242 EXPECT_EQ(0, GetOriginUsage(&client, kOriginA, kTemp));
243 EXPECT_EQ(50, GetOriginUsage(&client, kOriginB, kTemp));
244 }
245
246 } // namespace content
247