commit.cc revision 4e180b6a0b4720a9b8e9e959a882386f690f08ff
1// Copyright 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/engine/commit.h"
6
7#include "base/debug/trace_event.h"
8#include "sync/engine/commit_util.h"
9#include "sync/engine/sync_directory_commit_contribution.h"
10#include "sync/engine/syncer.h"
11#include "sync/engine/syncer_proto_util.h"
12#include "sync/sessions/sync_session.h"
13
14namespace syncer {
15
16Commit::Commit(
17    const std::map<ModelType, SyncDirectoryCommitContribution*>& contributions,
18    const sync_pb::ClientToServerMessage& message,
19    ExtensionsActivity::Records extensions_activity_buffer)
20  : contributions_(contributions),
21    deleter_(&contributions_),
22    message_(message),
23    extensions_activity_buffer_(extensions_activity_buffer),
24    cleaned_up_(false) {
25}
26
27Commit::~Commit() {
28  DCHECK(cleaned_up_);
29}
30
31Commit* Commit::Init(
32    ModelTypeSet requested_types,
33    size_t max_entries,
34    const std::string& account_name,
35    const std::string& cache_guid,
36    CommitContributorMap* contributor_map,
37    ExtensionsActivity* extensions_activity) {
38  // Gather per-type contributions.
39  ContributionMap contributions;
40  size_t num_entries = 0;
41  for (ModelTypeSet::Iterator it = requested_types.First();
42       it.Good(); it.Inc()) {
43    CommitContributorMap::iterator cm_it = contributor_map->find(it.Get());
44    if (cm_it == contributor_map->end()) {
45      NOTREACHED()
46          << "Could not find requested type " << ModelTypeToString(it.Get())
47          << " in contributor map.";
48      continue;
49    }
50    size_t spaces_remaining = max_entries - num_entries;
51    SyncDirectoryCommitContribution* contribution =
52        cm_it->second->GetContribution(spaces_remaining);
53    if (contribution) {
54      num_entries += contribution->GetNumEntries();
55      contributions.insert(std::make_pair(it.Get(), contribution));
56    }
57    if (num_entries == max_entries) {
58      break;  // No point in continuting to iterate in this case.
59    }
60  }
61
62  // Give up if no one had anything to commit.
63  if (contributions.empty())
64    return NULL;
65
66  sync_pb::ClientToServerMessage message;
67  message.set_message_contents(sync_pb::ClientToServerMessage::COMMIT);
68  message.set_share(account_name);
69
70  sync_pb::CommitMessage* commit_message = message.mutable_commit();
71  commit_message->set_cache_guid(cache_guid);
72
73  // Set extensions activity if bookmark commits are present.
74  ExtensionsActivity::Records extensions_activity_buffer;
75  ContributionMap::iterator it = contributions.find(syncer::BOOKMARKS);
76  if (it != contributions.end() && it->second->GetNumEntries() != 0) {
77    commit_util::AddExtensionsActivityToMessage(
78        extensions_activity,
79        &extensions_activity_buffer,
80        commit_message);
81  }
82
83  // Set the client config params.
84  ModelTypeSet enabled_types;
85  for (CommitContributorMap::iterator it = contributor_map->begin();
86       it != contributor_map->end(); ++it) {
87    enabled_types.Put(it->first);
88  }
89  commit_util::AddClientConfigParamsToMessage(enabled_types,
90                                                    commit_message);
91
92  // Finally, serialize all our contributions.
93  for (std::map<ModelType, SyncDirectoryCommitContribution*>::iterator it =
94           contributions.begin(); it != contributions.end(); ++it) {
95    it->second->AddToCommitMessage(&message);
96  }
97
98  // If we made it this far, then we've successfully prepared a commit message.
99  return new Commit(contributions, message, extensions_activity_buffer);
100}
101
102SyncerError Commit::PostAndProcessResponse(
103    sessions::SyncSession* session,
104    sessions::StatusController* status,
105    ExtensionsActivity* extensions_activity) {
106  ModelTypeSet request_types;
107  for (ContributionMap::const_iterator it = contributions_.begin();
108       it != contributions_.end(); ++it) {
109    request_types.Put(it->first);
110  }
111  session->mutable_status_controller()->set_commit_request_types(request_types);
112
113  DVLOG(1) << "Sending commit message.";
114  TRACE_EVENT_BEGIN0("sync", "PostCommit");
115  const SyncerError post_result = SyncerProtoUtil::PostClientToServerMessage(
116      &message_, &response_, session);
117  TRACE_EVENT_END0("sync", "PostCommit");
118
119  if (post_result != SYNCER_OK) {
120    LOG(WARNING) << "Post commit failed";
121    return post_result;
122  }
123
124  if (!response_.has_commit()) {
125    LOG(WARNING) << "Commit response has no commit body!";
126    return SERVER_RESPONSE_VALIDATION_FAILED;
127  }
128
129  size_t message_entries = message_.commit().entries_size();
130  size_t response_entries = response_.commit().entryresponse_size();
131  if (message_entries != response_entries) {
132    LOG(ERROR)
133       << "Commit response has wrong number of entries! "
134       << "Expected: " << message_entries << ", "
135       << "Got: " << response_entries;
136    return SERVER_RESPONSE_VALIDATION_FAILED;
137  }
138
139  // Let the contributors process the responses to each of their requests.
140  SyncerError processing_result = SYNCER_OK;
141  for (std::map<ModelType, SyncDirectoryCommitContribution*>::iterator it =
142       contributions_.begin(); it != contributions_.end(); ++it) {
143    TRACE_EVENT1("sync", "ProcessCommitResponse",
144                 "type", ModelTypeToString(it->first));
145    SyncerError type_result =
146        it->second->ProcessCommitResponse(response_, status);
147    if (processing_result == SYNCER_OK && type_result != SYNCER_OK) {
148      processing_result = type_result;
149    }
150  }
151
152  // Handle bookmarks' special extensions activity stats.
153  if (session->status_controller().
154          model_neutral_state().num_successful_bookmark_commits == 0) {
155    extensions_activity->PutRecords(extensions_activity_buffer_);
156  }
157
158  return processing_result;
159}
160
161void Commit::CleanUp() {
162  for (ContributionMap::iterator it = contributions_.begin();
163       it != contributions_.end(); ++it) {
164    it->second->CleanUp();
165  }
166  cleaned_up_ = true;
167}
168
169}  // namespace syncer
170