Index: chrome/browser/extensions/extension_override_apitest.cc |
diff --git a/chrome/browser/extensions/extension_override_apitest.cc b/chrome/browser/extensions/extension_override_apitest.cc |
index 5e180331c76c60c4655465b0ff20b152b63084f1..6c47fc93b035b97d04fdf5ffb0fa3df8b21c355b 100644 |
--- a/chrome/browser/extensions/extension_override_apitest.cc |
+++ b/chrome/browser/extensions/extension_override_apitest.cc |
@@ -19,10 +19,13 @@ |
#include "content/public/browser/navigation_entry.h" |
#include "content/public/browser/web_contents.h" |
#include "extensions/common/constants.h" |
+#include "extensions/test/extension_test_message_listener.h" |
#include "extensions/test/result_catcher.h" |
using content::WebContents; |
+namespace extensions { |
+ |
class ExtensionOverrideTest : public ExtensionApiTest { |
protected: |
bool CheckHistoryOverridesContainsNoDupes() { |
@@ -31,43 +34,142 @@ class ExtensionOverrideTest : public ExtensionApiTest { |
browser()->profile()->GetPrefs()->GetDictionary( |
ExtensionWebUI::kExtensionURLOverrides); |
- const base::ListValue* values = NULL; |
+ const base::ListValue* values = nullptr; |
if (!overrides->GetList("history", &values)) |
return false; |
std::set<std::string> seen_overrides; |
- for (size_t i = 0; i < values->GetSize(); ++i) { |
- std::string value; |
- if (!values->GetString(i, &value)) |
- return false; |
- |
- if (seen_overrides.find(value) != seen_overrides.end()) |
+ for (const base::Value* val : *values) { |
+ const base::DictionaryValue* dict = nullptr; |
+ std::string entry; |
+ if (!val->GetAsDictionary(&dict) || !dict->GetString("entry", &entry) || |
+ seen_overrides.count(entry) != 0) |
return false; |
- |
- seen_overrides.insert(value); |
+ seen_overrides.insert(entry); |
} |
return true; |
} |
+ |
+ // Returns AssertionSuccess() if the given |web_contents| is being actively |
+ // controlled by the extension with |extension_id|. |
+ testing::AssertionResult ExtensionControlsPage( |
+ content::WebContents* web_contents, |
+ const std::string& extension_id) { |
+ if (!web_contents->GetController().GetLastCommittedEntry()) |
+ return testing::AssertionFailure() << "No last committed entry."; |
+ // We can't just use WebContents::GetLastCommittedURL() here because |
+ // trickiness makes it think that it committed chrome://newtab. |
+ GURL gurl = web_contents->GetController().GetLastCommittedEntry()->GetURL(); |
+ if (!gurl.SchemeIs(kExtensionScheme)) |
+ return testing::AssertionFailure() << gurl; |
+ if (gurl.host() != extension_id) |
+ return testing::AssertionFailure() << gurl; |
+ return testing::AssertionSuccess(); |
+ } |
+ |
+ base::FilePath data_dir() { |
+ return test_data_dir_.AppendASCII("override"); |
+ } |
}; |
+// Basic test for overriding the NTP. |
IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, OverrideNewTab) { |
- ASSERT_TRUE(RunExtensionTest("override/newtab")) << message_; |
+ const Extension* extension = LoadExtension(data_dir().AppendASCII("newtab")); |
{ |
- extensions::ResultCatcher catcher; |
// Navigate to the new tab page. The overridden new tab page |
- // will call chrome.test.notifyPass() . |
+ // will call chrome.test.sendMessage('controlled by first'). |
+ ExtensionTestMessageListener listener(false); |
ui_test_utils::NavigateToURL(browser(), GURL("chrome://newtab/")); |
- WebContents* tab = browser()->tab_strip_model()->GetActiveWebContents(); |
- ASSERT_TRUE(tab->GetController().GetVisibleEntry()); |
- EXPECT_TRUE(tab->GetController().GetVisibleEntry()->GetURL(). |
- SchemeIs(extensions::kExtensionScheme)); |
+ EXPECT_TRUE(ExtensionControlsPage( |
+ browser()->tab_strip_model()->GetActiveWebContents(), |
+ extension->id())); |
+ EXPECT_TRUE(listener.WaitUntilSatisfied()); |
+ EXPECT_EQ("controlled by first", listener.message()); |
+ } |
+} |
- ASSERT_TRUE(catcher.GetNextResult()); |
+// Check having multiple extensions with the same override. |
+IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, OverrideNewTabMultiple) { |
+ // Prefer IDs because loading/unloading invalidates the extension ptrs. |
+ const std::string extension1_id = |
+ LoadExtension(data_dir().AppendASCII("newtab"))->id(); |
+ const std::string extension2_id = |
+ LoadExtension(data_dir().AppendASCII("newtab2"))->id(); |
+ { |
+ // Navigate to the new tab page. Last extension installed wins, so |
+ // the new tab page should be controlled by the second extension. |
+ ExtensionTestMessageListener listener(false); |
+ ui_test_utils::NavigateToURL(browser(), GURL("chrome://newtab/")); |
+ EXPECT_TRUE(ExtensionControlsPage( |
+ browser()->tab_strip_model()->GetActiveWebContents(), |
+ extension2_id)); |
+ EXPECT_TRUE(listener.WaitUntilSatisfied()); |
+ EXPECT_EQ("controlled by second", listener.message()); |
+ } |
+ |
+ // Unload and reload the first extension. This should *not* result in the |
+ // first extension moving to the front of the line. |
+ ReloadExtension(extension1_id); |
+ |
+ { |
+ // The page should still be controlled by the second extension. |
+ ExtensionTestMessageListener listener(false); |
+ ui_test_utils::NavigateToURL(browser(), GURL("chrome://newtab/")); |
+ EXPECT_TRUE(ExtensionControlsPage( |
+ browser()->tab_strip_model()->GetActiveWebContents(), |
+ extension2_id)); |
+ EXPECT_TRUE(listener.WaitUntilSatisfied()); |
+ EXPECT_EQ("controlled by second", listener.message()); |
} |
- // TODO(erikkay) Load a second extension with the same override. |
- // Verify behavior, then unload the first and verify behavior, etc. |
+ // Unload the (controlling) second extension. Now, and only now, should |
+ // extension1 take over. |
+ UnloadExtension(extension2_id); |
+ |
+ { |
+ ExtensionTestMessageListener listener(false); |
+ ui_test_utils::NavigateToURL(browser(), GURL("chrome://newtab/")); |
+ EXPECT_TRUE(ExtensionControlsPage( |
+ browser()->tab_strip_model()->GetActiveWebContents(), |
+ extension1_id)); |
+ EXPECT_TRUE(listener.WaitUntilSatisfied()); |
+ EXPECT_EQ("controlled by first", listener.message()); |
+ } |
+} |
+ |
+// Test that unloading an extension overriding the page reloads the page with |
+// the proper url. |
+IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, |
+ OverridingExtensionUnloadedWithPageOpen) { |
+ // Prefer IDs because loading/unloading invalidates the extension ptrs. |
+ const std::string extension1_id = |
+ LoadExtension(data_dir().AppendASCII("newtab"))->id(); |
+ const std::string extension2_id = |
+ LoadExtension(data_dir().AppendASCII("newtab2"))->id(); |
+ { |
+ // Navigate to the new tab page. Last extension installed wins, so |
+ // the new tab page should be controlled by the second extension. |
+ ExtensionTestMessageListener listener(false); |
+ ui_test_utils::NavigateToURL(browser(), GURL("chrome://newtab/")); |
+ EXPECT_TRUE(ExtensionControlsPage( |
+ browser()->tab_strip_model()->GetActiveWebContents(), |
+ extension2_id)); |
+ EXPECT_TRUE(listener.WaitUntilSatisfied()); |
+ EXPECT_EQ("controlled by second", listener.message()); |
+ } |
+ |
+ { |
+ // Unload the controlling extension. The page should be automatically |
+ // reloaded with the new controlling extension. |
+ ExtensionTestMessageListener listener(false); |
+ UnloadExtension(extension2_id); |
+ EXPECT_TRUE(listener.WaitUntilSatisfied()); |
+ EXPECT_EQ("controlled by first", listener.message()); |
+ EXPECT_TRUE(ExtensionControlsPage( |
+ browser()->tab_strip_model()->GetActiveWebContents(), |
+ extension1_id)); |
+ } |
} |
#if defined(OS_MACOSX) |
@@ -77,7 +179,7 @@ IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, OverrideNewTab) { |
#define MAYBE_OverrideNewTabIncognito OverrideNewTabIncognito |
#endif |
IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, MAYBE_OverrideNewTabIncognito) { |
- ASSERT_TRUE(RunExtensionTest("override/newtab")) << message_; |
+ LoadExtension(data_dir().AppendASCII("newtab")); |
// Navigate an incognito tab to the new tab page. We should get the actual |
// new tab page because we can't load chrome-extension URLs in incognito. |
@@ -86,7 +188,7 @@ IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, MAYBE_OverrideNewTabIncognito) { |
WebContents* tab = otr_browser->tab_strip_model()->GetActiveWebContents(); |
ASSERT_TRUE(tab->GetController().GetVisibleEntry()); |
EXPECT_FALSE(tab->GetController().GetVisibleEntry()->GetURL(). |
- SchemeIs(extensions::kExtensionScheme)); |
+ SchemeIs(kExtensionScheme)); |
} |
// Times out consistently on Win, http://crbug.com/45173. |
@@ -99,7 +201,7 @@ IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, MAYBE_OverrideNewTabIncognito) { |
IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, MAYBE_OverrideHistory) { |
ASSERT_TRUE(RunExtensionTest("override/history")) << message_; |
{ |
- extensions::ResultCatcher catcher; |
+ ResultCatcher catcher; |
// Navigate to the history page. The overridden history page |
// will call chrome.test.notifyPass() . |
ui_test_utils::NavigateToURL(browser(), GURL("chrome://history/")); |
@@ -109,29 +211,35 @@ IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, MAYBE_OverrideHistory) { |
// Regression test for http://crbug.com/41442. |
IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, ShouldNotCreateDuplicateEntries) { |
- const extensions::Extension* extension = |
+ const Extension* extension = |
LoadExtension(test_data_dir_.AppendASCII("override/history")); |
ASSERT_TRUE(extension); |
// Simulate several LoadExtension() calls happening over the lifetime of |
// a preferences file without corresponding UnloadExtension() calls. |
for (size_t i = 0; i < 3; ++i) { |
- ExtensionWebUI::RegisterChromeURLOverrides( |
+ ExtensionWebUI::RegisterOrActivateChromeURLOverrides( |
browser()->profile(), |
- extensions::URLOverrides::GetChromeURLOverrides(extension)); |
+ URLOverrides::GetChromeURLOverrides(extension)); |
} |
ASSERT_TRUE(CheckHistoryOverridesContainsNoDupes()); |
} |
+// TODO(devlin): This test seems a bit contrived. How would we end up with |
+// duplicate entries created? |
IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, ShouldCleanUpDuplicateEntries) { |
// Simulate several LoadExtension() calls happening over the lifetime of |
// a preferences file without corresponding UnloadExtension() calls. This is |
// the same as the above test, except for that it is testing the case where |
// the file already contains dupes when an extension is loaded. |
base::ListValue* list = new base::ListValue(); |
- for (size_t i = 0; i < 3; ++i) |
- list->Append(new base::StringValue("http://www.google.com/")); |
+ for (size_t i = 0; i < 3; ++i) { |
+ scoped_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); |
+ dict->SetString("entry", "http://www.google.com/"); |
+ dict->SetBoolean("active", true); |
+ list->Append(std::move(dict)); |
+ } |
{ |
DictionaryPrefUpdate update(browser()->profile()->GetPrefs(), |
@@ -141,7 +249,9 @@ IN_PROC_BROWSER_TEST_F(ExtensionOverrideTest, ShouldCleanUpDuplicateEntries) { |
ASSERT_FALSE(CheckHistoryOverridesContainsNoDupes()); |
- ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("override/history"))); |
+ ExtensionWebUI::InitializeChromeURLOverrides(profile()); |
ASSERT_TRUE(CheckHistoryOverridesContainsNoDupes()); |
} |
+ |
+} // namespace extensions |