Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(631)

Side by Side Diff: webkit/dom_storage/dom_storage_database_unittest.cc

Issue 9159020: Create a class to represent a DOM Storage Database. (Closed) Base URL: http://git.chromium.org/chromium/src.git@master
Patch Set: Disable the WebCore DB test to keep try bots happy while the binary test data is landed Created 8 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
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.
4
5 #include "webkit/dom_storage/dom_storage_database.h"
6
7 #include "base/file_path.h"
8 #include "base/file_util.h"
9 #include "base/path_service.h"
10 #include "base/scoped_temp_dir.h"
11 #include "base/utf_string_conversions.h"
12 #include "sql/statement.h"
13 #include "testing/gtest/include/gtest/gtest.h"
14
15 namespace dom_storage {
16
17 DomStorageDatabase::DomStorageDatabase()
18 : db_(NULL),
19 failed_to_open_(false),
20 tried_to_recreate_(false) {
21 }
22
23 void CreateV1Table(sql::Connection* db) {
24 ASSERT_TRUE(db->is_open());
25 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
26 ASSERT_TRUE(db->Execute(
27 "CREATE TABLE ItemTable ("
28 "key TEXT UNIQUE ON CONFLICT REPLACE, "
29 "value TEXT NOT NULL ON CONFLICT FAIL)"));
30 }
31
32 void CreateV2Table(sql::Connection* db) {
33 ASSERT_TRUE(db->is_open());
34 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
35 ASSERT_TRUE(db->Execute(
36 "CREATE TABLE ItemTable ("
37 "key TEXT UNIQUE ON CONFLICT REPLACE, "
38 "value BLOB NOT NULL ON CONFLICT FAIL)"));
39 }
40
41 void CreateInvalidKeyColumnTable(sql::Connection* db) {
42 // Create a table with the key type as FLOAT - this is "invalid"
43 // as far as the DOM Storage db is concerned.
44 ASSERT_TRUE(db->is_open());
45 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
46 ASSERT_TRUE(db->Execute(
47 "CREATE TABLE IF NOT EXISTS ItemTable ("
48 "key FLOAT UNIQUE ON CONFLICT REPLACE, "
49 "value BLOB NOT NULL ON CONFLICT FAIL)"));
50 }
51 void CreateInvalidValueColumnTable(sql::Connection* db) {
52 // Create a table with the value type as FLOAT - this is "invalid"
53 // as far as the DOM Storage db is concerned.
54 ASSERT_TRUE(db->is_open());
55 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
56 ASSERT_TRUE(db->Execute(
57 "CREATE TABLE IF NOT EXISTS ItemTable ("
58 "key TEXT UNIQUE ON CONFLICT REPLACE, "
59 "value FLOAT NOT NULL ON CONFLICT FAIL)"));
60 }
61
62 void InsertDataV1(sql::Connection* db,
63 const string16& key,
64 const string16& value) {
65 sql::Statement statement(db->GetCachedStatement(SQL_FROM_HERE,
66 "INSERT INTO ItemTable VALUES (?,?)"));
67 statement.BindString16(0, key);
68 statement.BindString16(1, value);
69 ASSERT_TRUE(statement.is_valid());
70 statement.Run();
71 }
72
73 void CheckValuesMatch(DomStorageDatabase* db,
74 const ValuesMap& expected) {
75 ValuesMap values_read;
76 db->ReadAllValues(&values_read);
77 EXPECT_EQ(expected.size(), values_read.size());
78
79 ValuesMap::const_iterator it = values_read.begin();
80 for (; it != values_read.end(); ++it) {
81 string16 key = it->first;
82 NullableString16 value = it->second;
83 NullableString16 expected_value = expected.find(key)->second;
84 EXPECT_EQ(expected_value.string(), value.string());
85 EXPECT_EQ(expected_value.is_null(), value.is_null());
86 }
87 }
88
89 void CreateMapWithValues(ValuesMap* values) {
90 string16 kCannedKeys[] = {
91 ASCIIToUTF16("test"),
92 ASCIIToUTF16("company"),
93 ASCIIToUTF16("date"),
94 ASCIIToUTF16("empty")
95 };
96 NullableString16 kCannedValues[] = {
97 NullableString16(ASCIIToUTF16("123"), false),
98 NullableString16(ASCIIToUTF16("Google"), false),
99 NullableString16(ASCIIToUTF16("18-01-2012"), false),
100 NullableString16(ASCIIToUTF16(""), false)
101 };
102 for (unsigned i = 0; i < sizeof(kCannedKeys) / sizeof(kCannedKeys[0]); i++)
103 (*values)[kCannedKeys[i]] = kCannedValues[i];
104 }
105
106 TEST(DomStorageDatabaseTest, SimpleOpenAndClose) {
107 DomStorageDatabase db;
108 EXPECT_FALSE(db.IsOpen());
109 ASSERT_TRUE(db.LazyOpen(true));
110 EXPECT_TRUE(db.IsOpen());
111 EXPECT_EQ(DomStorageDatabase::V2, db.DetectSchemaVersion());
112 db.Close();
113 EXPECT_FALSE(db.IsOpen());
114 }
115
116 TEST(DomStorageDatabaseTest, TestLazyOpenIsLazy) {
117 // This test needs to operate with a file on disk to ensure that we will
118 // open a file that already exists when only invoking ReadAllValues.
119 ScopedTempDir temp_dir;
120 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
121 FilePath file_name = temp_dir.path().AppendASCII("TestDomStorageDatabase.db");
122
123 DomStorageDatabase db(file_name);
124 EXPECT_FALSE(db.IsOpen());
125 ValuesMap values;
126 db.ReadAllValues(&values);
127 // Reading an empty db should not open the database.
128 EXPECT_FALSE(db.IsOpen());
129
130 values[ASCIIToUTF16("key")] = NullableString16(ASCIIToUTF16("value"), false);
131 db.CommitChanges(false, values);
132 // Writing content should open the database.
133 EXPECT_TRUE(db.IsOpen());
134
135 db.Close();
136 ASSERT_FALSE(db.IsOpen());
137
138 // Reading from an existing database should open the database.
139 db.ReadAllValues(&values);
140 EXPECT_TRUE(db.IsOpen());
michaeln 2012/02/08 22:12:08 With minor mods, this test could serve to verify w
benm (inactive) 2012/02/09 11:59:57 Done.
141 }
142
143 TEST(DomStorageDatabaseTest, TestDetectSchemaVersion) {
144 {
145 DomStorageDatabase db;
146 db.db_.reset(new sql::Connection());
147 ASSERT_TRUE(db.db_->OpenInMemory());
148 CreateInvalidValueColumnTable(db.db_.get());
149 EXPECT_EQ(DomStorageDatabase::INVALID, db.DetectSchemaVersion());
150 }
151
152 {
153 DomStorageDatabase db;
154 db.db_.reset(new sql::Connection());
155 ASSERT_TRUE(db.db_->OpenInMemory());
156 CreateInvalidKeyColumnTable(db.db_.get());
157 EXPECT_EQ(DomStorageDatabase::INVALID, db.DetectSchemaVersion());
158 }
159
160 {
161 DomStorageDatabase db;
162 db.db_.reset(new sql::Connection());
163 ASSERT_TRUE(db.db_->OpenInMemory());
164 CreateV1Table(db.db_.get());
165 EXPECT_EQ(DomStorageDatabase::V1, db.DetectSchemaVersion());
166 }
167
168 {
169 DomStorageDatabase db;
170 db.db_.reset(new sql::Connection());
171 ASSERT_TRUE(db.db_->OpenInMemory());
172 CreateV2Table(db.db_.get());
173 EXPECT_EQ(DomStorageDatabase::V2, db.DetectSchemaVersion());
michaeln 2012/02/08 22:12:08 looks like much of the boiler plate around each ca
benm (inactive) 2012/02/09 11:59:57 Thanks, done!
174 }
175 }
176
177 TEST(DomStorageDatabaseTest, TestLazyOpenUpgradesDatabase) {
178 // This test needs to operate with a file on disk so that we
179 // can create a table at version 1 and then close it again
180 // so that LazyOpen sees there is work to do (LazyOpen will return
181 // early if the database is already open).
182 ScopedTempDir temp_dir;
183 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
184 FilePath file_name = temp_dir.path().AppendASCII("TestDomStorageDatabase.db");
185
186 DomStorageDatabase db(file_name);
187 db.db_.reset(new sql::Connection());
188 ASSERT_TRUE(db.db_->Open(file_name));
189 CreateV1Table(db.db_.get());
190 db.Close();
191
192 EXPECT_TRUE(db.LazyOpen(true));
193 EXPECT_EQ(DomStorageDatabase::V2, db.DetectSchemaVersion());
194 }
195
196 TEST(DomStorageDatabaseTest, SimpleWriteAndReadBack) {
197 DomStorageDatabase db;
198
199 ValuesMap storage;
200 CreateMapWithValues(&storage);
201
202 // Test write.
michaeln 2012/02/08 22:12:08 this comment and the one below probably can be rem
benm (inactive) 2012/02/09 11:59:57 Done.
203 EXPECT_TRUE(db.CommitChanges(false, storage));
204
205 // CheckValuesMatch will invoke DomStorageDatabase::ReadAllValues
206 // to test reading.
207 CheckValuesMatch(&db, storage);
208 }
209
210 TEST(DomStorageDatabaseTest, WriteWithClear) {
211 DomStorageDatabase db;
212
213 ValuesMap storage;
214 CreateMapWithValues(&storage);
215
216 ASSERT_TRUE(db.CommitChanges(false, storage));
217 CheckValuesMatch(&db, storage);
218
219 // Insert some values, clearing the database first.
220 storage.clear();
221 storage[ASCIIToUTF16("another_key")] =
222 NullableString16(ASCIIToUTF16("test"), false);
223 ASSERT_TRUE(db.CommitChanges(true, storage));
224 CheckValuesMatch(&db, storage);
225
226 // Now clear the values without inserting any new ones.
227 storage.clear();
228 ASSERT_TRUE(db.CommitChanges(true, storage));
229 CheckValuesMatch(&db, storage);
230 }
231
232 TEST(DomStorageDatabaseTest, UpgradeFromV1ToV2WithData) {
233 const string16 kCannedKey = ASCIIToUTF16("foo");
234 const NullableString16 kCannedValue(ASCIIToUTF16("bar"), false);
235 ValuesMap expected;
236 expected[kCannedKey] = kCannedValue;
237
238 DomStorageDatabase db;
239 db.db_.reset(new sql::Connection());
240 ASSERT_TRUE(db.db_->OpenInMemory());
241 CreateV1Table(db.db_.get());
242 InsertDataV1(db.db_.get(), kCannedKey, kCannedValue.string());
243
244 ASSERT_TRUE(db.UpgradeVersion1To2());
245
246 EXPECT_EQ(DomStorageDatabase::V2, db.DetectSchemaVersion());
247
248 CheckValuesMatch(&db, expected);
249 }
250
251 TEST(DomStorageDatabaseTest, TestOpenCloseDataPreserved) {
252 // This test needs to operate on a file on disk as data
253 // must be preserved through a call to Close().
254 ScopedTempDir temp_dir;
255 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
256 FilePath file_name = temp_dir.path().AppendASCII("TestDomStorageDatabase.db");
257 DomStorageDatabase db(file_name);
258
259 const string16 kCannedKey = ASCIIToUTF16("test");
260 const NullableString16 kCannedValue(ASCIIToUTF16("data"), false);
261 ValuesMap expected;
262 expected[kCannedKey] = kCannedValue;
263 ASSERT_TRUE(db.CommitChanges(false, expected));
264 db.Close();
265
266 CheckValuesMatch(&db, expected);
267 }
268
269 TEST(DomStorageDatabaseTest, TestSimpleRemoveOneValue) {
270 DomStorageDatabase db;
271
272 ASSERT_TRUE(db.LazyOpen(true));
273 const string16 kCannedKey = ASCIIToUTF16("test");
274 const NullableString16 kCannedValue(ASCIIToUTF16("data"), false);
275 ValuesMap expected;
276 expected[kCannedKey] = kCannedValue;
277
278 // First write some data into the database.
279 ASSERT_TRUE(db.CommitChanges(false, expected));
280 CheckValuesMatch(&db, expected);
281
282 ValuesMap values;
283 // A null string in the map should mean that that key gets
284 // removed.
285 values[kCannedKey] = NullableString16(true);
286 EXPECT_TRUE(db.CommitChanges(false, values));
287
288 expected.clear();
289 CheckValuesMatch(&db, expected);
290 }
291
292 // TODO(benm): Enable this test in follow up patch once the test data has
293 // landed. (try-bots don't like adding binary files like test databases :) )
294 TEST(DomStorageDatabaseTest, DISABLED_TestCanOpenAndReadWebCoreDatabase) {
295 FilePath webcore_database;
296 PathService::Get(base::DIR_SOURCE_ROOT, &webcore_database);
297 webcore_database = webcore_database.AppendASCII("webkit");
298 webcore_database = webcore_database.AppendASCII("data");
299 webcore_database = webcore_database.AppendASCII("dom_storage");
300 webcore_database =
301 webcore_database.AppendASCII("webcore_test_database.localstorage");
302
303 ASSERT_TRUE(file_util::PathExists(webcore_database));
304
305 DomStorageDatabase db(webcore_database);
306 ValuesMap values;
307 db.ReadAllValues(&values);
308 EXPECT_TRUE(db.IsOpen());
309 EXPECT_EQ(2u, values.size());
310
311 ValuesMap::const_iterator it =
312 values.find(ASCIIToUTF16("value"));
313 EXPECT_TRUE(it != values.end());
314 EXPECT_EQ(ASCIIToUTF16("I am in local storage!"), it->second.string());
315
316 it = values.find(ASCIIToUTF16("timestamp"));
317 EXPECT_TRUE(it != values.end());
318 EXPECT_EQ(ASCIIToUTF16("1326738338841"), it->second.string());
319
320 it = values.find(ASCIIToUTF16("not_there"));
321 EXPECT_TRUE(it == values.end());
322 }
323
324 TEST(DomStorageDatabaseTest, TestCanOpenFileThatIsNotADatabase) {
325 // Write into the temporary file first.
326 ScopedTempDir temp_dir;
327 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
328 FilePath file_name = temp_dir.path().AppendASCII("TestDomStorageDatabase.db");
329
330 const char kData[] = "I am not a database.";
331 file_util::WriteFile(file_name, kData, strlen(kData));
332
333 {
334 // Try and open the file. As it's not a database, we should end up deleting
335 // it and creating a new, valid file, so everything should actually
336 // succeed.
337 DomStorageDatabase db(file_name);
338 ValuesMap values;
339 CreateMapWithValues(&values);
340 EXPECT_TRUE(db.CommitChanges(true, values));
341 EXPECT_TRUE(db.CommitChanges(false, values));
342 EXPECT_TRUE(db.IsOpen());
343
344 CheckValuesMatch(&db, values);
345 }
346
347 {
348 // Try to open a directory, we should fail gracefully and not attempt
349 // to delete it.
350 DomStorageDatabase db(temp_dir.path());
351 ValuesMap values;
352 CreateMapWithValues(&values);
353 EXPECT_FALSE(db.CommitChanges(true, values));
354 EXPECT_FALSE(db.CommitChanges(false, values));
355 EXPECT_FALSE(db.IsOpen());
356
357 values.clear();
358
359 db.ReadAllValues(&values);
360 EXPECT_EQ(0u, values.size());
361 EXPECT_FALSE(db.IsOpen());
362
363 EXPECT_TRUE(file_util::PathExists(temp_dir.path()));
364 }
365 }
366
367 } // namespace dom_storage
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698