 Chromium Code Reviews
 Chromium Code Reviews Issue 14963002:
  sync: Report GetUpdate triggers to the server  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@master
    
  
    Issue 14963002:
  sync: Report GetUpdate triggers to the server  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@master| OLD | NEW | 
|---|---|
| 1 // Copyright (c) 2013 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2013 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/sessions/nudge_tracker.h" | 5 #include "sync/sessions/nudge_tracker.h" | 
| 6 | |
| 7 #include "sync/internal_api/public/base/invalidation.h" | |
| 8 #include "sync/internal_api/public/sessions/sync_source_info.h" | |
| 6 #include "sync/protocol/sync.pb.h" | 9 #include "sync/protocol/sync.pb.h" | 
| 7 | 10 | 
| 8 namespace syncer { | 11 namespace syncer { | 
| 9 namespace sessions { | 12 namespace sessions { | 
| 10 | 13 | 
| 11 NudgeTracker::NudgeTracker() { } | 14 size_t NudgeTracker::kMaxPayloadsPerType = 10; | 
| 
tim (not reviewing)
2013/05/06 21:29:04
Might this be worth adding to client command as so
 
rlarocque
2013/05/07 18:45:39
I don't see much benefit in fiddling with this kno
 
tim (not reviewing)
2013/05/07 20:34:13
I agree with your logic, but the reason I ask is b
 
rlarocque
2013/05/07 21:50:57
I still don't think it's necessary, but I added it
 | |
| 15 | |
| 16 NudgeTracker::NudgeTracker() | |
| 17 : updates_source_(sync_pb::GetUpdatesCallerInfo::UNKNOWN), | |
| 18 invalidations_enabled_(false), | |
| 19 invalidations_out_of_sync_(true) { } | |
| 12 | 20 | 
| 13 NudgeTracker::~NudgeTracker() { } | 21 NudgeTracker::~NudgeTracker() { } | 
| 14 | 22 | 
| 15 void NudgeTracker::CoalesceSources(const SyncSourceInfo& source) { | 23 bool NudgeTracker::SyncRequired() { | 
| 16 CoalesceStates(source.types, &source_info_.types); | 24 if (!local_nudge_counts_.empty()) { | 
| 17 source_info_.updates_source = source.updates_source; | 25 return true; | 
| 26 } else if (!refresh_requested_counts_.empty()) { | |
| 27 return true; | |
| 28 } else if (!payload_list_map_.empty()) { | |
| 29 return true; | |
| 30 } else { | |
| 31 return false; | |
| 32 } | |
| 18 } | 33 } | 
| 19 | 34 | 
| 20 bool NudgeTracker::IsEmpty() { | 35 void NudgeTracker::RecordSuccessfulSyncCycle() { | 
| 21 return source_info_.types.empty(); | 36 updates_source_ = sync_pb::GetUpdatesCallerInfo::UNKNOWN; | 
| 37 local_nudge_counts_.clear(); | |
| 38 refresh_requested_counts_.clear(); | |
| 39 payload_list_map_.clear(); | |
| 40 locally_dropped_payload_types_.Clear(); | |
| 41 server_dropped_payload_types_.Clear(); | |
| 42 | |
| 43 // A successful cycle while invalidations are enabled puts us back into sync. | |
| 44 invalidations_out_of_sync_ = !invalidations_enabled_; | |
| 22 } | 45 } | 
| 23 | 46 | 
| 24 void NudgeTracker::Reset() { | 47 void NudgeTracker::RecordLocalChange(ModelTypeSet types) { | 
| 25 source_info_ = SyncSourceInfo(); | 48 // Don't overwrite an NOTIFICATION or DATATYPE_REFRESH source. The server | 
| 49 // makes some assumptions about the source; overriding these sources with | |
| 50 // LOCAL could lead to incorrect behaviour. This is part of the reason why | |
| 51 // we're deprecating 'source' in favor of 'origin'. | |
| 52 if (updates_source_ != sync_pb::GetUpdatesCallerInfo::NOTIFICATION | |
| 53 && updates_source_ != sync_pb::GetUpdatesCallerInfo::DATATYPE_REFRESH) { | |
| 54 updates_source_ = sync_pb::GetUpdatesCallerInfo::LOCAL; | |
| 55 } | |
| 56 | |
| 57 for (ModelTypeSet::Iterator it = types.First(); it.Good(); it.Inc()) { | |
| 58 local_nudge_counts_[it.Get()]++; | |
| 59 } | |
| 26 } | 60 } | 
| 27 | 61 | 
| 28 // TODO(rlarocque): This function often reports incorrect results. However, it | 62 void NudgeTracker::RecordLocalRefreshRequest(ModelTypeSet types) { | 
| 29 // is compatible with the "classic" behaviour. We would need to make the nudge | 63 // Don't overwrite an NOTIFICATION source. The server makes some assumptions | 
| 30 // tracker stop overwriting its own information (ie. fix crbug.com/231693) | 64 // about the source. Overriding this source with LOCAL could lead to | 
| 31 // before we could even try to report correct results. The main issue is that | 65 // incorrect behaviour. This is part of the reason why we're deprecating | 
| 32 // an notifications and local modifications nudges may overlap with each other | 66 // 'source' in favor of 'origin'. | 
| 33 // in sych a way that we lose track of which types were or were not locally | 67 if (updates_source_ != sync_pb::GetUpdatesCallerInfo::NOTIFICATION) { | 
| 34 // modified. | 68 updates_source_ = sync_pb::GetUpdatesCallerInfo::DATATYPE_REFRESH; | 
| 35 ModelTypeSet NudgeTracker::GetLocallyModifiedTypes() const { | |
| 36 ModelTypeSet locally_modified; | |
| 37 | |
| 38 if (source_info_.updates_source != sync_pb::GetUpdatesCallerInfo::LOCAL) { | |
| 39 return locally_modified; | |
| 40 } | 69 } | 
| 41 | 70 | 
| 42 for (ModelTypeInvalidationMap::const_iterator i = source_info_.types.begin(); | 71 for (ModelTypeSet::Iterator it = types.First(); it.Good(); it.Inc()) { | 
| 43 i != source_info().types.end(); ++i) { | 72 refresh_requested_counts_[it.Get()]++; | 
| 44 locally_modified.Put(i->first); | |
| 45 } | 73 } | 
| 46 return locally_modified; | 74 } | 
| 75 | |
| 76 void NudgeTracker::RecordRemoteInvalidation( | |
| 77 const ModelTypeInvalidationMap& invalidation_map) { | |
| 78 updates_source_ = sync_pb::GetUpdatesCallerInfo::NOTIFICATION; | |
| 79 | |
| 80 for (ModelTypeInvalidationMap::const_iterator i = invalidation_map.begin(); | |
| 81 i != invalidation_map.end(); ++i) { | |
| 82 const ModelType type = i->first; | |
| 83 const std::string& payload = i->second.payload; | |
| 84 | |
| 85 payload_list_map_[type].push_back(payload); | |
| 86 | |
| 87 // Respect the size limits on locally queued invalidation hints. | |
| 88 if (payload_list_map_[type].size() > kMaxPayloadsPerType) { | |
| 89 payload_list_map_[type].pop_front(); | |
| 90 locally_dropped_payload_types_.Put(type); | |
| 91 } | |
| 92 } | |
| 93 } | |
| 94 | |
| 95 void NudgeTracker::InvalidationsEnabled() { | |
| 96 invalidations_enabled_ = true; | |
| 97 } | |
| 98 | |
| 99 void NudgeTracker::InvalidationsDisabled() { | |
| 100 invalidations_enabled_ = false; | |
| 101 invalidations_out_of_sync_ = true; | |
| 102 } | |
| 103 | |
| 104 SyncSourceInfo NudgeTracker::source_info() const { | |
| 
tim (not reviewing)
2013/05/06 21:29:04
This isn't such a trivial method.  GetSourceInfo()
 
rlarocque
2013/05/07 18:45:39
Done.
 | |
| 105 // The old-style source added an empty hint for all locally nudge types. | |
| 106 // This will be overwritten with the proper hint for any types that were both | |
| 107 // locally nudged and invalidated. | |
| 108 ModelTypeSet nudged_types; | |
| 109 for (NudgeMap::const_iterator it = local_nudge_counts_.begin(); | |
| 110 it != local_nudge_counts_.end(); ++it) { | |
| 111 nudged_types.Put(it->first); | |
| 112 } | |
| 113 ModelTypeInvalidationMap invalidation_map = | |
| 114 ModelTypeSetToInvalidationMap(nudged_types, ""); | |
| 
tim (not reviewing)
2013/05/06 21:29:04
prefer std::string() to "" for efficiency & explic
 
rlarocque
2013/05/07 18:45:39
Done.
 | |
| 115 | |
| 116 // The old-style source info can contain only one hint per type. We grab the | |
| 117 // most recent, to mimic the old coalescing behaviour. | |
| 118 for (PayloadListMap::const_iterator it = payload_list_map_.begin(); | |
| 119 it != payload_list_map_.end(); ++it) { | |
| 120 ModelType type = it->first; | |
| 121 const PayloadList& list = it->second; | |
| 122 | |
| 123 if (!list.empty()) { | |
| 124 Invalidation invalidation; | |
| 125 invalidation.payload = list.back(); | |
| 126 if (invalidation_map.find(type) != invalidation_map.end()) { | |
| 127 invalidation_map[type] = invalidation; | |
| 128 } else { | |
| 129 invalidation_map.insert(std::make_pair(type, invalidation)); | |
| 130 } | |
| 131 } | |
| 132 } | |
| 133 | |
| 134 return SyncSourceInfo(updates_source_, invalidation_map); | |
| 135 } | |
| 136 | |
| 137 ModelTypeSet NudgeTracker::GetLocallyModifiedTypes() const { | |
| 138 ModelTypeSet nudged_types; | |
| 139 for (NudgeMap::const_iterator it = local_nudge_counts_.begin(); | |
| 140 it != local_nudge_counts_.end(); ++it) { | |
| 141 nudged_types.Put(it->first); | |
| 142 } | |
| 143 return nudged_types; | |
| 144 } | |
| 145 | |
| 146 sync_pb::GetUpdatesCallerInfo::GetUpdatesSource NudgeTracker::updates_source() | |
| 147 const { | |
| 148 return updates_source_; | |
| 149 } | |
| 150 | |
| 151 void NudgeTracker::FillProtoMessage( | |
| 152 ModelType type, | |
| 153 sync_pb::GetUpdateTriggers* msg) const { | |
| 154 // Fill the list of payloads, if applicable. The payloads must be ordered | |
| 155 // oldest to newest, so we insert them in the same order as we've been storing | |
| 156 // them internally. | |
| 157 PayloadListMap::const_iterator type_it = payload_list_map_.find(type); | |
| 158 if (type_it != payload_list_map_.end()) { | |
| 159 const PayloadList& payload_list = type_it->second; | |
| 160 for (PayloadList::const_iterator payload_it = payload_list.begin(); | |
| 161 payload_it != payload_list.end(); ++payload_it) { | |
| 162 msg->add_notification_hint(*payload_it); | |
| 163 } | |
| 164 } | |
| 165 | |
| 166 // TODO(rlarocque): Support Tango trickles. See crbug.com/223437. | |
| 167 // msg->set_server_dropped_hints(server_dropped_payload_types_.Has(type)); | |
| 168 | |
| 169 msg->set_client_dropped_hints(locally_dropped_payload_types_.Has(type)); | |
| 170 msg->set_invalidations_out_of_sync(invalidations_out_of_sync_); | |
| 171 | |
| 172 { | |
| 173 NudgeMap::const_iterator it = local_nudge_counts_.find(type); | |
| 174 if (it == local_nudge_counts_.end()) { | |
| 175 msg->set_local_modification_nudges(0); | |
| 176 } else { | |
| 177 msg->set_local_modification_nudges(it->second); | |
| 178 } | |
| 179 } | |
| 180 | |
| 181 { | |
| 182 NudgeMap::const_iterator it = refresh_requested_counts_.find(type); | |
| 183 if (it == refresh_requested_counts_.end()) { | |
| 184 msg->set_datatype_refresh_nudges(0); | |
| 185 } else { | |
| 186 msg->set_datatype_refresh_nudges(it->second); | |
| 187 } | |
| 188 } | |
| 47 } | 189 } | 
| 48 | 190 | 
| 49 } // namespace sessions | 191 } // namespace sessions | 
| 50 } // namespace syncer | 192 } // namespace syncer | 
| OLD | NEW |