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 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 | |
OLD | NEW |