OLD | NEW |
---|---|
(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, ¬_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 | |
OLD | NEW |