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

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: Address comments and add handling of a corrupt database file in LazyOpen 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 // TODO(benm): Rather than operating on files on disk, these tests could
18 // run much faster if they were operating on in-memory databases.
michaeln 2012/02/04 00:21:34 faster and more reliably too, i think it'd be wort
benm (inactive) 2012/02/06 14:02:36 Is there a runtime flag or an #ifdef I can use to
benm (inactive) 2012/02/06 15:41:33 I found #ifdef UNIT_TEST. I think that should do t
benm (inactive) 2012/02/06 17:03:21 Done - the variance in time to run is quite remark
michaeln 2012/02/08 06:11:33 yes, how many human lifetimes will be saved in try
19 class DomStorageDatabaseTest : public testing::Test {
20 protected:
21 virtual void SetUp() {
22 ASSERT_TRUE(temp_dir_.CreateUniqueTempDir());
23 file_name_ = temp_dir_.path().AppendASCII("TestDomStorageDatabase.db");
24 }
25
26 void CreateV1Table(sql::Connection* db) {
27 ASSERT_FALSE(db->is_open());
28 ASSERT_TRUE(db->Open(file_name_));
29 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
30 ASSERT_TRUE(db->Execute(
31 "CREATE TABLE IF NOT EXISTS ItemTable ("
32 "key TEXT UNIQUE ON CONFLICT REPLACE, "
33 "value TEXT NOT NULL ON CONFLICT FAIL)"));
34 }
35
36 void CreateInvalidTable(sql::Connection* db) {
37 // Create a table with the value type as FLOAT - this is "invalid"
38 // as far as the DOM Storage db is concerned.
39 ASSERT_FALSE(db->is_open());
40 ASSERT_TRUE(db->Open(file_name_));
41 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
42 ASSERT_TRUE(db->Execute(
43 "CREATE TABLE IF NOT EXISTS ItemTable ("
44 "key TEXT UNIQUE ON CONFLICT REPLACE, "
45 "value FLOAT NOT NULL ON CONFLICT FAIL)"));
46 }
47
48
49 void InsertDataV1(sql::Connection* db,
50 const string16& key,
51 const string16& value) {
52 sql::Statement statement(db->GetCachedStatement(SQL_FROM_HERE,
53 "INSERT INTO ItemTable VALUES (?,?)"));
54 statement.BindString16(0, key);
55 statement.BindString16(1, value);
56 ASSERT_TRUE(statement.is_valid());
57 statement.Run();
58 }
59
60 void InsertDataV2(sql::Connection* db,
61 const string16& key,
62 const string16& value) {
63 sql::Statement statement(db->GetCachedStatement(SQL_FROM_HERE,
64 "INSERT INTO ItemTable VALUES (?,?)"));
65 statement.BindString16(0, key);
66 statement.BindBlob(1, value.data(), value.length() * sizeof(char16));
67 ASSERT_TRUE(statement.is_valid());
68 statement.Run();
69 }
70
71 DomStorageDatabase::ValuesMap ReadAllRows(sql::Connection* db) {
72 DomStorageDatabase::ValuesMap values;
73 sql::Statement statement(db->GetCachedStatement(SQL_FROM_HERE,
74 "SELECT * from ItemTable"));
75 EXPECT_TRUE(statement.is_valid());
76 while (statement.Step()) {
77 string16 key = statement.ColumnString16(0);
78 string16 value;
79 statement.ColumnBlobAsString16(1, &value);
80 values[key] = NullableString16(value, false);
81 }
82 return values;
83 }
84
85 void CheckValuesMatch(sql::Connection* db,
86 const DomStorageDatabase::ValuesMap& expected) {
87 const DomStorageDatabase::ValuesMap values_read = ReadAllRows(db);
88 EXPECT_EQ(expected.size(), values_read.size());
89
90 DomStorageDatabase::ValuesMap::const_iterator it = values_read.begin();
91 for (; it != values_read.end(); ++it) {
92 string16 key = it->first;
93 NullableString16 value = it->second;
94 EXPECT_EQ((expected.find(key)->second).string(), value.string());
michaeln 2012/02/04 00:21:34 what about is_null()? if lhs is_null() with an em
benm (inactive) 2012/02/06 14:02:36 Done.
95 }
96 }
97
98 void VerifySchema(sql::Connection* db) {
99 sql::Statement statement(db->GetCachedStatement(SQL_FROM_HERE,
100 "SELECT * from ItemTable LIMIT 1"));
101 EXPECT_TRUE(db->DoesTableExist("ItemTable"));
102 // Ensure that we've got the colums we expect and they are of the
103 // correct type.
104 EXPECT_TRUE(db->DoesColumnExist("ItemTable", "key"));
105 EXPECT_TRUE(db->DoesColumnExist("ItemTable", "value"));
106 EXPECT_EQ(sql::COLUMN_TYPE_TEXT, statement.DeclaredColumnType(0));
107 EXPECT_EQ(sql::COLUMN_TYPE_BLOB, statement.DeclaredColumnType(1));
108 }
109
110 void CreateMapWith3Values(DomStorageDatabase::ValuesMap* values) {
111 string16 kCannedKeys[] = {
112 ASCIIToUTF16("test"),
113 ASCIIToUTF16("company"),
114 ASCIIToUTF16("date")
115 };
116 NullableString16 kCannedValues[] = {
117 NullableString16(ASCIIToUTF16("123"), false),
118 NullableString16(ASCIIToUTF16("Google"), false),
119 NullableString16(ASCIIToUTF16("18-01-2012"), false)
michaeln 2012/02/04 00:21:34 Maybe add a empty string as a value to ensure that
benm (inactive) 2012/02/06 14:02:36 Done.
120 };
121 for (int i = 0; i < 3; i++)
122 (*values)[kCannedKeys[i]] = kCannedValues[i];
123 }
124
125 ScopedTempDir temp_dir_;
126 FilePath file_name_;
127 };
128
129 TEST_F(DomStorageDatabaseTest, SimpleOpenAndClose) {
130 DomStorageDatabase db(file_name_);
131 ASSERT_TRUE(db.LazyOpen(true));
132 VerifySchema(db.db_.get());
133 db.Close();
134 EXPECT_FALSE(db.IsOpen());
135 }
136
137 TEST_F(DomStorageDatabaseTest, TestLazyOpenIsLazy) {
138 DomStorageDatabase db(file_name_);
139 EXPECT_FALSE(db.IsOpen());
140 DomStorageDatabase::ValuesMap values;
141 db.ReadAllValues(&values);
142 EXPECT_FALSE(db.IsOpen());
143 values[ASCIIToUTF16("key")] = NullableString16(ASCIIToUTF16("value"), false);
144 db.CommitChanges(false, values);
145 EXPECT_TRUE(db.IsOpen());
146
147 db.Close();
148 ASSERT_FALSE(db.IsOpen());
149
150 db.ReadAllValues(&values);
151 EXPECT_TRUE(db.IsOpen());
152 }
153
154 TEST_F(DomStorageDatabaseTest, TestLazyOpenUpgradesV1TableToV2) {
155 DomStorageDatabase db(file_name_);
156 db.db_.reset(new sql::Connection());
157 CreateV1Table(db.db_.get());
158 // Open a scope to run the statements to ensure that statements are cleaned
159 // up before we close the database.
160 {
161 sql::Statement statement(db.db_->GetCachedStatement(SQL_FROM_HERE,
162 "SELECT value from ItemTable LIMIT 1"));
163 ASSERT_EQ(sql::COLUMN_TYPE_TEXT, statement.DeclaredColumnType(0));
michaeln 2012/02/04 00:21:34 Not sure this block adds value? Its testing the re
benm (inactive) 2012/02/06 14:02:36 Good point. I put it as an ASSERT as there's no po
164 }
165 db.Close();
166
167 db.LazyOpen(true);
168 VerifySchema(db.db_.get());
169 }
170
171 TEST_F(DomStorageDatabaseTest, TestFailedUpgrade) {
172 DomStorageDatabase db(file_name_);
173 db.db_.reset(new sql::Connection());
174 CreateInvalidTable(db.db_.get());
175
176 EXPECT_FALSE(db.UpgradeVersion1To2IfNeeded());
177
178 db.Close();
179
180 // LazyOpen should be able to deal with the failed upgrade by
181 // dropping the table and creating a new one.
182 EXPECT_TRUE(db.LazyOpen(true));
183 VerifySchema(db.db_.get());
184 }
185
186
187 TEST_F(DomStorageDatabaseTest, TestIsOpen) {
188 DomStorageDatabase db(file_name_);
189 EXPECT_FALSE(db.IsOpen());
190 ASSERT_TRUE(db.LazyOpen(true));
191 EXPECT_TRUE(db.IsOpen());
192 db.Close();
193 EXPECT_FALSE(db.IsOpen());
194 }
195
196 TEST_F(DomStorageDatabaseTest, SimpleRead) {
197 DomStorageDatabase db(file_name_);
198 db.LazyOpen(true);
199
200 const string16 kCannedKey = ASCIIToUTF16("name");
201 const string16 kCannedValue = ASCIIToUTF16("Joe Bloggs");
202 InsertDataV2(db.db_.get(), kCannedKey, kCannedValue);
203 DomStorageDatabase::ValuesMap values;
204 db.ReadAllValues(&values);
205
206 EXPECT_EQ(1u, values.size());
207 EXPECT_TRUE(values.find(kCannedKey) != values.end());
208 EXPECT_EQ(kCannedValue, values[kCannedKey].string());
209 }
210
211 TEST_F(DomStorageDatabaseTest, SimpleWrite) {
212 DomStorageDatabase db(file_name_);
213
214 DomStorageDatabase::ValuesMap storage;
215 CreateMapWith3Values(&storage);
216 ASSERT_TRUE(db.CommitChanges(false, storage));
217
218 CheckValuesMatch(db.db_.get(), storage);
219 }
220
221 TEST_F(DomStorageDatabaseTest, WriteWithClear) {
222 DomStorageDatabase db(file_name_);
223
224 DomStorageDatabase::ValuesMap storage;
225 CreateMapWith3Values(&storage);
226
227 ASSERT_TRUE(db.CommitChanges(false, storage));
228 CheckValuesMatch(db.db_.get(), storage);
229
230 // Insert some values, clearing the database first.
231 storage.clear();
232 storage[ASCIIToUTF16("another_key")] =
233 NullableString16(ASCIIToUTF16("test"), false);
234 ASSERT_TRUE(db.CommitChanges(true, storage));
235 CheckValuesMatch(db.db_.get(), storage);
236
237 // Now clear the values without inserting any new ones.
238 storage.clear();
239 ASSERT_TRUE(db.CommitChanges(true, storage));
240 CheckValuesMatch(db.db_.get(), storage);
241 }
242
243 TEST_F(DomStorageDatabaseTest, UpgradeFromV1ToV2NoData) {
244 DomStorageDatabase db(file_name_);
245 db.db_.reset(new sql::Connection());
246 CreateV1Table(db.db_.get());
247
248 // The database has V1 structure, try to update it to V2.
249 sql::Statement statement(db.db_->GetCachedStatement(SQL_FROM_HERE,
250 "SELECT value from ItemTable LIMIT 1"));
251 ASSERT_EQ(sql::COLUMN_TYPE_TEXT, statement.DeclaredColumnType(0));
michaeln 2012/02/04 00:21:34 ditto
benm (inactive) 2012/02/06 14:02:36 Done.
252
253 ASSERT_TRUE(db.UpgradeVersion1To2IfNeeded());
254
255 // Verify the db now has V2 structure.
256 VerifySchema(db.db_.get());
257 }
258
259 TEST_F(DomStorageDatabaseTest, UpgradeFromV1ToV2WithData) {
260 const string16 kCannedKey = ASCIIToUTF16("foo");
261 const NullableString16 kCannedValue(ASCIIToUTF16("bar"), false);
262 DomStorageDatabase::ValuesMap expected;
263 expected[kCannedKey] = kCannedValue;
264
265 DomStorageDatabase db(file_name_);
266 db.db_.reset(new sql::Connection());
267 CreateV1Table(db.db_.get());
268
269 // The database has V1 structure, try to update it to V2.
270 sql::Statement statement(db.db_->GetCachedStatement(SQL_FROM_HERE,
271 "SELECT value from ItemTable LIMIT 1"));
272 ASSERT_EQ(sql::COLUMN_TYPE_TEXT, statement.DeclaredColumnType(0));
michaeln 2012/02/04 00:21:34 ditto
benm (inactive) 2012/02/06 14:02:36 Done.
273
274 InsertDataV1(db.db_.get(), kCannedKey, kCannedValue.string());
275
276 ASSERT_TRUE(db.UpgradeVersion1To2IfNeeded());
277
278 VerifySchema(db.db_.get());
279
280 CheckValuesMatch(db.db_.get(), expected);
281 }
282
283 TEST_F(DomStorageDatabaseTest, TestOpenCloseDataPreserved) {
284 DomStorageDatabase db(file_name_);
285
286 ASSERT_TRUE(db.LazyOpen(true));
287
288 const string16 kCannedKey = ASCIIToUTF16("test");
289 const NullableString16 kCannedValue(ASCIIToUTF16("data"), false);
290 InsertDataV2(db.db_.get(), kCannedKey, kCannedValue.string());
291 db.Close();
292
293 ASSERT_TRUE(db.LazyOpen(true));
294 DomStorageDatabase::ValuesMap expected;
295 expected[kCannedKey] = kCannedValue;
296 CheckValuesMatch(db.db_.get(), expected);
297 }
298
299 TEST_F(DomStorageDatabaseTest, TestSimpleRemoveOneValue) {
300 DomStorageDatabase db(file_name_);
301
302 ASSERT_TRUE(db.LazyOpen(true));
303 const string16 kCannedKey = ASCIIToUTF16("test");
304 const NullableString16 kCannedValue(ASCIIToUTF16("data"), false);
305 InsertDataV2(db.db_.get(), kCannedKey, kCannedValue.string());
306
307 DomStorageDatabase::ValuesMap expected;
308 expected[kCannedKey] = kCannedValue;
309 CheckValuesMatch(db.db_.get(), expected);
310
311 DomStorageDatabase::ValuesMap values;
312 values[kCannedKey] = NullableString16(true);
313
314 db.CommitChanges(false, values);
315
316 expected.clear();
317 CheckValuesMatch(db.db_.get(), expected);
318 }
319
320 // TODO(benm): Enable this test in follow up patch once the test data has
321 // landed. (try-bots don't like adding binary files like test databases :) )
322 TEST_F(DomStorageDatabaseTest, DISABLED_TestCanOpenAndReadWebCoreDatabase) {
323 FilePath webcore_database;
324 PathService::Get(base::DIR_SOURCE_ROOT, &webcore_database);
325 webcore_database = webcore_database.AppendASCII("webkit");
326 webcore_database = webcore_database.AppendASCII("data");
327 webcore_database = webcore_database.AppendASCII("dom_storage");
328 webcore_database =
329 webcore_database.AppendASCII("webcore_test_database.localstorage");
330
331 ASSERT_TRUE(file_util::PathExists(webcore_database));
332
333 DomStorageDatabase db(webcore_database);
334 DomStorageDatabase::ValuesMap values;
335 db.ReadAllValues(&values);
336 EXPECT_TRUE(db.IsOpen());
337 EXPECT_EQ(2u, values.size());
338
339 DomStorageDatabase::ValuesMap::const_iterator it =
340 values.find(ASCIIToUTF16("value"));
341 EXPECT_TRUE(it != values.end());
342 EXPECT_EQ(ASCIIToUTF16("I am in local storage!"), it->second.string());
343
344 it = values.find(ASCIIToUTF16("timestamp"));
345 EXPECT_TRUE(it != values.end());
346 EXPECT_EQ(ASCIIToUTF16("1326738338841"), it->second.string());
347
348 it = values.find(ASCIIToUTF16("not_there"));
349 EXPECT_TRUE(it == values.end());
350 }
351
352 TEST_F(DomStorageDatabaseTest, TestCanOpenFileThatIsNotADatabase) {
353 FilePath not_a_database;
354 PathService::Get(base::DIR_SOURCE_ROOT, &not_a_database);
michaeln 2012/02/04 00:21:34 would be more hermetic to not rely on an external
benm (inactive) 2012/02/06 14:02:36 Done.
355 not_a_database = not_a_database.AppendASCII("webkit");
356 not_a_database = not_a_database.AppendASCII("data");
357 not_a_database = not_a_database.AppendASCII("dom_storage");
358 not_a_database =
359 not_a_database.AppendASCII("not_a_database.localstorage");
360
361 ASSERT_TRUE(file_util::PathExists(not_a_database));
362
363 DomStorageDatabase db(not_a_database);
364 DomStorageDatabase::ValuesMap values;
365 CreateMapWith3Values(&values);
366 EXPECT_FALSE(db.CommitChanges(true, values));
michaeln 2012/02/04 00:21:34 we want the behavior to be such that this commit s
benm (inactive) 2012/02/06 14:02:36 Right, but don't have the code to do the delete an
367 EXPECT_FALSE(db.CommitChanges(false, values));
368 EXPECT_FALSE(db.IsOpen());
369
370 values.clear();
371
372 db.ReadAllValues(&values);
373 EXPECT_EQ(0u, values.size());
374 EXPECT_FALSE(db.IsOpen());
375 }
376
377 } // namespace dom_storage
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698