// Copyright (c) 2011 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 "ppapi/tests/test_post_message.h" #include #include "ppapi/c/dev/ppb_testing_dev.h" #include "ppapi/c/pp_var.h" #include "ppapi/cpp/dev/scriptable_object_deprecated.h" #include "ppapi/cpp/instance.h" #include "ppapi/cpp/var.h" #include "ppapi/tests/pp_thread.h" #include "ppapi/tests/test_utils.h" #include "ppapi/tests/testing_instance.h" // Windows defines 'PostMessage', so we have to undef it. #ifdef PostMessage #undef PostMessage #endif REGISTER_TEST_CASE(PostMessage); namespace { const char kTestString[] = "Hello world!"; const bool kTestBool = true; const int32_t kTestInt = 42; const double kTestDouble = 42.0; const int32_t kThreadsToRun = 4; const int32_t kMessagesToSendPerThread = 10; // The struct that invoke_post_message_thread_func expects for its argument. // It includes the instance on which to invoke PostMessage, and the value to // pass to PostMessage. struct InvokePostMessageThreadArg { InvokePostMessageThreadArg(pp::Instance* i, const pp::Var& v) : instance(i), value_to_send(v) {} pp::Instance* instance; pp::Var value_to_send; }; void InvokePostMessageThreadFunc(void* user_data) { InvokePostMessageThreadArg* arg = static_cast(user_data); for (int32_t i = 0; i < kMessagesToSendPerThread; ++i) arg->instance->PostMessage(arg->value_to_send); delete arg; } #define FINISHED_WAITING_MESSAGE "TEST_POST_MESSAGE_FINISHED_WAITING" } // namespace TestPostMessage::TestPostMessage(TestingInstance* instance) : TestCase(instance) { } TestPostMessage::~TestPostMessage() { // Remove the special listener that only responds to a FINISHED_WAITING // string. See Init for where it gets added. std::string js_code; js_code += "var plugin = document.getElementById('plugin');" "plugin.removeEventListener('message'," " plugin.wait_for_messages_handler);" "delete plugin.wait_for_messages_handler;"; pp::Var exception; instance_->ExecuteScript(js_code, &exception); } bool TestPostMessage::Init() { bool success = InitTestingInterface(); // Set up a special listener that only responds to a FINISHED_WAITING string. // This is for use by WaitForMessages. std::string js_code; // Note the following code is dependent on some features of test_case.html. // E.g., it is assumed that the DOM element where the plugin is embedded has // an id of 'plugin', and there is a function 'IsTestingMessage' that allows // us to ignore the messages that are intended for use by the testing // framework itself. js_code += "var plugin = document.getElementById('plugin');" "var wait_for_messages_handler = function(message_event) {" " if (!IsTestingMessage(message_event.data) &&" " message_event.data === '" FINISHED_WAITING_MESSAGE "') {" " plugin.postMessage('" FINISHED_WAITING_MESSAGE "');" " }" "};" "plugin.addEventListener('message', wait_for_messages_handler);" // Stash it on the plugin so we can remove it in the destructor. "plugin.wait_for_messages_handler = wait_for_messages_handler;"; pp::Var exception; instance_->ExecuteScript(js_code, &exception); success = success && exception.is_undefined(); // Set up the JavaScript message event listener to echo the data part of the // message event back to us. success = success && AddEchoingListener("message_event.data"); message_data_.clear(); // Send a message that the first test will expect to receive. This is to // verify that we can send messages when the 'Instance::Init' function is on // the stack. instance_->PostMessage(pp::Var(kTestString)); return success; } void TestPostMessage::RunTest() { // Note: SendInInit must be first, because it expects to receive a message // that was sent in Init above. RUN_TEST(SendInInit); RUN_TEST(SendingData); RUN_TEST(MessageEvent); RUN_TEST(NoHandler); RUN_TEST(ExtraParam); if (testing_interface_->IsOutOfProcess()) RUN_TEST(NonMainThread); } void TestPostMessage::HandleMessage(const pp::Var& message_data) { if (message_data.is_string() && (message_data.AsString() == FINISHED_WAITING_MESSAGE)) testing_interface_->QuitMessageLoop(instance_->pp_instance()); else message_data_.push_back(message_data); } bool TestPostMessage::AddEchoingListener(const std::string& expression) { std::string js_code; // Note the following code is dependent on some features of test_case.html. // E.g., it is assumed that the DOM element where the plugin is embedded has // an id of 'plugin', and there is a function 'IsTestingMessage' that allows // us to ignore the messages that are intended for use by the testing // framework itself. js_code += "var plugin = document.getElementById('plugin');" "var message_handler = function(message_event) {" " if (!IsTestingMessage(message_event.data) &&" " !(message_event.data === '" FINISHED_WAITING_MESSAGE "')) {" " plugin.postMessage("; js_code += expression; js_code += " );" " }" "};" "plugin.addEventListener('message', message_handler);" // Maintain an array of all event listeners, attached to the // plugin. This is so that we can easily remove them later (see // ClearListeners()). "if (!plugin.eventListeners) plugin.eventListeners = [];" "plugin.eventListeners.push(message_handler);"; pp::Var exception; instance_->ExecuteScript(js_code, &exception); return exception.is_undefined(); } bool TestPostMessage::ClearListeners() { std::string js_code( "var plugin = document.getElementById('plugin');" "while (plugin.eventListeners.length) {" " plugin.removeEventListener('message', plugin.eventListeners.pop());" "}"); pp::Var exception; instance_->ExecuteScript(js_code, &exception); return(exception.is_undefined()); } int TestPostMessage::WaitForMessages() { size_t message_size_before = message_data_.size(); // We first post a FINISHED_WAITING_MESSAGE. This should be guaranteed to // come back _after_ any other incoming messages that were already pending. instance_->PostMessage(pp::Var(FINISHED_WAITING_MESSAGE)); testing_interface_->RunMessageLoop(instance_->pp_instance()); // Now that the FINISHED_WAITING_MESSAGE has been echoed back to us, we know // that all pending messages have been slurped up. Return the number we // received (which may be zero). return message_data_.size() - message_size_before; } std::string TestPostMessage::TestSendInInit() { ASSERT_EQ(WaitForMessages(), 1); // This test assumes Init already sent a message. ASSERT_EQ(message_data_.size(), 1); ASSERT_TRUE(message_data_.back().is_string()); ASSERT_EQ(message_data_.back().AsString(), kTestString); PASS(); } std::string TestPostMessage::TestSendingData() { // Set up the JavaScript message event listener to echo the data part of the // message event back to us. ASSERT_TRUE(ClearListeners()); ASSERT_TRUE(AddEchoingListener("message_event.data")); // Test sending a message to JavaScript for each supported type. The JS sends // the data back to us, and we check that they match. message_data_.clear(); instance_->PostMessage(pp::Var(kTestString)); // PostMessage is asynchronous, so we should not receive a response yet. ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_string()); ASSERT_EQ(message_data_.back().AsString(), kTestString); message_data_.clear(); instance_->PostMessage(pp::Var(kTestBool)); ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_bool()); ASSERT_EQ(message_data_.back().AsBool(), kTestBool); message_data_.clear(); instance_->PostMessage(pp::Var(kTestInt)); ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_number()); ASSERT_DOUBLE_EQ(message_data_.back().AsDouble(), static_cast(kTestInt)); message_data_.clear(); instance_->PostMessage(pp::Var(kTestDouble)); ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_number()); ASSERT_DOUBLE_EQ(message_data_.back().AsDouble(), kTestDouble); message_data_.clear(); instance_->PostMessage(pp::Var()); ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_undefined()); message_data_.clear(); instance_->PostMessage(pp::Var(pp::Var::Null())); ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_null()); ASSERT_TRUE(ClearListeners()); PASS(); } std::string TestPostMessage::TestMessageEvent() { // Set up the JavaScript message event listener to pass us some values from // the MessageEvent and make sure they match our expectations. // Have the listener pass back the type of message_event and make sure it's // "object". ASSERT_TRUE(ClearListeners()); ASSERT_TRUE(AddEchoingListener("typeof(message_event)")); message_data_.clear(); instance_->PostMessage(pp::Var(kTestInt)); ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_string()); ASSERT_EQ(message_data_.back().AsString(), "object"); ASSERT_TRUE(ClearListeners()); // Make sure all the non-data properties have the expected values. bool success = AddEchoingListener("((message_event.origin == '')" " && (message_event.lastEventId == '')" " && (message_event.source == null)" " && (message_event.ports.length == 0)" " && (message_event.bubbles == false)" " && (message_event.cancelable == false)" ")"); ASSERT_TRUE(success); message_data_.clear(); instance_->PostMessage(pp::Var(kTestInt)); ASSERT_EQ(message_data_.size(), 0); ASSERT_EQ(WaitForMessages(), 1); ASSERT_TRUE(message_data_.back().is_bool()); ASSERT_TRUE(message_data_.back().AsBool()); ASSERT_TRUE(ClearListeners()); // Add some event handlers to make sure they receive messages. ASSERT_TRUE(AddEchoingListener("1")); ASSERT_TRUE(AddEchoingListener("2")); ASSERT_TRUE(AddEchoingListener("3")); message_data_.clear(); instance_->PostMessage(pp::Var(kTestInt)); // Make sure we don't get a response in a re-entrant fashion. ASSERT_EQ(message_data_.size(), 0); // We should get 3 messages. ASSERT_EQ(WaitForMessages(), 3); // Copy to a vector of doubles and sort; w3c does not specify the order for // event listeners. (Copying is easier than writing an operator< for pp::Var.) // // See http://www.w3.org/TR/2000/REC-DOM-Level-2-Events-20001113/events.html. VarVector::iterator iter(message_data_.begin()), the_end(message_data_.end()); std::vector double_vec; for (; iter != the_end; ++iter) { ASSERT_TRUE(iter->is_number()); double_vec.push_back(iter->AsDouble()); } std::sort(double_vec.begin(), double_vec.end()); ASSERT_DOUBLE_EQ(double_vec[0], 1.0); ASSERT_DOUBLE_EQ(double_vec[1], 2.0); ASSERT_DOUBLE_EQ(double_vec[2], 3.0); ASSERT_TRUE(ClearListeners()); PASS(); } std::string TestPostMessage::TestNoHandler() { // Delete any lingering event listeners. ASSERT_TRUE(ClearListeners()); // Now send a message. We shouldn't get a response. message_data_.clear(); instance_->PostMessage(pp::Var()); ASSERT_EQ(WaitForMessages(), 0); ASSERT_TRUE(message_data_.empty()); PASS(); } std::string TestPostMessage::TestExtraParam() { // Delete any lingering event listeners. ASSERT_TRUE(ClearListeners()); // Add a listener that will respond with 1 and an empty array (where the // message port array would appear if it was Worker postMessage). ASSERT_TRUE(AddEchoingListener("1, []")); // Now send a message. We shouldn't get a response. message_data_.clear(); instance_->PostMessage(pp::Var()); ASSERT_EQ(WaitForMessages(), 0); ASSERT_TRUE(message_data_.empty()); PASS(); } std::string TestPostMessage::TestNonMainThread() { ASSERT_TRUE(ClearListeners()); ASSERT_TRUE(AddEchoingListener("message_event.data")); message_data_.clear(); // Set up a thread for each integer from 0 to (kThreadsToRun - 1). Make each // thread send the number that matches its index kMessagesToSendPerThread // times. For good measure, call postMessage from the main thread // kMessagesToSendPerThread times. At the end, we make sure we got all the // values we expected. PP_ThreadType threads[kThreadsToRun]; for (int32_t i = 0; i < kThreadsToRun; ++i) { // Set up a thread to send a value of i. void* arg = new InvokePostMessageThreadArg(instance_, pp::Var(i)); PP_CreateThread(&threads[i], &InvokePostMessageThreadFunc, arg); } // Invoke PostMessage right now to send a value of (kThreadsToRun). for (int32_t i = 0; i < kMessagesToSendPerThread; ++i) instance_->PostMessage(pp::Var(kThreadsToRun)); // Now join all threads. for (int32_t i = 0; i < kThreadsToRun; ++i) PP_JoinThread(threads[i]); // PostMessage is asynchronous, so we should not receive a response yet. ASSERT_EQ(message_data_.size(), 0); // Make sure we got all values that we expected. Note that because it's legal // for the JavaScript engine to treat our integers as floating points, we // can't just use std::find or equality comparison. So we instead, we convert // each incoming value to an integer, and count them in received_counts. int32_t expected_num = (kThreadsToRun + 1) * kMessagesToSendPerThread; // Count how many we receive per-index. std::vector expected_counts(kThreadsToRun + 1, kMessagesToSendPerThread); std::vector received_counts(kThreadsToRun + 1, 0); ASSERT_EQ(WaitForMessages(), expected_num); for (int32_t i = 0; i < expected_num; ++i) { const pp::Var& latest_var(message_data_[i]); ASSERT_TRUE(latest_var.is_int() || latest_var.is_double()); int32_t received_value = -1; if (latest_var.is_int()) { received_value = latest_var.AsInt(); } else if (latest_var.is_double()) { received_value = static_cast(latest_var.AsDouble() + 0.5); } ASSERT_TRUE(received_value >= 0); ASSERT_TRUE(received_value <= kThreadsToRun); ++received_counts[received_value]; } ASSERT_EQ(received_counts, expected_counts); PASS(); }