// Copyright (c) 2010 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef CHROME_BROWSER_SYNC_PROFILE_SYNC_TEST_UTIL_H_ #define CHROME_BROWSER_SYNC_PROFILE_SYNC_TEST_UTIL_H_ #include #include "base/logging.h" #include "base/message_loop.h" #include "base/ref_counted.h" #include "base/scoped_ptr.h" #include "base/task.h" #include "base/thread.h" #include "base/waitable_event.h" #include "chrome/browser/chrome_thread.h" #include "chrome/browser/profile.h" #include "chrome/browser/webdata/web_database.h" #include "chrome/browser/sync/glue/bookmark_change_processor.h" #include "chrome/browser/sync/glue/bookmark_data_type_controller.h" #include "chrome/browser/sync/glue/bookmark_model_associator.h" #include "chrome/browser/sync/glue/change_processor.h" #include "chrome/browser/sync/glue/data_type_manager_impl.h" #include "chrome/browser/sync/notification_method.h" #include "chrome/browser/sync/profile_sync_factory.h" #include "chrome/browser/sync/profile_sync_service.h" #include "chrome/browser/sync/unrecoverable_error_handler.h" #include "chrome/common/notification_details.h" #include "chrome/common/notification_service.h" #include "chrome/common/notification_source.h" #include "chrome/common/notification_type.h" #include "chrome/test/sync/test_http_bridge_factory.h" #include "testing/gmock/include/gmock/gmock.h" ACTION_P(Notify, type) { NotificationService::current()->Notify(type, NotificationService::AllSources(), NotificationService::NoDetails()); } ACTION(QuitUIMessageLoop) { DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); MessageLoop::current()->Quit(); } ACTION_P(InvokeTask, task) { if (task) task->Run(); } class TestModelAssociatorHelper { public: template bool GetSyncIdForTaggedNode(ModelAssociatorImpl* associator, const std::string& tag, int64* sync_id) { std::wstring tag_wide; if (!UTF8ToWide(tag.c_str(), tag.length(), &tag_wide)) { NOTREACHED() << "Unable to convert UTF8 to wide for string: " << tag; return false; } sync_api::WriteTransaction trans( associator->sync_service()->backend()->GetUserShareHandle()); sync_api::ReadNode root(&trans); root.InitByRootLookup(); // First, try to find a node with the title among the root's children. // This will be the case if we are testing model persistence, and // are reloading a sync repository created earlier in the test. int64 last_child_id = sync_api::kInvalidId; for (int64 id = root.GetFirstChildId(); id != sync_api::kInvalidId; /***/) { sync_api::ReadNode child(&trans); child.InitByIdLookup(id); last_child_id = id; if (tag_wide == child.GetTitle()) { *sync_id = id; return true; } id = child.GetSuccessorId(); } sync_api::ReadNode predecessor_node(&trans); sync_api::ReadNode* predecessor = NULL; if (last_child_id != sync_api::kInvalidId) { predecessor_node.InitByIdLookup(last_child_id); predecessor = &predecessor_node; } sync_api::WriteNode node(&trans); // Create new fake tagged nodes at the end of the ordering. node.InitByCreation(ModelAssociatorImpl::model_type(), root, predecessor); node.SetIsFolder(true); node.SetTitle(tag_wide); node.SetExternalId(0); *sync_id = node.GetId(); return true; } ~TestModelAssociatorHelper() {} }; class ProfileSyncServiceObserverMock : public ProfileSyncServiceObserver { public: MOCK_METHOD0(OnStateChanged, void()); }; class ThreadNotificationService : public base::RefCountedThreadSafe { public: explicit ThreadNotificationService(base::Thread* notification_thread) : done_event_(false, false), notification_thread_(notification_thread) {} void Init() { DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); notification_thread_->message_loop()->PostTask( FROM_HERE, NewRunnableMethod(this, &ThreadNotificationService::InitTask)); done_event_.Wait(); } void TearDown() { DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); notification_thread_->message_loop()->PostTask( FROM_HERE, NewRunnableMethod(this, &ThreadNotificationService::TearDownTask)); done_event_.Wait(); } private: friend class base::RefCountedThreadSafe; void InitTask() { service_.reset(new NotificationService()); done_event_.Signal(); } void TearDownTask() { service_.reset(NULL); done_event_.Signal(); } base::WaitableEvent done_event_; base::Thread* notification_thread_; scoped_ptr service_; }; class ThreadNotifier : // NOLINT public base::RefCountedThreadSafe { public: explicit ThreadNotifier(base::Thread* notify_thread) : done_event_(false, false), notify_thread_(notify_thread) {} void Notify(NotificationType type, const NotificationDetails& details) { Notify(type, NotificationService::AllSources(), details); } void Notify(NotificationType type, const NotificationSource& source, const NotificationDetails& details) { DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); notify_thread_->message_loop()->PostTask( FROM_HERE, NewRunnableMethod(this, &ThreadNotifier::NotifyTask, type, source, details)); done_event_.Wait(); } private: friend class base::RefCountedThreadSafe; void NotifyTask(NotificationType type, const NotificationSource& source, const NotificationDetails& details) { NotificationService::current()->Notify(type, source, details); done_event_.Signal(); } base::WaitableEvent done_event_; base::Thread* notify_thread_; }; #endif // CHROME_BROWSER_SYNC_PROFILE_SYNC_TEST_UTIL_H_