OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "sync/syncable/on_disk_directory_backing_store.h" | 5 #include "sync/syncable/on_disk_directory_backing_store.h" |
6 | 6 |
7 #include "base/logging.h" | 7 #include "base/logging.h" |
8 #include "base/metrics/histogram.h" | |
9 #include "sync/syncable/syncable-inl.h" | |
8 | 10 |
9 namespace syncer { | 11 namespace syncer { |
10 namespace syncable { | 12 namespace syncable { |
11 | 13 |
14 namespace { | |
15 | |
16 enum HistogramResultEnum { | |
17 FIRST_TRY_SUCCESS, | |
18 SECOND_TRY_SUCCESS, | |
19 SECOND_TRY_FAILURE, | |
20 RESULT_COUNT | |
21 }; | |
22 | |
23 } // namespace | |
24 | |
12 OnDiskDirectoryBackingStore::OnDiskDirectoryBackingStore( | 25 OnDiskDirectoryBackingStore::OnDiskDirectoryBackingStore( |
13 const std::string& dir_name, const FilePath& backing_filepath) | 26 const std::string& dir_name, const FilePath& backing_filepath) |
14 : DirectoryBackingStore(dir_name), | 27 : DirectoryBackingStore(dir_name), |
28 allow_failure_for_test_(false), | |
15 backing_filepath_(backing_filepath) { | 29 backing_filepath_(backing_filepath) { |
16 db_->set_exclusive_locking(); | 30 db_->set_exclusive_locking(); |
17 db_->set_page_size(4096); | 31 db_->set_page_size(4096); |
18 } | 32 } |
19 | 33 |
20 DirOpenResult OnDiskDirectoryBackingStore::Load( | 34 OnDiskDirectoryBackingStore::~OnDiskDirectoryBackingStore() { } |
35 | |
36 DirOpenResult OnDiskDirectoryBackingStore::TryLoad( | |
21 MetahandlesIndex* entry_bucket, | 37 MetahandlesIndex* entry_bucket, |
22 Directory::KernelLoadInfo* kernel_load_info) { | 38 Directory::KernelLoadInfo* kernel_load_info) { |
23 DCHECK(CalledOnValidThread()); | 39 DCHECK(CalledOnValidThread()); |
24 if (!db_->is_open()) { | 40 if (!db_->is_open()) { |
25 if (!db_->Open(backing_filepath_)) | 41 if (!db_->Open(backing_filepath_)) |
26 return FAILED_OPEN_DATABASE; | 42 return FAILED_OPEN_DATABASE; |
27 } | 43 } |
28 | 44 |
29 if (!InitializeTables()) | 45 if (!InitializeTables()) |
30 return FAILED_OPEN_DATABASE; | 46 return FAILED_OPEN_DATABASE; |
31 | 47 |
32 if (!DropDeletedEntries()) | 48 if (!DropDeletedEntries()) |
33 return FAILED_DATABASE_CORRUPT; | 49 return FAILED_DATABASE_CORRUPT; |
34 if (!LoadEntries(entry_bucket)) | 50 if (!LoadEntries(entry_bucket)) |
35 return FAILED_DATABASE_CORRUPT; | 51 return FAILED_DATABASE_CORRUPT; |
36 if (!LoadInfo(kernel_load_info)) | 52 if (!LoadInfo(kernel_load_info)) |
37 return FAILED_DATABASE_CORRUPT; | 53 return FAILED_DATABASE_CORRUPT; |
54 if (!VerifyReferenceIntegrity(*entry_bucket)) | |
55 return FAILED_DATABASE_CORRUPT; | |
38 | 56 |
39 return OPENED; | 57 return OPENED; |
58 | |
59 } | |
60 | |
61 DirOpenResult OnDiskDirectoryBackingStore::Load( | |
62 MetahandlesIndex* entry_bucket, | |
63 Directory::KernelLoadInfo* kernel_load_info) { | |
64 DirOpenResult result = TryLoad(entry_bucket, kernel_load_info); | |
65 if (result == OPENED) { | |
66 HISTOGRAM_ENUMERATION( | |
67 "Sync.DirectoryOpenResult", FIRST_TRY_SUCCESS, RESULT_COUNT); | |
68 return OPENED; | |
69 } | |
70 | |
71 ReportFirstTryOpenFailure(); | |
72 | |
73 // The fallback: delete the current database and return a fresh one. We can | |
74 // fetch the user's data from the could. | |
75 entry_bucket->clear(); | |
76 db_.reset(new sql::Connection); | |
77 file_util::Delete(backing_filepath_, false); | |
tim (not reviewing)
2012/08/02 00:31:18
How did you test this?
We have integration test
rlarocque
2012/08/02 00:47:20
We have to instantiate a special version of the di
| |
78 | |
79 result = TryLoad(entry_bucket, kernel_load_info); | |
80 if (result == OPENED) { | |
81 HISTOGRAM_ENUMERATION( | |
82 "Sync.DirectoryOpenResult", SECOND_TRY_SUCCESS, RESULT_COUNT); | |
83 } else { | |
84 HISTOGRAM_ENUMERATION( | |
85 "Sync.DirectoryOpenResult", SECOND_TRY_FAILURE, RESULT_COUNT); | |
86 } | |
87 | |
88 return result; | |
89 } | |
90 | |
91 void OnDiskDirectoryBackingStore::ReportFirstTryOpenFailure() { | |
92 // In debug builds, the last thing we want is to silently clear the database. | |
93 // It's full of evidence that might help us determine what went wrong. It | |
94 // might be sqlite's fault, but it could also be a bug in sync. We crash | |
95 // immediately so a developer can investigate. | |
96 // | |
97 // Developers: If you're not interested in debugging this right now, just move | |
98 // aside the 'Sync Data' directory in your profile. This is similar to what | |
99 // the code would do if this DCHECK were disabled. | |
100 NOTREACHED() << "Crashing to preserve corrupt sync database"; | |
40 } | 101 } |
41 | 102 |
42 } // namespace syncable | 103 } // namespace syncable |
43 } // namespace syncer | 104 } // namespace syncer |
OLD | NEW |