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

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: Tidy up. 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_FALSE(db->is_open());
25 ASSERT_TRUE(db->OpenInMemory());
26 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
michaeln 2012/02/08 06:11:33 since it's a brand new in memory db, no need to dr
benm (inactive) 2012/02/08 14:59:40 I've changed this to work on an already-open datab
27 ASSERT_TRUE(db->Execute(
28 "CREATE TABLE IF NOT EXISTS ItemTable ("
29 "key TEXT UNIQUE ON CONFLICT REPLACE, "
30 "value TEXT NOT NULL ON CONFLICT FAIL)"));
31 }
32
33 void CreateInvalidTable(sql::Connection* db) {
34 // Create a table with the value type as FLOAT - this is "invalid"
35 // as far as the DOM Storage db is concerned.
36 ASSERT_FALSE(db->is_open());
37 ASSERT_TRUE(db->OpenInMemory());
38 ASSERT_TRUE(db->Execute("DROP TABLE IF EXISTS ItemTable"));
michaeln 2012/02/08 06:11:33 ditto
39 ASSERT_TRUE(db->Execute(
40 "CREATE TABLE IF NOT EXISTS ItemTable ("
41 "key TEXT UNIQUE ON CONFLICT REPLACE, "
42 "value FLOAT NOT NULL ON CONFLICT FAIL)"));
43 }
44
45 void InsertDataV1(sql::Connection* db,
46 const string16& key,
47 const string16& value) {
48 sql::Statement statement(db->GetCachedStatement(SQL_FROM_HERE,
49 "INSERT INTO ItemTable VALUES (?,?)"));
50 statement.BindString16(0, key);
51 statement.BindString16(1, value);
52 ASSERT_TRUE(statement.is_valid());
53 statement.Run();
54 }
55
56 void CheckValuesMatch(DomStorageDatabase* db,
57 const ValuesMap& expected) {
58 ValuesMap values_read;
59 db->ReadAllValues(&values_read);
60 EXPECT_EQ(expected.size(), values_read.size());
61
62 ValuesMap::const_iterator it = values_read.begin();
63 for (; it != values_read.end(); ++it) {
64 string16 key = it->first;
65 NullableString16 value = it->second;
66 NullableString16 expected_value = expected.find(key)->second;
67 EXPECT_EQ(expected_value.string(), value.string());
68 EXPECT_EQ(expected_value.is_null(), value.is_null());
69 }
70 }
71
72 void VerifySchema(sql::Connection* db) {
73 sql::Statement statement(db->GetCachedStatement(SQL_FROM_HERE,
74 "SELECT * from ItemTable LIMIT 1"));
75 EXPECT_TRUE(db->DoesTableExist("ItemTable"));
76 // Ensure that we've got the colums we expect and they are of the
77 // correct type.
78 EXPECT_TRUE(db->DoesColumnExist("ItemTable", "key"));
79 EXPECT_TRUE(db->DoesColumnExist("ItemTable", "value"));
80 EXPECT_EQ(sql::COLUMN_TYPE_TEXT, statement.DeclaredColumnType(0));
81 EXPECT_EQ(sql::COLUMN_TYPE_BLOB, statement.DeclaredColumnType(1));
82 }
83
84 void CreateMapWithValues(ValuesMap* values) {
85 string16 kCannedKeys[] = {
86 ASCIIToUTF16("test"),
87 ASCIIToUTF16("company"),
88 ASCIIToUTF16("date"),
89 ASCIIToUTF16("empty")
90 };
91 NullableString16 kCannedValues[] = {
92 NullableString16(ASCIIToUTF16("123"), false),
93 NullableString16(ASCIIToUTF16("Google"), false),
94 NullableString16(ASCIIToUTF16("18-01-2012"), false),
95 NullableString16(ASCIIToUTF16(""), false)
96 };
97 for (unsigned i = 0; i < sizeof(kCannedKeys) / sizeof(kCannedKeys[0]); i++)
98 (*values)[kCannedKeys[i]] = kCannedValues[i];
99 }
100
101 TEST(DomStorageDatabaseTest, SimpleOpenAndClose) {
102 DomStorageDatabase db;
103 ASSERT_TRUE(db.LazyOpen(true));
104 VerifySchema(db.db_.get());
105 db.Close();
106 EXPECT_FALSE(db.IsOpen());
107 }
108
109 TEST(DomStorageDatabaseTest, TestLazyOpenIsLazy) {
110 // This test needs to operate with a file on disk to ensure that we will
111 // open a file that already exists when only invoking ReadAllValues.
112 ScopedTempDir temp_dir;
113 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
114 FilePath file_name = temp_dir.path().AppendASCII("TestDomStorageDatabase.db");
115
116 DomStorageDatabase db(file_name);
117 EXPECT_FALSE(db.IsOpen());
118 ValuesMap values;
119 db.ReadAllValues(&values);
120 EXPECT_FALSE(db.IsOpen());
121 values[ASCIIToUTF16("key")] = NullableString16(ASCIIToUTF16("value"), false);
122 db.CommitChanges(false, values);
123 EXPECT_TRUE(db.IsOpen());
124
125 db.Close();
126 ASSERT_FALSE(db.IsOpen());
127
128 db.ReadAllValues(&values);
129 EXPECT_TRUE(db.IsOpen());
130 }
131
132 TEST(DomStorageDatabaseTest, TestLazyOpenUpgradesV1TableToV2) {
133 DomStorageDatabase db;
134 db.db_.reset(new sql::Connection());
135 CreateV1Table(db.db_.get());
136 db.Close();
michaeln 2012/02/08 06:11:33 wait a minute... close will delete the in-mem data
benm (inactive) 2012/02/08 14:59:40 good spot. I've made this test use a disk backed d
137
138 EXPECT_TRUE(db.LazyOpen(true));
139 VerifySchema(db.db_.get());
140 }
141
142 TEST(DomStorageDatabaseTest, TestFailedUpgrade) {
143 DomStorageDatabase db;
144 db.db_.reset(new sql::Connection());
145 CreateInvalidTable(db.db_.get());
146
147 EXPECT_FALSE(db.UpgradeVersion1To2IfNeeded());
148
149 db.Close();
michaeln 2012/02/08 06:11:33 ditto the call below not testing the path its tryi
benm (inactive) 2012/02/08 14:59:40 This test is now deleted as the code it exercised
150
151 // LazyOpen should be able to deal with the failed upgrade by
152 // dropping the table and creating a new one.
153 EXPECT_TRUE(db.LazyOpen(true));
154 VerifySchema(db.db_.get());
155 }
156
157
158 TEST(DomStorageDatabaseTest, TestIsOpen) {
michaeln 2012/02/08 06:11:33 can we merge this with SimpleOpenAndClose() which
benm (inactive) 2012/02/08 14:59:40 Done.
159 DomStorageDatabase db;
160 EXPECT_FALSE(db.IsOpen());
161 ASSERT_TRUE(db.LazyOpen(true));
162 EXPECT_TRUE(db.IsOpen());
163 db.Close();
164 EXPECT_FALSE(db.IsOpen());
165 }
166
167 TEST(DomStorageDatabaseTest, SimpleWriteAndReadBack) {
168 DomStorageDatabase db;
169
170 ValuesMap storage;
171 CreateMapWithValues(&storage);
172
173 // Test write.
174 EXPECT_TRUE(db.CommitChanges(false, storage));
175
176 // CheckValuesMatch will invoke DomStorageDatabase::ReadAllValues
177 // to test reading.
178 CheckValuesMatch(&db, storage);
179 }
180
181 TEST(DomStorageDatabaseTest, WriteWithClear) {
182 DomStorageDatabase db;
183
184 ValuesMap storage;
185 CreateMapWithValues(&storage);
186
187 ASSERT_TRUE(db.CommitChanges(false, storage));
188 CheckValuesMatch(&db, storage);
189
190 // Insert some values, clearing the database first.
191 storage.clear();
192 storage[ASCIIToUTF16("another_key")] =
193 NullableString16(ASCIIToUTF16("test"), false);
194 ASSERT_TRUE(db.CommitChanges(true, storage));
195 CheckValuesMatch(&db, storage);
196
197 // Now clear the values without inserting any new ones.
198 storage.clear();
199 ASSERT_TRUE(db.CommitChanges(true, storage));
200 CheckValuesMatch(&db, storage);
201 }
202
203 TEST(DomStorageDatabaseTest, UpgradeFromV1ToV2WithData) {
204 const string16 kCannedKey = ASCIIToUTF16("foo");
205 const NullableString16 kCannedValue(ASCIIToUTF16("bar"), false);
206 ValuesMap expected;
207 expected[kCannedKey] = kCannedValue;
208
209 DomStorageDatabase db;
210 db.db_.reset(new sql::Connection());
211 CreateV1Table(db.db_.get());
212 InsertDataV1(db.db_.get(), kCannedKey, kCannedValue.string());
213
214 ASSERT_TRUE(db.UpgradeVersion1To2IfNeeded());
215
216 VerifySchema(db.db_.get());
217
218 CheckValuesMatch(&db, expected);
219 }
220
221 TEST(DomStorageDatabaseTest, TestOpenCloseDataPreserved) {
222 // This test needs to operate on a file on disk.
223 ScopedTempDir temp_dir;
224 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
225 FilePath file_name = temp_dir.path().AppendASCII("TestDomStorageDatabase.db");
226
227 DomStorageDatabase db(file_name);
228
229 ASSERT_TRUE(db.LazyOpen(true));
230
231 const string16 kCannedKey = ASCIIToUTF16("test");
232 const NullableString16 kCannedValue(ASCIIToUTF16("data"), false);
233 ValuesMap expected;
234 expected[kCannedKey] = kCannedValue;
235 db.CommitChanges(false, expected);
236 db.Close();
237
238 ASSERT_TRUE(db.LazyOpen(true));
239
240 CheckValuesMatch(&db, expected);
241 }
242
243 TEST(DomStorageDatabaseTest, TestSimpleRemoveOneValue) {
244 DomStorageDatabase db;
245
246 ASSERT_TRUE(db.LazyOpen(true));
247 const string16 kCannedKey = ASCIIToUTF16("test");
248 const NullableString16 kCannedValue(ASCIIToUTF16("data"), false);
249 ValuesMap expected;
250 expected[kCannedKey] = kCannedValue;
251 db.CommitChanges(false, expected);
252
253 CheckValuesMatch(&db, expected);
254
255 ValuesMap values;
256 // A null string in the map should mean that that key gets
257 // removed.
258 values[kCannedKey] = NullableString16(true);
259 db.CommitChanges(false, values);
260
261 expected.clear();
262 CheckValuesMatch(&db, expected);
263 }
264
265 // TODO(benm): Enable this test in follow up patch once the test data has
266 // landed. (try-bots don't like adding binary files like test databases :) )
267 TEST(DomStorageDatabaseTest, DISABLED_TestCanOpenAndReadWebCoreDatabase) {
268 FilePath webcore_database;
269 PathService::Get(base::DIR_SOURCE_ROOT, &webcore_database);
270 webcore_database = webcore_database.AppendASCII("webkit");
271 webcore_database = webcore_database.AppendASCII("data");
272 webcore_database = webcore_database.AppendASCII("dom_storage");
273 webcore_database =
274 webcore_database.AppendASCII("webcore_test_database.localstorage");
275
276 ASSERT_TRUE(file_util::PathExists(webcore_database));
277
278 DomStorageDatabase db(webcore_database);
279 ValuesMap values;
280 db.ReadAllValues(&values);
281 EXPECT_TRUE(db.IsOpen());
282 EXPECT_EQ(2u, values.size());
283
284 ValuesMap::const_iterator it =
285 values.find(ASCIIToUTF16("value"));
286 EXPECT_TRUE(it != values.end());
287 EXPECT_EQ(ASCIIToUTF16("I am in local storage!"), it->second.string());
288
289 it = values.find(ASCIIToUTF16("timestamp"));
290 EXPECT_TRUE(it != values.end());
291 EXPECT_EQ(ASCIIToUTF16("1326738338841"), it->second.string());
292
293 it = values.find(ASCIIToUTF16("not_there"));
294 EXPECT_TRUE(it == values.end());
295 }
296
297 TEST(DomStorageDatabaseTest, TestCanOpenFileThatIsNotADatabase) {
298 // Write into the temporary file first.
299 ScopedTempDir temp_dir;
300 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
301 FilePath file_name = temp_dir.path().AppendASCII("TestDomStorageDatabase.db");
302
303 const char kData[] = "I am not a database.";
304 file_util::WriteFile(file_name, kData, strlen(kData));
305
306 {
307 // Try and open the file. We should end up deleting it and creating a new
308 // file, so everything should actually succeed.
309 DomStorageDatabase db(file_name);
310 ValuesMap values;
311 CreateMapWithValues(&values);
312 EXPECT_TRUE(db.CommitChanges(true, values));
313 EXPECT_TRUE(db.CommitChanges(false, values));
314 EXPECT_TRUE(db.IsOpen());
315
316 CheckValuesMatch(&db, values);
317 }
318
319 {
320 // Try to open a directory, we should fail gracefully.
321 DomStorageDatabase db(temp_dir.path());
322 ValuesMap values;
323 CreateMapWithValues(&values);
324 EXPECT_FALSE(db.CommitChanges(true, values));
325 EXPECT_FALSE(db.CommitChanges(false, values));
326 EXPECT_FALSE(db.IsOpen());
327
328 values.clear();
329
330 db.ReadAllValues(&values);
331 EXPECT_EQ(0u, values.size());
332 EXPECT_FALSE(db.IsOpen());
333 }
334 }
335
336 } // namespace dom_storage
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698