• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 "sync/syncable/entry_kernel.h"
6 
7 #include "base/strings/string_number_conversions.h"
8 #include "sync/protocol/proto_value_conversions.h"
9 #include "sync/syncable/syncable_enum_conversions.h"
10 #include "sync/util/cryptographer.h"
11 
12 namespace syncer {
13 namespace syncable {
14 
EntryKernel()15 EntryKernel::EntryKernel() : dirty_(false) {
16   // Everything else should already be default-initialized.
17   for (int i = INT64_FIELDS_BEGIN; i < INT64_FIELDS_END; ++i) {
18     int64_fields[i] = 0;
19   }
20 }
21 
~EntryKernel()22 EntryKernel::~EntryKernel() {}
23 
GetModelType() const24 ModelType EntryKernel::GetModelType() const {
25   ModelType specifics_type = GetModelTypeFromSpecifics(ref(SPECIFICS));
26   if (specifics_type != UNSPECIFIED)
27     return specifics_type;
28   if (ref(ID).IsRoot())
29     return TOP_LEVEL_FOLDER;
30   // Loose check for server-created top-level folders that aren't
31   // bound to a particular model type.
32   if (!ref(UNIQUE_SERVER_TAG).empty() && ref(SERVER_IS_DIR))
33     return TOP_LEVEL_FOLDER;
34 
35   return UNSPECIFIED;
36 }
37 
GetServerModelType() const38 ModelType EntryKernel::GetServerModelType() const {
39   ModelType specifics_type = GetModelTypeFromSpecifics(ref(SERVER_SPECIFICS));
40   if (specifics_type != UNSPECIFIED)
41     return specifics_type;
42   if (ref(ID).IsRoot())
43     return TOP_LEVEL_FOLDER;
44   // Loose check for server-created top-level folders that aren't
45   // bound to a particular model type.
46   if (!ref(UNIQUE_SERVER_TAG).empty() && ref(SERVER_IS_DIR))
47     return TOP_LEVEL_FOLDER;
48 
49   return UNSPECIFIED;
50 }
51 
ShouldMaintainPosition() const52 bool EntryKernel::ShouldMaintainPosition() const {
53   // We maintain positions for all bookmarks, except those that are
54   // server-created top-level folders.
55   return (GetModelTypeFromSpecifics(ref(SPECIFICS)) == syncer::BOOKMARKS)
56       && !(!ref(UNIQUE_SERVER_TAG).empty() && ref(IS_DIR));
57 }
58 
ShouldMaintainHierarchy() const59 bool EntryKernel::ShouldMaintainHierarchy() const {
60   // We maintain hierarchy for bookmarks, device info, and top-level folders,
61   // but no other types.  Note that the Nigori node consists of a single
62   // top-level folder, so it's included in this set.
63   return (GetModelTypeFromSpecifics(ref(SPECIFICS)) == syncer::BOOKMARKS)
64       || (!ref(UNIQUE_SERVER_TAG).empty());
65 }
66 
67 namespace {
68 
69 // Utility function to loop through a set of enum values and add the
70 // field keys/values in the kernel to the given dictionary.
71 //
72 // V should be convertible to Value.
73 template <class T, class U, class V>
SetFieldValues(const EntryKernel & kernel,base::DictionaryValue * dictionary_value,const char * (* enum_key_fn)(T),V * (* enum_value_fn)(U),int field_key_min,int field_key_max)74 void SetFieldValues(const EntryKernel& kernel,
75                     base::DictionaryValue* dictionary_value,
76                     const char* (*enum_key_fn)(T),
77                     V* (*enum_value_fn)(U),
78                     int field_key_min, int field_key_max) {
79   DCHECK_LE(field_key_min, field_key_max);
80   for (int i = field_key_min; i <= field_key_max; ++i) {
81     T field = static_cast<T>(i);
82     const std::string& key = enum_key_fn(field);
83     V* value = enum_value_fn(kernel.ref(field));
84     dictionary_value->Set(key, value);
85   }
86 }
87 
SetEncryptableProtoValues(const EntryKernel & kernel,Cryptographer * cryptographer,base::DictionaryValue * dictionary_value,int field_key_min,int field_key_max)88 void SetEncryptableProtoValues(
89     const EntryKernel& kernel,
90     Cryptographer* cryptographer,
91     base::DictionaryValue* dictionary_value,
92     int field_key_min, int field_key_max) {
93   DCHECK_LE(field_key_min, field_key_max);
94   for (int i = field_key_min; i <= field_key_max; ++i) {
95     ProtoField field = static_cast<ProtoField>(i);
96     const std::string& key = GetProtoFieldString(field);
97 
98     base::DictionaryValue* value = NULL;
99     sync_pb::EntitySpecifics decrypted;
100     const sync_pb::EncryptedData& encrypted = kernel.ref(field).encrypted();
101     if (cryptographer &&
102         kernel.ref(field).has_encrypted() &&
103         cryptographer->CanDecrypt(encrypted) &&
104         cryptographer->Decrypt(encrypted, &decrypted)) {
105       value = EntitySpecificsToValue(decrypted);
106       value->SetBoolean("encrypted", true);
107     } else {
108       value = EntitySpecificsToValue(kernel.ref(field));
109     }
110     dictionary_value->Set(key, value);
111   }
112 }
113 
114 // Helper functions for SetFieldValues().
115 
Int64ToValue(int64 i)116 base::StringValue* Int64ToValue(int64 i) {
117   return new base::StringValue(base::Int64ToString(i));
118 }
119 
TimeToValue(const base::Time & t)120 base::StringValue* TimeToValue(const base::Time& t) {
121   return new base::StringValue(GetTimeDebugString(t));
122 }
123 
IdToValue(const Id & id)124 base::StringValue* IdToValue(const Id& id) {
125   return id.ToValue();
126 }
127 
BooleanToValue(bool bool_val)128 base::FundamentalValue* BooleanToValue(bool bool_val) {
129   return new base::FundamentalValue(bool_val);
130 }
131 
StringToValue(const std::string & str)132 base::StringValue* StringToValue(const std::string& str) {
133   return new base::StringValue(str);
134 }
135 
UniquePositionToValue(const UniquePosition & pos)136 base::StringValue* UniquePositionToValue(const UniquePosition& pos) {
137   return new base::StringValue(pos.ToDebugString());
138 }
139 
AttachmentMetadataToValue(const sync_pb::AttachmentMetadata & a)140 base::StringValue* AttachmentMetadataToValue(
141     const sync_pb::AttachmentMetadata& a) {
142   return new base::StringValue(a.SerializeAsString());
143 }
144 
145 }  // namespace
146 
ToValue(Cryptographer * cryptographer) const147 base::DictionaryValue* EntryKernel::ToValue(
148     Cryptographer* cryptographer) const {
149   base::DictionaryValue* kernel_info = new base::DictionaryValue();
150   kernel_info->SetBoolean("isDirty", is_dirty());
151   kernel_info->Set("serverModelType", ModelTypeToValue(GetServerModelType()));
152 
153   // Int64 fields.
154   SetFieldValues(*this, kernel_info,
155                  &GetMetahandleFieldString, &Int64ToValue,
156                  INT64_FIELDS_BEGIN, META_HANDLE);
157   SetFieldValues(*this, kernel_info,
158                  &GetBaseVersionString, &Int64ToValue,
159                  META_HANDLE + 1, BASE_VERSION);
160   SetFieldValues(*this, kernel_info,
161                  &GetInt64FieldString, &Int64ToValue,
162                  BASE_VERSION + 1, INT64_FIELDS_END - 1);
163 
164   // Time fields.
165   SetFieldValues(*this, kernel_info,
166                  &GetTimeFieldString, &TimeToValue,
167                  TIME_FIELDS_BEGIN, TIME_FIELDS_END - 1);
168 
169   // ID fields.
170   SetFieldValues(*this, kernel_info,
171                  &GetIdFieldString, &IdToValue,
172                  ID_FIELDS_BEGIN, ID_FIELDS_END - 1);
173 
174   // Bit fields.
175   SetFieldValues(*this, kernel_info,
176                  &GetIndexedBitFieldString, &BooleanToValue,
177                  BIT_FIELDS_BEGIN, INDEXED_BIT_FIELDS_END - 1);
178   SetFieldValues(*this, kernel_info,
179                  &GetIsDelFieldString, &BooleanToValue,
180                  INDEXED_BIT_FIELDS_END, IS_DEL);
181   SetFieldValues(*this, kernel_info,
182                  &GetBitFieldString, &BooleanToValue,
183                  IS_DEL + 1, BIT_FIELDS_END - 1);
184 
185   // String fields.
186   {
187     // Pick out the function overload we want.
188     SetFieldValues(*this, kernel_info,
189                    &GetStringFieldString, &StringToValue,
190                    STRING_FIELDS_BEGIN, STRING_FIELDS_END - 1);
191   }
192 
193   // Proto fields.
194   SetEncryptableProtoValues(*this, cryptographer, kernel_info,
195                             PROTO_FIELDS_BEGIN, PROTO_FIELDS_END - 1);
196 
197   // UniquePosition fields
198   SetFieldValues(*this, kernel_info,
199                  &GetUniquePositionFieldString, &UniquePositionToValue,
200                  UNIQUE_POSITION_FIELDS_BEGIN, UNIQUE_POSITION_FIELDS_END - 1);
201 
202   // AttachmentMetadata fields
203   SetFieldValues(*this,
204                  kernel_info,
205                  &GetAttachmentMetadataFieldString,
206                  &AttachmentMetadataToValue,
207                  ATTACHMENT_METADATA_FIELDS_BEGIN,
208                  ATTACHMENT_METADATA_FIELDS_END - 1);
209 
210   // Bit temps.
211   SetFieldValues(*this, kernel_info,
212                  &GetBitTempString, &BooleanToValue,
213                  BIT_TEMPS_BEGIN, BIT_TEMPS_END - 1);
214 
215   return kernel_info;
216 }
217 
EntryKernelMutationMapToValue(const EntryKernelMutationMap & mutations)218 base::ListValue* EntryKernelMutationMapToValue(
219     const EntryKernelMutationMap& mutations) {
220   base::ListValue* list = new base::ListValue();
221   for (EntryKernelMutationMap::const_iterator it = mutations.begin();
222        it != mutations.end(); ++it) {
223     list->Append(EntryKernelMutationToValue(it->second));
224   }
225   return list;
226 }
227 
EntryKernelMutationToValue(const EntryKernelMutation & mutation)228 base::DictionaryValue* EntryKernelMutationToValue(
229     const EntryKernelMutation& mutation) {
230   base::DictionaryValue* dict = new base::DictionaryValue();
231   dict->Set("original", mutation.original.ToValue(NULL));
232   dict->Set("mutated", mutation.mutated.ToValue(NULL));
233   return dict;
234 }
235 
236 }  // namespace syncer
237 }  // namespace syncable
238