// 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. // Tests for CppBoundClass, in conjunction with CppBindingExample. Binds // a CppBindingExample class into JavaScript in a custom test shell and tests // the binding from the outside by loading JS into the shell. #include #include "base/bind.h" #include "base/bind_helpers.h" #include "base/message_loop.h" #include "base/string_util.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/WebKit/Source/Platform/chromium/public/WebURLRequest.h" #include "third_party/WebKit/Source/Platform/chromium/public/WebSize.h" #include "third_party/WebKit/Source/Platform/chromium/public/WebString.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrameClient.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebKit.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebSettings.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebViewClient.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebData.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebKitPlatformSupport.h" #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURL.h" #include "webkit/glue/cpp_binding_example.h" #include "webkit/glue/webkit_glue.h" #include "webkit/user_agent/user_agent.h" #include "webkit/user_agent/user_agent_util.h" using WebKit::WebFrame; using WebKit::WebView; using webkit_glue::CppArgumentList; using webkit_glue::CppBindingExample; using webkit_glue::CppVariant; namespace { class CppBindingExampleSubObject : public CppBindingExample { public: CppBindingExampleSubObject() { sub_value_.Set("sub!"); BindProperty("sub_value", &sub_value_); } private: CppVariant sub_value_; }; class CppBindingExampleWithOptionalFallback : public CppBindingExample { public: CppBindingExampleWithOptionalFallback() { BindProperty("sub_object", sub_object_.GetAsCppVariant()); } void set_fallback_method_enabled(bool state) { BindFallbackCallback(state ? base::Bind(&CppBindingExampleWithOptionalFallback::fallbackMethod, base::Unretained(this)) : CppBoundClass::Callback()); } // The fallback method does nothing, but because of it the JavaScript keeps // running when a nonexistent method is called on an object. void fallbackMethod(const CppArgumentList& args, CppVariant* result) { } private: CppBindingExampleSubObject sub_object_; }; class TestWebFrameClient : public WebKit::WebFrameClient { public: virtual void didClearWindowObject(WebKit::WebFrame* frame) OVERRIDE { example_bound_class_.BindToJavascript(frame, "example"); } void set_fallback_method_enabled(bool use_fallback) { example_bound_class_.set_fallback_method_enabled(use_fallback); } private: CppBindingExampleWithOptionalFallback example_bound_class_; }; class TestWebViewClient : public WebKit::WebViewClient { }; class CppBoundClassTest : public testing::Test, public WebKit::WebFrameClient { public: CppBoundClassTest() : webview_(NULL) { } virtual void SetUp() OVERRIDE { webview_ = WebView::create(&webview_client_); webview_->settings()->setJavaScriptEnabled(true); webview_->initializeMainFrame(&webframe_client_); webframe_client_.set_fallback_method_enabled(useFallback()); webkit_glue::SetUserAgent(webkit_glue::BuildUserAgentFromProduct( "TestShell/0.0.0.0"), false); WebKit::WebURLRequest urlRequest; urlRequest.initialize(); urlRequest.setURL(GURL("about:blank")); webframe()->loadRequest(urlRequest); } virtual void TearDown() OVERRIDE { if (webview_) webview_->close(); } WebFrame* webframe() { return webview_->mainFrame(); } // Wraps the given JavaScript snippet in "); webframe()->loadHTMLString(html, GURL("about:blank")); MessageLoop::current()->RunAllPending(); } // Executes the specified JavaScript and checks to be sure that the resulting // document text is exactly "SUCCESS". void CheckJavaScriptSuccess(const std::string& javascript) { ExecuteJavaScript(javascript); EXPECT_EQ("SUCCESS", UTF16ToASCII(webkit_glue::DumpDocumentText(webframe()))); } // Executes the specified JavaScript and checks that the resulting document // text is empty. void CheckJavaScriptFailure(const std::string& javascript) { ExecuteJavaScript(javascript); EXPECT_EQ("", UTF16ToASCII(webkit_glue::DumpDocumentText(webframe()))); } // Constructs a JavaScript snippet that evaluates and compares the left and // right expressions, printing "SUCCESS" to the page if they are equal and // printing their actual values if they are not. Any strings in the // expressions should be enclosed in single quotes, and no double quotes // should appear in either expression (even if escaped). (If a test case // is added that needs fancier quoting, Json::valueToQuotedString could be // used here. For now, it's not worth adding the dependency.) std::string BuildJSCondition(std::string left, std::string right) { return "var leftval = " + left + ";" + "var rightval = " + right + ";" + "if (leftval == rightval) {" + " document.writeln('SUCCESS');" + "} else {" + " document.writeln(\"" + left + " [\" + leftval + \"] != " + right + " [\" + rightval + \"]\");" + "}"; } protected: virtual bool useFallback() { return false; } private: WebView* webview_; TestWebFrameClient webframe_client_; TestWebViewClient webview_client_; }; class CppBoundClassWithFallbackMethodTest : public CppBoundClassTest { protected: virtual bool useFallback() { return true; } }; // Ensures that the example object has been bound to JS. TEST_F(CppBoundClassTest, ObjectExists) { std::string js = BuildJSCondition("typeof window.example", "'object'"); CheckJavaScriptSuccess(js); // An additional check to test our test. js = BuildJSCondition("typeof window.invalid_object", "'undefined'"); CheckJavaScriptSuccess(js); } TEST_F(CppBoundClassTest, PropertiesAreInitialized) { std::string js = BuildJSCondition("example.my_value", "10"); CheckJavaScriptSuccess(js); js = BuildJSCondition("example.my_other_value", "'Reinitialized!'"); CheckJavaScriptSuccess(js); } TEST_F(CppBoundClassTest, SubOject) { std::string js = BuildJSCondition("typeof window.example.sub_object", "'object'"); CheckJavaScriptSuccess(js); js = BuildJSCondition("example.sub_object.sub_value", "'sub!'"); CheckJavaScriptSuccess(js); } TEST_F(CppBoundClassTest, SetAndGetProperties) { // The property on the left will be set to the value on the right, then // checked to make sure it holds that same value. static const std::string tests[] = { "example.my_value", "7", "example.my_value", "'test'", "example.my_other_value", "3.14", "example.my_other_value", "false", "" // Array end marker: insert additional test pairs before this. }; for (int i = 0; tests[i] != ""; i += 2) { std::string left = tests[i]; std::string right = tests[i + 1]; // left = right; std::string js = left; js.append(" = "); js.append(right); js.append(";"); js.append(BuildJSCondition(left, right)); CheckJavaScriptSuccess(js); } } TEST_F(CppBoundClassTest, SetAndGetPropertiesWithCallbacks) { // TODO(dglazkov): fix NPObject issues around failing property setters and // getters and add tests for situations when GetProperty or SetProperty fail. std::string js = "var result = 'SUCCESS';\n" "example.my_value_with_callback = 10;\n" "if (example.my_value_with_callback != 10)\n" " result = 'FAIL: unable to set property.';\n" "example.my_value_with_callback = 11;\n" "if (example.my_value_with_callback != 11)\n" " result = 'FAIL: unable to set property again';\n" "if (example.same != 42)\n" " result = 'FAIL: same property should always be 42';\n" "example.same = 24;\n" "if (example.same != 42)\n" " result = 'FAIL: same property should always be 42';\n" "document.writeln(result);\n"; CheckJavaScriptSuccess(js); } TEST_F(CppBoundClassTest, InvokeMethods) { // The expression on the left is expected to return the value on the right. static const std::string tests[] = { "example.echoValue(true)", "true", "example.echoValue(13)", "13", "example.echoValue(2.718)", "2.718", "example.echoValue('yes')", "'yes'", "example.echoValue()", "null", // Too few arguments "example.echoType(false)", "true", "example.echoType(19)", "3.14159", "example.echoType(9.876)", "3.14159", "example.echoType('test string')", "'Success!'", "example.echoType()", "null", // Too few arguments // Comparing floats that aren't integer-valued is usually problematic due // to rounding, but exact powers of 2 should also be safe. "example.plus(2.5, 18.0)", "20.5", "example.plus(2, 3.25)", "5.25", "example.plus(2, 3)", "5", "example.plus()", "null", // Too few arguments "example.plus(1)", "null", // Too few arguments "example.plus(1, 'test')", "null", // Wrong argument type "example.plus('test', 2)", "null", // Wrong argument type "example.plus('one', 'two')", "null", // Wrong argument type "" // Array end marker: insert additional test pairs before this. }; for (int i = 0; tests[i] != ""; i+= 2) { std::string left = tests[i]; std::string right = tests[i + 1]; std::string js = BuildJSCondition(left, right); CheckJavaScriptSuccess(js); } std::string js = "example.my_value = 3.25; example.my_other_value = 1.25;"; js.append(BuildJSCondition( "example.plus(example.my_value, example.my_other_value)", "4.5")); CheckJavaScriptSuccess(js); } // Tests that invoking a nonexistent method with no fallback method stops the // script's execution TEST_F(CppBoundClassTest, InvokeNonexistentMethodNoFallback) { std::string js = "example.nonExistentMethod();document.writeln('SUCCESS');"; CheckJavaScriptFailure(js); } // Ensures existent methods can be invoked successfully when the fallback method // is used TEST_F(CppBoundClassWithFallbackMethodTest, InvokeExistentMethodsWithFallback) { std::string js = BuildJSCondition("example.echoValue(34)", "34"); CheckJavaScriptSuccess(js); } } // namespace