Chromium Code Reviews| 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/engine/net/server_connection_manager.h" | 5 #include "sync/engine/net/server_connection_manager.h" |
| 6 | 6 |
| 7 #include <errno.h> | 7 #include <errno.h> |
| 8 | 8 |
| 9 #include <ostream> | 9 #include <ostream> |
| 10 #include <string> | 10 #include <string> |
| 11 #include <vector> | 11 #include <vector> |
| 12 | 12 |
| 13 #include "base/metrics/histogram.h" | 13 #include "base/metrics/histogram.h" |
| 14 #include "build/build_config.h" | 14 #include "build/build_config.h" |
| 15 #include "net/base/net_errors.h" | 15 #include "net/base/net_errors.h" |
| 16 #include "net/http/http_status_code.h" | 16 #include "net/http/http_status_code.h" |
| 17 #include "sync/engine/net/url_translator.h" | 17 #include "sync/engine/net/url_translator.h" |
| 18 #include "sync/engine/syncer.h" | 18 #include "sync/engine/syncer.h" |
| 19 #include "sync/internal_api/public/base/cancelation_signal.h" | |
| 19 #include "sync/protocol/sync.pb.h" | 20 #include "sync/protocol/sync.pb.h" |
| 20 #include "sync/syncable/directory.h" | 21 #include "sync/syncable/directory.h" |
| 21 #include "url/gurl.h" | 22 #include "url/gurl.h" |
| 22 | 23 |
| 23 namespace syncer { | 24 namespace syncer { |
| 24 | 25 |
| 25 using std::ostream; | 26 using std::ostream; |
| 26 using std::string; | 27 using std::string; |
| 27 using std::vector; | 28 using std::vector; |
| 28 | 29 |
| (...skipping 78 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 107 if (bytes_read != response->content_length) { | 108 if (bytes_read != response->content_length) { |
| 108 LOG(ERROR) << "Mismatched content lengths, server claimed " << | 109 LOG(ERROR) << "Mismatched content lengths, server claimed " << |
| 109 response->content_length << ", but sent " << bytes_read; | 110 response->content_length << ", but sent " << bytes_read; |
| 110 response->server_status = HttpResponse::IO_ERROR; | 111 response->server_status = HttpResponse::IO_ERROR; |
| 111 return false; | 112 return false; |
| 112 } | 113 } |
| 113 return true; | 114 return true; |
| 114 } | 115 } |
| 115 | 116 |
| 116 ServerConnectionManager::ScopedConnectionHelper::ScopedConnectionHelper( | 117 ServerConnectionManager::ScopedConnectionHelper::ScopedConnectionHelper( |
| 117 ServerConnectionManager* manager, Connection* connection) | 118 CancelationSignal* signaller, scoped_ptr<Connection> connection) |
| 118 : manager_(manager), connection_(connection) {} | 119 : cancelation_signal_(signaller), connection_(connection.Pass()) { |
| 120 if (!cancelation_signal_->TryRegisterHandler(this)) { | |
| 121 connection_.reset(); | |
| 122 } | |
| 123 } | |
| 124 | |
| 125 // This function may be called from another thread. | |
| 126 void ServerConnectionManager::ScopedConnectionHelper::OnStopRequested() { | |
| 127 DCHECK(connection_); | |
| 128 connection_->Abort(); | |
| 129 } | |
| 119 | 130 |
| 120 ServerConnectionManager::ScopedConnectionHelper::~ScopedConnectionHelper() { | 131 ServerConnectionManager::ScopedConnectionHelper::~ScopedConnectionHelper() { |
| 121 if (connection_) | 132 // We should be registered iff connection_.get() != NULL. |
|
tim (not reviewing)
2013/09/06 21:47:34
Is this true in tests as well? The SCM::MakeConne
rlarocque
2013/09/06 22:42:53
Good point. I wasn't thinking about the tests. I
| |
| 122 manager_->OnConnectionDestroyed(connection_.get()); | 133 if (connection_) { |
|
tim (not reviewing)
2013/09/06 21:47:34
I'd rather use connection_.get() so the scoped_ptr
rlarocque
2013/09/06 22:42:53
I've changed this to connection_.get().
What do y
| |
| 123 connection_.reset(); | 134 // It is important that this be called before this destructor completes. |
| 135 // Until the unregistration is complete, it's possible that the virtual | |
| 136 // OnStopRequested() function may be called from a different thread. We | |
| 137 // need to unregister it before destruction modifies our vptr. | |
| 138 cancelation_signal_->UnregisterHandler(this); | |
| 139 } | |
| 124 } | 140 } |
| 125 | 141 |
| 126 ServerConnectionManager::Connection* | 142 ServerConnectionManager::Connection* |
| 127 ServerConnectionManager::ScopedConnectionHelper::get() { | 143 ServerConnectionManager::ScopedConnectionHelper::get() { |
| 128 return connection_.get(); | 144 return connection_.get(); |
| 129 } | 145 } |
| 130 | 146 |
| 131 namespace { | 147 namespace { |
| 132 | 148 |
| 133 string StripTrailingSlash(const string& s) { | 149 string StripTrailingSlash(const string& s) { |
| (...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 170 } | 186 } |
| 171 | 187 |
| 172 ScopedServerStatusWatcher::~ScopedServerStatusWatcher() { | 188 ScopedServerStatusWatcher::~ScopedServerStatusWatcher() { |
| 173 conn_mgr_->SetServerStatus(response_->server_status); | 189 conn_mgr_->SetServerStatus(response_->server_status); |
| 174 } | 190 } |
| 175 | 191 |
| 176 ServerConnectionManager::ServerConnectionManager( | 192 ServerConnectionManager::ServerConnectionManager( |
| 177 const string& server, | 193 const string& server, |
| 178 int port, | 194 int port, |
| 179 bool use_ssl, | 195 bool use_ssl, |
| 180 bool use_oauth2_token) | 196 bool use_oauth2_token, |
| 197 CancelationSignal* cancelation_signal) | |
| 181 : sync_server_(server), | 198 : sync_server_(server), |
| 182 sync_server_port_(port), | 199 sync_server_port_(port), |
| 183 use_ssl_(use_ssl), | 200 use_ssl_(use_ssl), |
| 184 use_oauth2_token_(use_oauth2_token), | 201 use_oauth2_token_(use_oauth2_token), |
| 185 proto_sync_path_(kSyncServerSyncPath), | 202 proto_sync_path_(kSyncServerSyncPath), |
| 186 server_status_(HttpResponse::NONE), | 203 server_status_(HttpResponse::NONE), |
| 187 terminated_(false), | 204 cancelation_signal_(cancelation_signal) { |
| 188 active_connection_(NULL) { | |
| 189 } | 205 } |
| 190 | 206 |
| 191 ServerConnectionManager::~ServerConnectionManager() { | 207 ServerConnectionManager::~ServerConnectionManager() { |
| 192 } | 208 } |
| 193 | 209 |
| 194 ServerConnectionManager::Connection* | |
| 195 ServerConnectionManager::MakeActiveConnection() { | |
| 196 base::AutoLock lock(terminate_connection_lock_); | |
| 197 DCHECK(!active_connection_); | |
| 198 if (terminated_) | |
| 199 return NULL; | |
| 200 | |
| 201 active_connection_ = MakeConnection(); | |
| 202 return active_connection_; | |
| 203 } | |
| 204 | |
| 205 void ServerConnectionManager::OnConnectionDestroyed(Connection* connection) { | |
| 206 DCHECK(connection); | |
| 207 base::AutoLock lock(terminate_connection_lock_); | |
| 208 // |active_connection_| can be NULL already if it was aborted. Also, | |
| 209 // it can legitimately be a different Connection object if a new Connection | |
| 210 // was created after a previous one was Aborted and destroyed. | |
| 211 if (active_connection_ != connection) | |
| 212 return; | |
| 213 | |
| 214 active_connection_ = NULL; | |
| 215 } | |
| 216 | |
| 217 bool ServerConnectionManager::SetAuthToken(const std::string& auth_token) { | 210 bool ServerConnectionManager::SetAuthToken(const std::string& auth_token) { |
| 218 DCHECK(thread_checker_.CalledOnValidThread()); | 211 DCHECK(thread_checker_.CalledOnValidThread()); |
| 219 if (previously_invalidated_token != auth_token) { | 212 if (previously_invalidated_token != auth_token) { |
| 220 auth_token_.assign(auth_token); | 213 auth_token_.assign(auth_token); |
| 221 previously_invalidated_token = std::string(); | 214 previously_invalidated_token = std::string(); |
| 222 return true; | 215 return true; |
| 223 } | 216 } |
| 224 return false; | 217 return false; |
| 225 } | 218 } |
| 226 | 219 |
| (...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 271 // workaround for M29 blocker to avoid sending RPC to sync with known invalid | 264 // workaround for M29 blocker to avoid sending RPC to sync with known invalid |
| 272 // token but instead to trigger refreshing token in ProfileSyncService. Need | 265 // token but instead to trigger refreshing token in ProfileSyncService. Need |
| 273 // to clean it. | 266 // to clean it. |
| 274 if (auth_token.empty() || auth_token == "credentials_lost") { | 267 if (auth_token.empty() || auth_token == "credentials_lost") { |
| 275 params->response.server_status = HttpResponse::SYNC_AUTH_ERROR; | 268 params->response.server_status = HttpResponse::SYNC_AUTH_ERROR; |
| 276 return false; | 269 return false; |
| 277 } | 270 } |
| 278 | 271 |
| 279 // When our connection object falls out of scope, it clears itself from | 272 // When our connection object falls out of scope, it clears itself from |
| 280 // active_connection_. | 273 // active_connection_. |
| 281 ScopedConnectionHelper post(this, MakeActiveConnection()); | 274 ScopedConnectionHelper post(cancelation_signal_, MakeConnection()); |
| 282 if (!post.get()) { | 275 if (!post.get()) { |
| 283 params->response.server_status = HttpResponse::CONNECTION_UNAVAILABLE; | 276 params->response.server_status = HttpResponse::CONNECTION_UNAVAILABLE; |
| 284 return false; | 277 return false; |
| 285 } | 278 } |
| 286 | 279 |
| 287 // Note that |post| may be aborted by now, which will just cause Init to fail | 280 // Note that |post| may be aborted by now, which will just cause Init to fail |
| 288 // with CONNECTION_UNAVAILABLE. | 281 // with CONNECTION_UNAVAILABLE. |
| 289 bool ok = post.get()->Init( | 282 bool ok = post.get()->Init( |
| 290 path.c_str(), auth_token, params->buffer_in, ¶ms->response); | 283 path.c_str(), auth_token, params->buffer_in, ¶ms->response); |
| 291 | 284 |
| (...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 336 DCHECK(thread_checker_.CalledOnValidThread()); | 329 DCHECK(thread_checker_.CalledOnValidThread()); |
| 337 listeners_.AddObserver(listener); | 330 listeners_.AddObserver(listener); |
| 338 } | 331 } |
| 339 | 332 |
| 340 void ServerConnectionManager::RemoveListener( | 333 void ServerConnectionManager::RemoveListener( |
| 341 ServerConnectionEventListener* listener) { | 334 ServerConnectionEventListener* listener) { |
| 342 DCHECK(thread_checker_.CalledOnValidThread()); | 335 DCHECK(thread_checker_.CalledOnValidThread()); |
| 343 listeners_.RemoveObserver(listener); | 336 listeners_.RemoveObserver(listener); |
| 344 } | 337 } |
| 345 | 338 |
| 346 ServerConnectionManager::Connection* ServerConnectionManager::MakeConnection() | 339 scoped_ptr<ServerConnectionManager::Connection> |
| 340 ServerConnectionManager::MakeConnection() | |
| 347 { | 341 { |
| 348 return NULL; // For testing. | 342 return scoped_ptr<Connection>(); // For testing. |
| 349 } | |
| 350 | |
| 351 void ServerConnectionManager::TerminateAllIO() { | |
| 352 base::AutoLock lock(terminate_connection_lock_); | |
| 353 terminated_ = true; | |
| 354 if (active_connection_) | |
| 355 active_connection_->Abort(); | |
| 356 | |
| 357 // Sever our ties to this connection object. Note that it still may exist, | |
| 358 // since we don't own it, but it has been neutered. | |
| 359 active_connection_ = NULL; | |
| 360 } | 343 } |
| 361 | 344 |
| 362 std::ostream& operator << (std::ostream& s, const struct HttpResponse& hr) { | 345 std::ostream& operator << (std::ostream& s, const struct HttpResponse& hr) { |
| 363 s << " Response Code (bogus on error): " << hr.response_code; | 346 s << " Response Code (bogus on error): " << hr.response_code; |
| 364 s << " Content-Length (bogus on error): " << hr.content_length; | 347 s << " Content-Length (bogus on error): " << hr.content_length; |
| 365 s << " Server Status: " | 348 s << " Server Status: " |
| 366 << HttpResponse::GetServerConnectionCodeString(hr.server_status); | 349 << HttpResponse::GetServerConnectionCodeString(hr.server_status); |
| 367 return s; | 350 return s; |
| 368 } | 351 } |
| 369 | 352 |
| 370 } // namespace syncer | 353 } // namespace syncer |
| OLD | NEW |