1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
|
// Copyright (c) 2010 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/json_pref_store.h"
#include <algorithm>
#include "base/file_util.h"
#include "base/values.h"
#include "chrome/common/json_value_serializer.h"
namespace {
// Some extensions we'll tack on to copies of the Preferences files.
const FilePath::CharType* kBadExtension = FILE_PATH_LITERAL("bad");
} // namespace
JsonPrefStore::JsonPrefStore(const FilePath& filename)
: path_(filename),
prefs_(new DictionaryValue()),
read_only_(false),
writer_(filename) { }
JsonPrefStore::~JsonPrefStore() {
if (writer_.HasPendingWrite() && !read_only_)
writer_.DoScheduledWrite();
}
PrefStore::PrefReadError JsonPrefStore::ReadPrefs() {
JSONFileValueSerializer serializer(path_);
int error_code = 0;
std::string error_msg;
scoped_ptr<Value> value(serializer.Deserialize(&error_code, &error_msg));
if (!value.get()) {
#if defined(GOOGLE_CHROME_BUILD)
// This log could be used for more detailed client-side error diagnosis,
// but since this triggers often with unit tests, we need to disable it
// in non-official builds.
PLOG(ERROR) << "Error reading Preferences: " << error_msg << " " <<
path_.value();
#endif
PrefReadError error;
switch (error_code) {
case JSONFileValueSerializer::JSON_ACCESS_DENIED:
// If the file exists but is simply unreadable, put the file into a
// state where we don't try to save changes. Otherwise, we could
// clobber the existing prefs.
error = PREF_READ_ERROR_ACCESS_DENIED;
read_only_ = true;
break;
case JSONFileValueSerializer::JSON_CANNOT_READ_FILE:
error = PREF_READ_ERROR_FILE_OTHER;
read_only_ = true;
break;
case JSONFileValueSerializer::JSON_FILE_LOCKED:
error = PREF_READ_ERROR_FILE_LOCKED;
read_only_ = true;
break;
case JSONFileValueSerializer::JSON_NO_SUCH_FILE:
// If the file just doesn't exist, maybe this is first run. In any case
// there's no harm in writing out default prefs in this case.
error = PREF_READ_ERROR_NO_FILE;
break;
default:
error = PREF_READ_ERROR_JSON_PARSE;
// JSON errors indicate file corruption of some sort.
// Since the file is corrupt, move it to the side and continue with
// empty preferences. This will result in them losing their settings.
// We keep the old file for possible support and debugging assistance
// as well as to detect if they're seeing these errors repeatedly.
// TODO(erikkay) Instead, use the last known good file.
FilePath bad = path_.ReplaceExtension(kBadExtension);
// If they've ever had a parse error before, put them in another bucket.
// TODO(erikkay) if we keep this error checking for very long, we may
// want to differentiate between recent and long ago errors.
if (file_util::PathExists(bad))
error = PREF_READ_ERROR_JSON_REPEAT;
file_util::Move(path_, bad);
break;
}
return error;
}
// Preferences should always have a dictionary root.
if (!value->IsType(Value::TYPE_DICTIONARY)) {
// See comment for the default case above.
read_only_ = true;
return PREF_READ_ERROR_JSON_TYPE;
}
prefs_.reset(static_cast<DictionaryValue*>(value.release()));
return PREF_READ_ERROR_NONE;
}
bool JsonPrefStore::WritePrefs() {
std::string data;
if (!SerializeData(&data))
return false;
// Lie about our ability to save.
if (read_only_)
return true;
writer_.WriteNow(data);
return true;
}
void JsonPrefStore::ScheduleWritePrefs() {
if (read_only_)
return;
writer_.ScheduleWrite(this);
}
bool JsonPrefStore::SerializeData(std::string* output) {
// TODO(tc): Do we want to prune webkit preferences that match the default
// value?
JSONStringValueSerializer serializer(output);
serializer.set_pretty_print(true);
scoped_ptr<DictionaryValue> copy(prefs_->DeepCopyWithoutEmptyChildren());
return serializer.Serialize(*(copy.get()));
}
|