// Copyright (c) 2012 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. #include "sync/js/sync_js_controller.h" #include "base/message_loop/message_loop.h" #include "base/values.h" #include "sync/js/js_arg_list.h" #include "sync/js/js_event_details.h" #include "sync/js/js_test_util.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" namespace syncer { namespace { using ::testing::_; using ::testing::InSequence; using ::testing::Mock; using ::testing::StrictMock; class SyncJsControllerTest : public testing::Test { protected: void PumpLoop() { message_loop_.RunUntilIdle(); } private: base::MessageLoop message_loop_; }; ACTION_P(ReplyToMessage, reply_name) { arg2.Call(FROM_HERE, &JsReplyHandler::HandleJsReply, reply_name, JsArgList()); } TEST_F(SyncJsControllerTest, Messages) { InSequence dummy; // |mock_backend| needs to outlive |sync_js_controller|. StrictMock mock_backend; StrictMock mock_reply_handler; SyncJsController sync_js_controller; base::ListValue arg_list1, arg_list2; arg_list1.Append(new base::FundamentalValue(false)); arg_list2.Append(new base::FundamentalValue(5)); JsArgList args1(&arg_list1), args2(&arg_list2); EXPECT_CALL(mock_backend, SetJsEventHandler(_)); EXPECT_CALL(mock_backend, ProcessJsMessage("test1", HasArgs(args2), _)) .WillOnce(ReplyToMessage("test1_reply")); EXPECT_CALL(mock_backend, ProcessJsMessage("test2", HasArgs(args1), _)) .WillOnce(ReplyToMessage("test2_reply")); sync_js_controller.AttachJsBackend(mock_backend.AsWeakHandle()); sync_js_controller.ProcessJsMessage("test1", args2, mock_reply_handler.AsWeakHandle()); sync_js_controller.ProcessJsMessage("test2", args1, mock_reply_handler.AsWeakHandle()); // The replies should be waiting on our message loop. EXPECT_CALL(mock_reply_handler, HandleJsReply("test1_reply", _)); EXPECT_CALL(mock_reply_handler, HandleJsReply("test2_reply", _)); PumpLoop(); // Let destructor of |sync_js_controller| call RemoveBackend(). } TEST_F(SyncJsControllerTest, QueuedMessages) { // |mock_backend| needs to outlive |sync_js_controller|. StrictMock mock_backend; StrictMock mock_reply_handler; SyncJsController sync_js_controller; base::ListValue arg_list1, arg_list2; arg_list1.Append(new base::FundamentalValue(false)); arg_list2.Append(new base::FundamentalValue(5)); JsArgList args1(&arg_list1), args2(&arg_list2); // Should queue messages. sync_js_controller.ProcessJsMessage( "test1", args2, mock_reply_handler.AsWeakHandle()); sync_js_controller.ProcessJsMessage( "test2", args1, mock_reply_handler.AsWeakHandle()); // Should do nothing. PumpLoop(); Mock::VerifyAndClearExpectations(&mock_backend); // Should call the queued messages. EXPECT_CALL(mock_backend, SetJsEventHandler(_)); EXPECT_CALL(mock_backend, ProcessJsMessage("test1", HasArgs(args2), _)) .WillOnce(ReplyToMessage("test1_reply")); EXPECT_CALL(mock_backend, ProcessJsMessage("test2", HasArgs(args1), _)) .WillOnce(ReplyToMessage("test2_reply")); EXPECT_CALL(mock_reply_handler, HandleJsReply("test1_reply", _)); EXPECT_CALL(mock_reply_handler, HandleJsReply("test2_reply", _)); sync_js_controller.AttachJsBackend(mock_backend.AsWeakHandle()); PumpLoop(); // Should do nothing. sync_js_controller.AttachJsBackend(WeakHandle()); PumpLoop(); // Should also do nothing. sync_js_controller.AttachJsBackend(WeakHandle()); PumpLoop(); } TEST_F(SyncJsControllerTest, Events) { InSequence dummy; SyncJsController sync_js_controller; base::DictionaryValue details_dict1, details_dict2; details_dict1.SetString("foo", "bar"); details_dict2.SetInteger("baz", 5); JsEventDetails details1(&details_dict1), details2(&details_dict2); StrictMock event_handler1, event_handler2; EXPECT_CALL(event_handler1, HandleJsEvent("event", HasDetails(details1))); EXPECT_CALL(event_handler2, HandleJsEvent("event", HasDetails(details1))); EXPECT_CALL(event_handler1, HandleJsEvent("anotherevent", HasDetails(details2))); EXPECT_CALL(event_handler2, HandleJsEvent("anotherevent", HasDetails(details2))); sync_js_controller.AddJsEventHandler(&event_handler1); sync_js_controller.AddJsEventHandler(&event_handler2); sync_js_controller.HandleJsEvent("event", details1); sync_js_controller.HandleJsEvent("anotherevent", details2); sync_js_controller.RemoveJsEventHandler(&event_handler1); sync_js_controller.RemoveJsEventHandler(&event_handler2); sync_js_controller.HandleJsEvent("droppedevent", details2); PumpLoop(); } } // namespace } // namespace syncer