summaryrefslogtreecommitdiffstats
path: root/sync
diff options
context:
space:
mode:
authorpvalenzuela <pvalenzuela@chromium.org>2015-06-26 11:38:59 -0700
committerCommit bot <commit-bot@chromium.org>2015-06-26 18:39:30 +0000
commit4f4dc3834e1aa811300703074f824685d2cdeabc (patch)
tree30b55c2c21696f6bbe29175956400f0921ca41f6 /sync
parentac53d3da52014db50e638e96fd9f52817072fd64 (diff)
downloadchromium_src-4f4dc3834e1aa811300703074f824685d2cdeabc.zip
chromium_src-4f4dc3834e1aa811300703074f824685d2cdeabc.tar.gz
chromium_src-4f4dc3834e1aa811300703074f824685d2cdeabc.tar.bz2
Sync: store FakeServer as WeakPtr
This CL changes the FakeServerNetworkResources stack to store FakeServer as a WeakPtr. This is done because FakeServerHttpPostProvider posts a task to another thread and FakeServer may be deleted before the task executes. This CL adds a ThreadChecker to FakeServer to ensure that all actions (especially AsWeakPtr() and destruction) happen on the same thread. As a result of this thread restriction, FakeServerHelper has been changed to call all FakeServer methods on the UI thread. FakeServerHttpPostProvider's task-posting has also been reconfigured to be cleaner. This class now calls PostTask (instead of unnecessarily calling PostNonNestableTask previously) and has removed the possibility of FakeServer::HandleCommand (UI thread) calling a method on FakeServerHttpPostProvider (Sync thread). BUG=481192 Review URL: https://codereview.chromium.org/1149723007 Cr-Commit-Position: refs/heads/master@{#336422}
Diffstat (limited to 'sync')
-rw-r--r--sync/test/fake_server/android/fake_server_helper_android.cc2
-rw-r--r--sync/test/fake_server/fake_server.cc60
-rw-r--r--sync/test/fake_server/fake_server.h30
-rw-r--r--sync/test/fake_server/fake_server_http_post_provider.cc60
-rw-r--r--sync/test/fake_server/fake_server_http_post_provider.h26
-rw-r--r--sync/test/fake_server/fake_server_network_resources.cc6
-rw-r--r--sync/test/fake_server/fake_server_network_resources.h6
7 files changed, 131 insertions, 59 deletions
diff --git a/sync/test/fake_server/android/fake_server_helper_android.cc b/sync/test/fake_server/android/fake_server_helper_android.cc
index cfb42e9..2cc64f5 100644
--- a/sync/test/fake_server/android/fake_server_helper_android.cc
+++ b/sync/test/fake_server/android/fake_server_helper_android.cc
@@ -46,7 +46,7 @@ jlong FakeServerHelperAndroid::CreateNetworkResources(JNIEnv* env,
fake_server::FakeServer* fake_server_ptr =
reinterpret_cast<fake_server::FakeServer*>(fake_server);
syncer::NetworkResources* resources =
- new fake_server::FakeServerNetworkResources(fake_server_ptr);
+ new fake_server::FakeServerNetworkResources(fake_server_ptr->AsWeakPtr());
return reinterpret_cast<intptr_t>(resources);
}
diff --git a/sync/test/fake_server/fake_server.cc b/sync/test/fake_server/fake_server.cc
index dbe3105..e8f8dc0 100644
--- a/sync/test/fake_server/fake_server.cc
+++ b/sync/test/fake_server/fake_server.cc
@@ -170,17 +170,22 @@ FakeServer::FakeServer() : version_(0),
error_type_(sync_pb::SyncEnums::SUCCESS),
alternate_triggered_errors_(false),
request_counter_(0),
- network_enabled_(true) {
+ network_enabled_(true),
+ weak_ptr_factory_(this) {
keystore_keys_.push_back(kDefaultKeystoreKey);
- CHECK(CreateDefaultPermanentItems());
+
+ const bool create_result = CreateDefaultPermanentItems();
+ DCHECK(create_result) << "Permanent items were not created successfully.";
}
FakeServer::~FakeServer() {
+ DCHECK(thread_checker_.CalledOnValidThread());
STLDeleteContainerPairSecondPointers(entities_.begin(), entities_.end());
}
bool FakeServer::CreatePermanentBookmarkFolder(const std::string& server_tag,
const std::string& name) {
+ DCHECK(thread_checker_.CalledOnValidThread());
FakeServerEntity* entity =
PermanentEntity::Create(syncer::BOOKMARKS, server_tag, name,
ModelTypeToRootTag(syncer::BOOKMARKS));
@@ -222,15 +227,25 @@ void FakeServer::SaveEntity(FakeServerEntity* entity) {
}
void FakeServer::HandleCommand(const string& request,
- const HandleCommandCallback& callback) {
+ const base::Closure& completion_closure,
+ int* error_code,
+ int* response_code,
+ std::string* response) {
+ DCHECK(thread_checker_.CalledOnValidThread());
if (!network_enabled_) {
- callback.Run(net::ERR_FAILED, net::ERR_FAILED, string());
+ *error_code = net::ERR_FAILED;
+ *response_code = net::ERR_FAILED;
+ *response = string();
+ completion_closure.Run();
return;
}
request_counter_++;
if (!authenticated_) {
- callback.Run(0, net::HTTP_UNAUTHORIZED, string());
+ *error_code = 0;
+ *response_code = net::HTTP_UNAUTHORIZED;
+ *response = string();
+ completion_closure.Run();
return;
}
@@ -263,14 +278,20 @@ void FakeServer::HandleCommand(const string& request,
response_proto.mutable_commit());
break;
default:
- callback.Run(net::ERR_NOT_IMPLEMENTED, 0, string());;
+ *error_code = net::ERR_NOT_IMPLEMENTED;
+ *response_code = 0;
+ *response = string();
+ completion_closure.Run();
return;
}
if (!success) {
// TODO(pvalenzuela): Add logging here so that tests have more info about
// the failure.
- callback.Run(net::ERR_FAILED, 0, string());
+ *error_code = net::ERR_FAILED;
+ *response_code = 0;
+ *response = string();
+ completion_closure.Run();
return;
}
@@ -278,7 +299,11 @@ void FakeServer::HandleCommand(const string& request,
}
response_proto.set_store_birthday(store_birthday_);
- callback.Run(0, net::HTTP_OK, response_proto.SerializeAsString());
+
+ *error_code = 0;
+ *response_code = net::HTTP_OK;
+ *response = response_proto.SerializeAsString();
+ completion_closure.Run();
}
bool FakeServer::HandleGetUpdatesRequest(
@@ -469,6 +494,7 @@ bool FakeServer::HandleCommitRequest(
}
scoped_ptr<base::DictionaryValue> FakeServer::GetEntitiesAsDictionaryValue() {
+ DCHECK(thread_checker_.CalledOnValidThread());
scoped_ptr<base::DictionaryValue> dictionary(new base::DictionaryValue());
// Initialize an empty ListValue for all ModelTypes.
@@ -503,6 +529,7 @@ scoped_ptr<base::DictionaryValue> FakeServer::GetEntitiesAsDictionaryValue() {
std::vector<sync_pb::SyncEntity> FakeServer::GetSyncEntitiesByModelType(
ModelType model_type) {
std::vector<sync_pb::SyncEntity> sync_entities;
+ DCHECK(thread_checker_.CalledOnValidThread());
for (EntityMap::const_iterator it = entities_.begin(); it != entities_.end();
++it) {
FakeServerEntity* entity = it->second;
@@ -516,10 +543,12 @@ std::vector<sync_pb::SyncEntity> FakeServer::GetSyncEntitiesByModelType(
}
void FakeServer::InjectEntity(scoped_ptr<FakeServerEntity> entity) {
+ DCHECK(thread_checker_.CalledOnValidThread());
SaveEntity(entity.release());
}
bool FakeServer::SetNewStoreBirthday(const string& store_birthday) {
+ DCHECK(thread_checker_.CalledOnValidThread());
if (store_birthday_ == store_birthday)
return false;
@@ -528,14 +557,17 @@ bool FakeServer::SetNewStoreBirthday(const string& store_birthday) {
}
void FakeServer::SetAuthenticated() {
+ DCHECK(thread_checker_.CalledOnValidThread());
authenticated_ = true;
}
void FakeServer::SetUnauthenticated() {
+ DCHECK(thread_checker_.CalledOnValidThread());
authenticated_ = false;
}
bool FakeServer::TriggerError(const sync_pb::SyncEnums::ErrorType& error_type) {
+ DCHECK(thread_checker_.CalledOnValidThread());
if (triggered_actionable_error_.get()) {
DVLOG(1) << "Only one type of error can be triggered at any given time.";
return false;
@@ -550,6 +582,7 @@ bool FakeServer::TriggerActionableError(
const string& description,
const string& url,
const sync_pb::SyncEnums::Action& action) {
+ DCHECK(thread_checker_.CalledOnValidThread());
if (error_type_ != sync_pb::SyncEnums::SUCCESS) {
DVLOG(1) << "Only one type of error can be triggered at any given time.";
return false;
@@ -566,6 +599,7 @@ bool FakeServer::TriggerActionableError(
}
bool FakeServer::EnableAlternatingTriggeredErrors() {
+ DCHECK(thread_checker_.CalledOnValidThread());
if (error_type_ == sync_pb::SyncEnums::SUCCESS &&
!triggered_actionable_error_.get()) {
DVLOG(1) << "No triggered error set. Alternating can't be enabled.";
@@ -588,22 +622,27 @@ bool FakeServer::ShouldSendTriggeredError() const {
}
void FakeServer::AddObserver(Observer* observer) {
+ DCHECK(thread_checker_.CalledOnValidThread());
observers_.AddObserver(observer);
}
void FakeServer::RemoveObserver(Observer* observer) {
+ DCHECK(thread_checker_.CalledOnValidThread());
observers_.RemoveObserver(observer);
}
void FakeServer::EnableNetwork() {
+ DCHECK(thread_checker_.CalledOnValidThread());
network_enabled_ = true;
}
void FakeServer::DisableNetwork() {
+ DCHECK(thread_checker_.CalledOnValidThread());
network_enabled_ = false;
}
std::string FakeServer::GetBookmarkBarFolderId() const {
+ DCHECK(thread_checker_.CalledOnValidThread());
for (EntityMap::const_iterator it = entities_.begin(); it != entities_.end();
++it) {
FakeServerEntity* entity = it->second;
@@ -617,4 +656,9 @@ std::string FakeServer::GetBookmarkBarFolderId() const {
return "";
}
+base::WeakPtr<FakeServer> FakeServer::AsWeakPtr() {
+ DCHECK(thread_checker_.CalledOnValidThread());
+ return weak_ptr_factory_.GetWeakPtr();
+}
+
} // namespace fake_server
diff --git a/sync/test/fake_server/fake_server.h b/sync/test/fake_server/fake_server.h
index 8478076..f4374e7 100644
--- a/sync/test/fake_server/fake_server.h
+++ b/sync/test/fake_server/fake_server.h
@@ -13,6 +13,7 @@
#include "base/callback.h"
#include "base/memory/scoped_ptr.h"
#include "base/observer_list.h"
+#include "base/threading/thread_checker.h"
#include "base/values.h"
#include "sync/internal_api/public/base/model_type.h"
#include "sync/protocol/sync.pb.h"
@@ -20,12 +21,10 @@
namespace fake_server {
-// A fake version of the Sync server used for testing.
+// A fake version of the Sync server used for testing. This class is not thread
+// safe.
class FakeServer {
public:
- typedef base::Callback<void(int, int, const std::string&)>
- HandleCommandCallback;
-
class Observer {
public:
virtual ~Observer() {}
@@ -40,11 +39,16 @@ class FakeServer {
FakeServer();
virtual ~FakeServer();
- // Asynchronously handles a /command POST to the server. If the error_code is
- // passed to |callback| is 0 (success), the POST's response code and content
- // will also be passed.
+ // Handles a /command POST (with the given |request|) to the server. Three
+ // output arguments, |error_code|, |response_code|, and |response|, are used
+ // to pass data back to the caller. The command has failed if the value
+ // pointed to by |error_code| is nonzero. |completion_closure| will be called
+ // immediately before return.
void HandleCommand(const std::string& request,
- const HandleCommandCallback& callback);
+ const base::Closure& completion_closure,
+ int* error_code,
+ int* response_code,
+ std::string* response);
// Creates a DicionaryValue representation of all entities present in the
// server. The dictionary keys are the strings generated by ModelTypeToString
@@ -119,6 +123,9 @@ class FakeServer {
// Returns the entity ID of the Bookmark Bar folder.
std::string GetBookmarkBarFolderId() const;
+ // Returns the current FakeServer as a WeakPtr.
+ base::WeakPtr<FakeServer> AsWeakPtr();
+
private:
typedef std::map<std::string, FakeServerEntity*> EntityMap;
@@ -211,6 +218,13 @@ class FakeServer {
// When true, the server operates normally. When false, a failure is returned
// on every request. This is used to simulate a network failure on the client.
bool network_enabled_;
+
+ // Used to verify that FakeServer is only used from one thread.
+ base::ThreadChecker thread_checker_;
+
+ // Creates WeakPtr versions of the current FakeServer. This must be the last
+ // data member!
+ base::WeakPtrFactory<FakeServer> weak_ptr_factory_;
};
} // namespace fake_server
diff --git a/sync/test/fake_server/fake_server_http_post_provider.cc b/sync/test/fake_server/fake_server_http_post_provider.cc
index fbfe16f..75c5d46e 100644
--- a/sync/test/fake_server/fake_server_http_post_provider.cc
+++ b/sync/test/fake_server/fake_server_http_post_provider.cc
@@ -9,6 +9,7 @@
#include "base/bind.h"
#include "base/location.h"
#include "base/memory/ref_counted.h"
+#include "base/memory/weak_ptr.h"
#include "base/sequenced_task_runner.h"
#include "base/synchronization/waitable_event.h"
#include "sync/test/fake_server/fake_server.h"
@@ -18,9 +19,10 @@ using syncer::HttpPostProviderInterface;
namespace fake_server {
FakeServerHttpPostProviderFactory::FakeServerHttpPostProviderFactory(
- FakeServer* fake_server,
- scoped_refptr<base::SequencedTaskRunner> task_runner)
- : fake_server_(fake_server), task_runner_(task_runner) { }
+ const base::WeakPtr<FakeServer>& fake_server,
+ scoped_refptr<base::SequencedTaskRunner> fake_server_task_runner)
+ : fake_server_(fake_server),
+ fake_server_task_runner_(fake_server_task_runner) { }
FakeServerHttpPostProviderFactory::~FakeServerHttpPostProviderFactory() { }
@@ -28,7 +30,7 @@ void FakeServerHttpPostProviderFactory::Init(const std::string& user_agent) { }
HttpPostProviderInterface* FakeServerHttpPostProviderFactory::Create() {
FakeServerHttpPostProvider* http =
- new FakeServerHttpPostProvider(fake_server_, task_runner_);
+ new FakeServerHttpPostProvider(fake_server_, fake_server_task_runner_);
http->AddRef();
return http;
}
@@ -39,11 +41,10 @@ void FakeServerHttpPostProviderFactory::Destroy(
}
FakeServerHttpPostProvider::FakeServerHttpPostProvider(
- FakeServer* fake_server,
- scoped_refptr<base::SequencedTaskRunner> task_runner)
+ const base::WeakPtr<FakeServer>& fake_server,
+ scoped_refptr<base::SequencedTaskRunner> fake_server_task_runner)
: fake_server_(fake_server),
- task_runner_(task_runner),
- post_complete_(false, false) { }
+ fake_server_task_runner_(fake_server_task_runner) { }
FakeServerHttpPostProvider::~FakeServerHttpPostProvider() { }
@@ -65,26 +66,35 @@ void FakeServerHttpPostProvider::SetPostPayload(const char* content_type,
request_content_.assign(content, content_length);
}
-void FakeServerHttpPostProvider::OnPostComplete(int error_code,
- int response_code,
- const std::string& response) {
- post_error_code_ = error_code;
- post_response_code_ = response_code;
- response_ = response;
- post_complete_.Signal();
-}
-
bool FakeServerHttpPostProvider::MakeSynchronousPost(int* error_code,
int* response_code) {
// It is assumed that a POST is being made to /command.
- FakeServer::HandleCommandCallback callback = base::Bind(
- &FakeServerHttpPostProvider::OnPostComplete, base::Unretained(this));
- task_runner_->PostNonNestableTask(FROM_HERE,
- base::Bind(&FakeServer::HandleCommand,
- base::Unretained(fake_server_),
- base::ConstRef(request_content_),
- base::ConstRef(callback)));
- post_complete_.Wait();
+ int post_error_code = -1;
+ int post_response_code = -1;
+ std::string post_response;
+
+ base::WaitableEvent post_complete(false, false);
+ base::Closure signal_closure = base::Bind(&base::WaitableEvent::Signal,
+ base::Unretained(&post_complete));
+
+ bool result = fake_server_task_runner_->PostTask(
+ FROM_HERE,
+ base::Bind(&FakeServer::HandleCommand,
+ fake_server_,
+ base::ConstRef(request_content_),
+ base::ConstRef(signal_closure),
+ &post_error_code,
+ &post_response_code,
+ &post_response));
+
+ if (!result)
+ return false;
+
+ post_complete.Wait();
+ post_error_code_ = post_error_code;
+ post_response_code_ = post_response_code;
+ response_ = post_response;
+
*error_code = post_error_code_;
*response_code = post_response_code_;
return *error_code == 0;
diff --git a/sync/test/fake_server/fake_server_http_post_provider.h b/sync/test/fake_server/fake_server_http_post_provider.h
index 50d2317..593be1d 100644
--- a/sync/test/fake_server/fake_server_http_post_provider.h
+++ b/sync/test/fake_server/fake_server_http_post_provider.h
@@ -8,6 +8,7 @@
#include <string>
#include "base/memory/ref_counted.h"
+#include "base/memory/weak_ptr.h"
#include "base/sequenced_task_runner.h"
#include "base/synchronization/waitable_event.h"
#include "sync/internal_api/public/http_post_provider_factory.h"
@@ -22,8 +23,8 @@ class FakeServerHttpPostProvider
public base::RefCountedThreadSafe<FakeServerHttpPostProvider> {
public:
FakeServerHttpPostProvider(
- FakeServer* fake_server,
- scoped_refptr<base::SequencedTaskRunner> task_runner);
+ const base::WeakPtr<FakeServer>& fake_server,
+ scoped_refptr<base::SequencedTaskRunner> fake_server_task_runner);
// HttpPostProviderInterface implementation.
void SetExtraRequestHeaders(const char* headers) override;
@@ -43,12 +44,10 @@ class FakeServerHttpPostProvider
~FakeServerHttpPostProvider() override;
private:
- void OnPostComplete(int error_code,
- int response_code,
- const std::string& response);
-
- FakeServer* const fake_server_;
- scoped_refptr<base::SequencedTaskRunner> task_runner_;
+ // |fake_server_| should only be dereferenced on the same thread as
+ // |fake_server_task_runner_| runs on.
+ base::WeakPtr<FakeServer> fake_server_;
+ scoped_refptr<base::SequencedTaskRunner> fake_server_task_runner_;
std::string response_;
std::string request_url_;
@@ -58,7 +57,6 @@ class FakeServerHttpPostProvider
std::string extra_request_headers_;
int post_error_code_;
int post_response_code_;
- base::WaitableEvent post_complete_;
DISALLOW_COPY_AND_ASSIGN(FakeServerHttpPostProvider);
};
@@ -67,8 +65,8 @@ class FakeServerHttpPostProviderFactory
: public syncer::HttpPostProviderFactory {
public:
FakeServerHttpPostProviderFactory(
- FakeServer* fake_server,
- scoped_refptr<base::SequencedTaskRunner> task_runner);
+ const base::WeakPtr<FakeServer>& fake_server,
+ scoped_refptr<base::SequencedTaskRunner> fake_server_task_runner);
~FakeServerHttpPostProviderFactory() override;
// HttpPostProviderFactory:
@@ -77,8 +75,10 @@ class FakeServerHttpPostProviderFactory
void Destroy(syncer::HttpPostProviderInterface* http) override;
private:
- FakeServer* const fake_server_;
- scoped_refptr<base::SequencedTaskRunner> task_runner_;
+ // |fake_server_| should only be dereferenced on the same thread as
+ // |fake_server_task_runner_| runs on.
+ base::WeakPtr<FakeServer> fake_server_;
+ scoped_refptr<base::SequencedTaskRunner> fake_server_task_runner_;
DISALLOW_COPY_AND_ASSIGN(FakeServerHttpPostProviderFactory);
};
diff --git a/sync/test/fake_server/fake_server_network_resources.cc b/sync/test/fake_server/fake_server_network_resources.cc
index 8d7c647..b6b573d 100644
--- a/sync/test/fake_server/fake_server_network_resources.cc
+++ b/sync/test/fake_server/fake_server_network_resources.cc
@@ -5,6 +5,7 @@
#include "sync/test/fake_server/fake_server_network_resources.h"
#include "base/memory/scoped_ptr.h"
+#include "base/memory/weak_ptr.h"
#include "base/message_loop/message_loop.h"
#include "sync/internal_api/public/base/cancelation_signal.h"
#include "sync/internal_api/public/http_post_provider_factory.h"
@@ -18,8 +19,9 @@ using syncer::NetworkTimeUpdateCallback;
namespace fake_server {
-FakeServerNetworkResources::FakeServerNetworkResources(FakeServer* fake_server)
- : fake_server_(fake_server) { }
+FakeServerNetworkResources::FakeServerNetworkResources(
+ const base::WeakPtr<FakeServer>& fake_server)
+ : fake_server_(fake_server) { }
FakeServerNetworkResources::~FakeServerNetworkResources() {}
diff --git a/sync/test/fake_server/fake_server_network_resources.h b/sync/test/fake_server/fake_server_network_resources.h
index f6e1fe3..ba8579e 100644
--- a/sync/test/fake_server/fake_server_network_resources.h
+++ b/sync/test/fake_server/fake_server_network_resources.h
@@ -6,6 +6,7 @@
#define SYNC_TEST_FAKE_SERVER_FAKE_SERVER_NETWORK_RESOURCES_H_
#include "base/memory/scoped_ptr.h"
+#include "base/memory/weak_ptr.h"
#include "sync/internal_api/public/network_resources.h"
#include "sync/internal_api/public/network_time_update_callback.h"
@@ -20,7 +21,8 @@ class HttpPostProviderFactory;
class FakeServerNetworkResources : public syncer::NetworkResources {
public:
- explicit FakeServerNetworkResources(FakeServer* fake_server);
+ explicit FakeServerNetworkResources(
+ const base::WeakPtr<FakeServer>& fake_server);
~FakeServerNetworkResources() override;
// NetworkResources
@@ -31,7 +33,7 @@ class FakeServerNetworkResources : public syncer::NetworkResources {
syncer::CancelationSignal* cancelation_signal) override;
private:
- FakeServer* const fake_server_;
+ base::WeakPtr<FakeServer> fake_server_;
};
} // namespace fake_server