// Copyright (c) 2009 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 "base/file_util.h" #include "base/json_reader.h" #include "base/path_service.h" #include "base/string_util.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/render_messages.h" #include "chrome/renderer/extensions/extension_process_bindings.h" #include "chrome/renderer/extensions/renderer_extension_bindings.h" #include "chrome/test/render_view_test.h" #include "testing/gtest/include/gtest/gtest.h" class ExtensionAPIClientTest : public RenderViewTest { protected: virtual void SetUp() { RenderViewTest::SetUp(); render_thread_.sink().ClearMessages(); LoadHTML(""); } std::string GetConsoleMessage() { const IPC::Message* message = render_thread_.sink().GetUniqueMessageMatching( ViewHostMsg_AddMessageToConsole::ID); ViewHostMsg_AddMessageToConsole::Param params; if (message) { ViewHostMsg_AddMessageToConsole::Read(message, ¶ms); render_thread_.sink().ClearMessages(); return WideToASCII(params.a); } else { return ""; } } void ExpectJsFail(const std::string& js, const std::string& message) { ExecuteJavaScript(js.c_str()); EXPECT_EQ(message, GetConsoleMessage()); render_thread_.sink().ClearMessages(); } void ExpectJsPass(const std::string& js, const std::string& function, const std::string& arg1) { ExecuteJavaScript(js.c_str()); const IPC::Message* request_msg = render_thread_.sink().GetUniqueMessageMatching( ViewHostMsg_ExtensionRequest::ID); ASSERT_EQ("", GetConsoleMessage()) << js; ASSERT_TRUE(request_msg) << js; ViewHostMsg_ExtensionRequest::Param params; ViewHostMsg_ExtensionRequest::Read(request_msg, ¶ms); ASSERT_EQ(function.c_str(), params.a) << js; Value* args = NULL; ASSERT_TRUE(params.b.IsType(Value::TYPE_LIST)); ASSERT_TRUE(static_cast(¶ms.b)->Get(0, &args)); JSONReader reader; scoped_ptr arg1_value(reader.JsonToValue(arg1, false, false)); ASSERT_TRUE(args->Equals(arg1_value.get())) << js; render_thread_.sink().ClearMessages(); } }; // Tests that callback dispatching works correctly and that JSON is properly // deserialized before handing off to the extension code. We use the createTab // API here, but we could use any of them since they all dispatch callbacks the // same way. TEST_F(ExtensionAPIClientTest, CallbackDispatching) { ExecuteJavaScript( "function assert(truth, message) {" " if (!truth) {" " throw new Error(message);" " }" "}" "function callback(result) {" " assert(typeof result == 'object', 'result not object');" " assert(JSON.stringify(result) == '{\"id\":1,\"index\":1,\"windowId\":1," "\"selected\":true," "\"url\":\"http://www.google.com/\"}'," " 'incorrect result');" " console.log('pass')" "}" "chrome.tabs.create({}, callback);" ); EXPECT_EQ("", GetConsoleMessage()); // Ok, we should have gotten a message to create a tab, grab the callback ID. const IPC::Message* request_msg = render_thread_.sink().GetUniqueMessageMatching( ViewHostMsg_ExtensionRequest::ID); ASSERT_TRUE(request_msg); ViewHostMsg_ExtensionRequest::Param params; ViewHostMsg_ExtensionRequest::Read(request_msg, ¶ms); int callback_id = params.c; ASSERT_GE(callback_id, 0); // Now send the callback a response ExtensionProcessBindings::HandleResponse( callback_id, true, "{\"id\":1,\"index\":1,\"windowId\":1,\"selected\":true," "\"url\":\"http://www.google.com/\"}", ""); // And verify that it worked ASSERT_EQ("pass", GetConsoleMessage()); } // The remainder of these tests exercise the client side of the various // extension functions. We test both error and success conditions, but do not // test errors exhaustively as json schema code is well tested by itself. // Window API tests TEST_F(ExtensionAPIClientTest, GetWindow) { ExpectJsFail("chrome.windows.get(32, function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.windows.get(32);", "Uncaught Error: Parameter 1 is required."); ExpectJsFail("chrome.windows.get('abc', function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.get(1, 1);", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); ExpectJsPass("chrome.windows.get(2, function(){})", "windows.get", "2"); } TEST_F(ExtensionAPIClientTest, GetCurentWindow) { ExpectJsFail("chrome.windows.getCurrent(function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.windows.getCurrent();", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.windows.getCurrent('abc');", "Uncaught Error: Invalid value for argument 0. " "Expected 'function' but got 'string'."); ExpectJsPass("chrome.windows.getCurrent(function(){})", "windows.getCurrent", "null"); } // http://crbug.com/22248 TEST_F(ExtensionAPIClientTest, DISABLED_GetLastFocusedWindow) { ExpectJsFail("chrome.windows.getLastFocused(function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.windows.getLastFocused();", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.windows.getLastFocused('abc');", "Uncaught Error: Invalid value for argument 0. " "Expected 'function' but got 'string'."); ExpectJsPass("chrome.windows.getLastFocused(function(){})", "windows.getLastFocused", "null"); } TEST_F(ExtensionAPIClientTest, GetAllWindows) { ExpectJsFail("chrome.windows.getAll({populate: true}, function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.windows.getAll(1, function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'object' but got 'integer'."); ExpectJsPass("chrome.windows.getAll({populate:true}, function(){})", "windows.getAll", "{\"populate\":true}"); ExpectJsPass("chrome.windows.getAll(null, function(){})", "windows.getAll", "null"); ExpectJsPass("chrome.windows.getAll({}, function(){})", "windows.getAll", "{}"); ExpectJsPass("chrome.windows.getAll(undefined, function(){})", "windows.getAll", "null"); } TEST_F(ExtensionAPIClientTest, CreateWindow) { ExpectJsFail("chrome.windows.create({url: 1}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'url': Expected 'string' but got 'integer'."); ExpectJsFail("chrome.windows.create({left: 'foo'}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'left': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.create({top: 'foo'}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'top': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.create({width: 'foo'}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'width': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.create({height: 'foo'}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'height': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.create({foo: 42}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'foo': Unexpected property."); ExpectJsPass("chrome.windows.create({" " url:'http://www.google.com/'," " left:0," " top: 10," " width:100," " height:200" "})", "windows.create", "{\"url\":\"http://www.google.com/\"," "\"left\":0," "\"top\":10," "\"width\":100," "\"height\":200}"); } TEST_F(ExtensionAPIClientTest, UpdateWindow) { ExpectJsFail("chrome.windows.update(null);", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.windows.update(42, {left: 'foo'});", "Uncaught Error: Invalid value for argument 1. Property " "'left': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.update(42, {top: 'foo'});", "Uncaught Error: Invalid value for argument 1. Property " "'top': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.update(42, {height: false});", "Uncaught Error: Invalid value for argument 1. Property " "'height': Expected 'integer' but got 'boolean'."); ExpectJsFail("chrome.windows.update(42, {width: false});", "Uncaught Error: Invalid value for argument 1. Property " "'width': Expected 'integer' but got 'boolean'."); ExpectJsFail("chrome.windows.update(42, {foo: false});", "Uncaught Error: Invalid value for argument 1. Property " "'foo': Unexpected property."); ExpectJsPass("chrome.windows.update(42, {" " width:100," " height:200" "})", "windows.update", "[42," "{\"width\":100," "\"height\":200}]"); } TEST_F(ExtensionAPIClientTest, RemoveWindow) { ExpectJsFail("chrome.windows.remove(32, function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.windows.remove('abc', function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'string'."); ExpectJsFail("chrome.windows.remove(1, 1);", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); ExpectJsPass("chrome.windows.remove(2, function(){})", "windows.remove", "2"); ExpectJsPass("chrome.windows.remove(2)", "windows.remove", "2"); } // Tab API tests TEST_F(ExtensionAPIClientTest, GetTab) { ExpectJsFail("chrome.tabs.get(32, function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.tabs.get(32);", "Uncaught Error: Parameter 1 is required."); ExpectJsFail("chrome.tabs.get('abc', function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'string'."); ExpectJsFail("chrome.tabs.get(1, 1);", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); ExpectJsPass("chrome.tabs.get(2, function(){})", "tabs.get", "2"); } #if defined(OS_WIN) TEST_F(ExtensionAPIClientTest, DetectTabLanguage) { ExpectJsFail("chrome.tabs.detectLanguage(32, function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.tabs.detectLanguage('abc', function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'string'."); ExpectJsFail("chrome.tabs.detectLanguage(1, 1);", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); ExpectJsPass("chrome.tabs.detectLanguage(null, function(){})", "tabs.detectLanguage", "null"); } #endif TEST_F(ExtensionAPIClientTest, GetSelectedTab) { ExpectJsFail("chrome.tabs.getSelected(32, function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.tabs.getSelected(32);", "Uncaught Error: Parameter 1 is required."); ExpectJsFail("chrome.tabs.getSelected('abc', function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'string'."); ExpectJsFail("chrome.tabs.getSelected(1, 1);", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); ExpectJsPass("chrome.tabs.getSelected(2, function(){})", "tabs.getSelected", "2"); ExpectJsPass("chrome.tabs.getSelected(null, function(){})", "tabs.getSelected", "null"); } TEST_F(ExtensionAPIClientTest, GetAllTabsInWindow) { ExpectJsFail("chrome.tabs.getAllInWindow(42, function(){}, 'asd');", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.tabs.getAllInWindow(32);", "Uncaught Error: Parameter 1 is required."); ExpectJsFail("chrome.tabs.getAllInWindow(1, 1);", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); ExpectJsFail("chrome.tabs.getAllInWindow('asd', function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'string'."); ExpectJsPass("chrome.tabs.getAllInWindow(32, function(){})", "tabs.getAllInWindow", "32"); ExpectJsPass("chrome.tabs.getAllInWindow(undefined, function(){})", "tabs.getAllInWindow", "null"); } TEST_F(ExtensionAPIClientTest, CreateTab) { ExpectJsFail("chrome.tabs.create({windowId: 'foo'}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'windowId': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.tabs.create({url: 42}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'url': Expected 'string' but got 'integer'."); ExpectJsFail("chrome.tabs.create({foo: 42}, function(){});", "Uncaught Error: Invalid value for argument 0. Property " "'foo': Unexpected property."); ExpectJsPass("chrome.tabs.create({" " url:'http://www.google.com/'," " selected:true," " index: 2," " windowId:4" "})", "tabs.create", "{\"url\":\"http://www.google.com/\"," "\"selected\":true," "\"index\":2," "\"windowId\":4}"); } TEST_F(ExtensionAPIClientTest, UpdateTab) { ExpectJsFail("chrome.tabs.update(null);", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.tabs.update(42, {selected: 'foo'});", "Uncaught Error: Invalid value for argument 1. Property " "'selected': Expected 'boolean' but got 'string'."); ExpectJsFail("chrome.tabs.update(42, {url: 42});", "Uncaught Error: Invalid value for argument 1. Property " "'url': Expected 'string' but got 'integer'."); ExpectJsPass("chrome.tabs.update(42, {" " url:'http://www.google.com/'," " selected:true" "})", "tabs.update", "[42," "{\"url\":\"http://www.google.com/\"," "\"selected\":true}]"); } TEST_F(ExtensionAPIClientTest, MoveTab) { ExpectJsFail("chrome.tabs.move(null);", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.tabs.move(42, {index: 'foo'});", "Uncaught Error: Invalid value for argument 1. Property " "'index': Expected 'integer' but got 'string'."); ExpectJsFail("chrome.tabs.move(42, {index: 3, windowId: 'foo'});", "Uncaught Error: Invalid value for argument 1. Property " "'windowId': Expected 'integer' but got 'string'."); ExpectJsPass("chrome.tabs.move(42, {" " index:3," " windowId:21" "})", "tabs.move", "[42," "{\"index\":3," "\"windowId\":21}]"); } TEST_F(ExtensionAPIClientTest, RemoveTab) { ExpectJsFail("chrome.tabs.remove(32, function(){}, 20);", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.tabs.remove('abc', function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'string'."); ExpectJsFail("chrome.tabs.remove(1, 1);", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); ExpectJsPass("chrome.tabs.remove(2, function(){})", "tabs.remove", "2"); ExpectJsPass("chrome.tabs.remove(2)", "tabs.remove", "2"); } TEST_F(ExtensionAPIClientTest, CaptureVisibleTab) { ExpectJsFail("chrome.tabs.captureVisibleTab(0);", "Uncaught Error: Parameter 1 is required."); ExpectJsFail("chrome.tabs.captureVisibleTab(function(){}, 0)", "Uncaught Error: Invalid value for argument 0. " "Expected 'integer' but got 'function'."); ExpectJsPass("chrome.tabs.captureVisibleTab(null, function(img){});", "tabs.captureVisibleTab", "null"); } // Bookmark API tests // TODO(erikkay) add more variations here TEST_F(ExtensionAPIClientTest, CreateBookmark) { ExpectJsFail( "chrome.bookmarks.create({parentId:0, title:0}, function(){})", "Uncaught Error: Invalid value for argument 0. " "Property 'parentId': Expected 'string' but got 'integer', " "Property 'title': Expected 'string' but got 'integer'."); ExpectJsPass( "chrome.bookmarks.create({parentId:'0', title:'x'}, function(){})", "bookmarks.create", "{\"parentId\":\"0\",\"title\":\"x\"}"); } TEST_F(ExtensionAPIClientTest, GetBookmarks) { ExpectJsPass("chrome.bookmarks.get('0', function(){});", "bookmarks.get", "\"0\""); ExpectJsPass("chrome.bookmarks.get(['0','1','2','3'], function(){});", "bookmarks.get", "[\"0\",\"1\",\"2\",\"3\"]"); ExpectJsFail("chrome.bookmarks.get(null, function(){});", "Uncaught Error: Parameter 0 is required."); // TODO(erikkay) This is succeeding, when it should fail. // BUG=13719 #if 0 ExpectJsFail("chrome.bookmarks.get({}, function(){});", "Uncaught Error: Invalid value for argument 0. " "Expected 'array' but got 'object'."); #endif } TEST_F(ExtensionAPIClientTest, GetBookmarkChildren) { ExpectJsPass("chrome.bookmarks.getChildren('42', function(){});", "bookmarks.getChildren", "\"42\""); } TEST_F(ExtensionAPIClientTest, GetBookmarkTree) { ExpectJsPass("chrome.bookmarks.getTree(function(){});", "bookmarks.getTree", "null"); } TEST_F(ExtensionAPIClientTest, SearchBookmarks) { ExpectJsPass("chrome.bookmarks.search('hello',function(){});", "bookmarks.search", "\"hello\""); } TEST_F(ExtensionAPIClientTest, RemoveBookmark) { ExpectJsPass("chrome.bookmarks.remove('42');", "bookmarks.remove", "\"42\""); } TEST_F(ExtensionAPIClientTest, RemoveBookmarkTree) { ExpectJsPass("chrome.bookmarks.removeTree('42');", "bookmarks.removeTree", "\"42\""); } TEST_F(ExtensionAPIClientTest, MoveBookmark) { ExpectJsPass("chrome.bookmarks.move('42',{parentId:'1',index:0});", "bookmarks.move", "[\"42\",{\"parentId\":\"1\",\"index\":0}]"); } TEST_F(ExtensionAPIClientTest, SetBookmarkTitle) { ExpectJsPass("chrome.bookmarks.update('42',{title:'x'});", "bookmarks.update", "[\"42\",{\"title\":\"x\"}]"); } TEST_F(ExtensionAPIClientTest, EnablePageAction) { // Basic old-school enablePageAction call. ExpectJsPass("chrome.pageActions.enableForTab(" "'dummy', {tabId: 0, url: 'http://foo/'});", "pageActions.enableForTab", "[\"dummy\",{\"tabId\":0,\"url\":\"http://foo/\"}]"); // Try both optional parameters (title and iconId). ExpectJsPass("chrome.pageActions.enableForTab(" "'dummy', {tabId: 0, url: 'http://foo/'," "title: 'a', iconId: 0});", "pageActions.enableForTab", "[\"dummy\",{\"tabId\":0,\"url\":\"http://foo/\"," "\"title\":\"a\",\"iconId\":0}]"); // Now try disablePageAction. ExpectJsPass("chrome.pageActions.disableForTab(" "'dummy', {tabId: 0, url: 'http://foo/'});", "pageActions.disableForTab", "[\"dummy\",{\"tabId\":0,\"url\":\"http://foo/\"}]"); } TEST_F(ExtensionAPIClientTest, ExpandToolstrip) { ExpectJsPass("chrome.toolstrip.expand({height:100, url:'http://foo/'})", "toolstrip.expand", "{\"height\":100,\"url\":\"http://foo/\"}"); ExpectJsPass("chrome.toolstrip.expand({height:100}, null)", "toolstrip.expand", "{\"height\":100}"); ExpectJsPass("chrome.toolstrip.expand({height:100,url:'http://foo/'}, " "function(){})", "toolstrip.expand", "{\"height\":100,\"url\":\"http://foo/\"}"); ExpectJsFail("chrome.toolstrip.expand()", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.toolstrip.expand(1)", "Uncaught Error: Invalid value for argument 0. " "Expected 'object' but got 'integer'."); ExpectJsFail("chrome.toolstrip.expand({height:'100', url:'http://foo/'})", "Uncaught Error: Invalid value for argument 0. " "Property 'height': " "Expected 'integer' but got 'string'."); ExpectJsFail("chrome.toolstrip.expand({height:100,url:100})", "Uncaught Error: Invalid value for argument 0. Property 'url': " "Expected 'string' but got 'integer'."); ExpectJsFail("chrome.toolstrip.expand({height:100,'url':'http://foo/'}, 32)", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); } TEST_F(ExtensionAPIClientTest, CollapseToolstrip) { ExpectJsPass("chrome.toolstrip.collapse({url:'http://foo/'})", "toolstrip.collapse", "{\"url\":\"http://foo/\"}"); ExpectJsPass("chrome.toolstrip.collapse(null)", "toolstrip.collapse", "null"); ExpectJsPass("chrome.toolstrip.collapse({url:'http://foo/'}, function(){})", "toolstrip.collapse", "{\"url\":\"http://foo/\"}"); ExpectJsFail("chrome.toolstrip.collapse(1)", "Uncaught Error: Invalid value for argument 0. " "Expected 'object' but got 'integer'."); ExpectJsFail("chrome.toolstrip.collapse(100)", "Uncaught Error: Invalid value for argument 0. " "Expected 'object' but got 'integer'."); ExpectJsFail("chrome.toolstrip.collapse({url:'http://foo/'}, 32)", "Uncaught Error: Invalid value for argument 1. " "Expected 'function' but got 'integer'."); } // I18N API TEST_F(ExtensionAPIClientTest, GetAcceptLanguages) { ExpectJsFail("chrome.i18n.getAcceptLanguages(32, function(){})", "Uncaught Error: Too many arguments."); ExpectJsFail("chrome.i18n.getAcceptLanguages()", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.i18n.getAcceptLanguages('abc')", "Uncaught Error: Invalid value for argument 0. " "Expected 'function' but got 'string'."); ExpectJsPass("chrome.i18n.getAcceptLanguages(function(){})", "i18n.getAcceptLanguages", "null"); } TEST_F(ExtensionAPIClientTest, GetL10nMessage) { ExpectJsFail("chrome.i18n.getMessage()", "Uncaught Error: Parameter 0 is required."); ExpectJsFail("chrome.i18n.getMessage(1)", "Uncaught Error: Invalid value for argument 0. " "Expected 'string' but got 'integer'."); ExpectJsFail("chrome.i18n.getMessage('name', [])", "Uncaught Error: Invalid value for argument 1. Value does not " "match any valid type choices."); ExpectJsFail("chrome.i18n.getMessage('name', ['p1', 'p2', 'p3', 'p4', 'p5', " "'p6', 'p7', 'p8', 'p9', 'p10'])", "Uncaught Error: Invalid value for argument 1. Value does not " "match any valid type choices."); }