diff --git a/browser/components/search/test/browser.ini b/browser/components/search/test/browser.ini index 3c208184748e..f67da8b09c55 100644 --- a/browser/components/search/test/browser.ini +++ b/browser/components/search/test/browser.ini @@ -23,6 +23,8 @@ support-files = [browser_contextmenu.js] [browser_contextSearchTabPosition.js] skip-if = os == "mac" # bug 967013 +[browser_ddg.js] +[browser_ddg_behavior.js] [browser_google.js] [browser_google_codes.js] [browser_google_behavior.js] diff --git a/browser/components/search/test/browser_ddg.js b/browser/components/search/test/browser_ddg.js new file mode 100644 index 000000000000..206d310181a2 --- /dev/null +++ b/browser/components/search/test/browser_ddg.js @@ -0,0 +1,108 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +/* + * Test DuckDuckGo search plugin URLs + */ + +"use strict"; + +const BROWSER_SEARCH_PREF = "browser.search."; + +function test() { + let engine = Services.search.getEngineByName("DuckDuckGo"); + ok(engine, "DuckDuckGo"); + + let base = "https://duckduckgo.com/?q=foo"; + let url; + + // Test search URLs (including purposes). + url = engine.getSubmission("foo").uri.spec; + is(url, base + "&t=ffsb", "Check search URL for 'foo'"); + url = engine.getSubmission("foo", null, "contextmenu").uri.spec; + is(url, base + "&t=ffcm", "Check context menu search URL for 'foo'"); + url = engine.getSubmission("foo", null, "keyword").uri.spec; + is(url, base + "&t=ffab", "Check keyword search URL for 'foo'"); + url = engine.getSubmission("foo", null, "searchbar").uri.spec; + is(url, base + "&t=ffsb", "Check search bar search URL for 'foo'"); + url = engine.getSubmission("foo", null, "homepage").uri.spec; + is(url, base + "&t=ffhp", "Check homepage search URL for 'foo'"); + url = engine.getSubmission("foo", null, "newtab").uri.spec; + is(url, base + "&t=ffnt", "Check newtab search URL for 'foo'"); + + // Check search suggestion URL. + url = engine.getSubmission("foo", "application/x-suggestions+json").uri.spec; + is(url, "https://ac.duckduckgo.com/ac/?q=foo&type=list", "Check search suggestion URL for 'foo'"); + + // Check all other engine properties. + const EXPECTED_ENGINE = { + name: "DuckDuckGo", + alias: null, + description: "Search DuckDuckGo", + searchForm: "https://duckduckgo.com/?q=&t=ffsb", + hidden: false, + wrappedJSObject: { + queryCharset: "UTF-8", + "_iconURL": "", + _urls : [ + { + type: "text/html", + method: "GET", + template: "https://duckduckgo.com/", + params: [ + { + name: "q", + value: "{searchTerms}", + purpose: undefined, + }, + { + name: "t", + value: "ffcm", + purpose: "contextmenu", + }, + { + name: "t", + value: "ffab", + purpose:"keyword", + }, + { + name: "t", + value: "ffsb", + purpose: "searchbar", + }, + { + name: "t", + value: "ffhp", + purpose: "homepage", + }, + { + name: "t", + value: "ffnt", + purpose: "newtab", + }, + ], + mozparams: {}, + }, + { + type: "application/x-suggestions+json", + method: "GET", + template: "https://ac.duckduckgo.com/ac/", + params: [ + { + name: "q", + value: "{searchTerms}", + purpose: undefined, + }, + { + name: "type", + value: "list", + purpose: undefined, + }, + ], + }, + ], + }, + }; + + isSubObjectOf(EXPECTED_ENGINE, engine, "DuckDuckGo"); +} diff --git a/browser/components/search/test/browser_ddg_behavior.js b/browser/components/search/test/browser_ddg_behavior.js new file mode 100644 index 000000000000..c086dea47c0e --- /dev/null +++ b/browser/components/search/test/browser_ddg_behavior.js @@ -0,0 +1,165 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +/* + * Test DuckDuckGo search plugin URLs + */ + +"use strict"; + +const BROWSER_SEARCH_PREF = "browser.search."; + + +function test() { + let engine = Services.search.getEngineByName("DuckDuckGo"); + ok(engine, "DuckDuckGo is installed"); + + let previouslySelectedEngine = Services.search.currentEngine; + Services.search.currentEngine = engine; + engine.alias = "d"; + + let base = "https://duckduckgo.com/?q=foo"; + let url; + + // Test search URLs (including purposes). + url = engine.getSubmission("foo").uri.spec; + is(url, base + "&t=ffsb", "Check search URL for 'foo'"); + + waitForExplicitFinish(); + + var gCurrTest; + var gTests = [ + { + name: "context menu search", + searchURL: base + "&t=ffcm", + run() { + // Simulate a contextmenu search + // FIXME: This is a bit "low-level"... + BrowserSearch.loadSearch("foo", false, "contextmenu"); + } + }, + { + name: "keyword search", + searchURL: base + "&t=ffab", + run() { + gURLBar.value = "? foo"; + gURLBar.focus(); + EventUtils.synthesizeKey("VK_RETURN", {}); + } + }, + { + name: "keyword search with alias", + searchURL: base + "&t=ffab", + run() { + gURLBar.value = "d foo"; + gURLBar.focus(); + EventUtils.synthesizeKey("VK_RETURN", {}); + } + }, + { + name: "search bar search", + searchURL: base + "&t=ffsb", + run() { + let sb = BrowserSearch.searchBar; + sb.focus(); + sb.value = "foo"; + registerCleanupFunction(function() { + sb.value = ""; + }); + EventUtils.synthesizeKey("VK_RETURN", {}); + } + }, + { + name: "new tab search", + searchURL: base + "&t=ffnt", + run() { + function doSearch(doc) { + // Re-add the listener, and perform a search + gBrowser.addProgressListener(listener); + doc.getElementById("newtab-search-text").value = "foo"; + doc.getElementById("newtab-search-submit").click(); + } + + // load about:newtab, but remove the listener first so it doesn't + // get in the way + gBrowser.removeProgressListener(listener); + gBrowser.loadURI("about:newtab"); + info("Waiting for about:newtab load"); + tab.linkedBrowser.addEventListener("load", function load(loadEvent) { + if (loadEvent.originalTarget != tab.linkedBrowser.contentDocumentAsCPOW || + loadEvent.target.location.href == "about:blank") { + info("skipping spurious load event"); + return; + } + tab.linkedBrowser.removeEventListener("load", load, true); + + // Observe page setup + let win = gBrowser.contentWindowAsCPOW; + if (win.gSearch.currentEngineName == + Services.search.currentEngine.name) { + doSearch(win.document); + } else { + info("Waiting for newtab search init"); + win.addEventListener("ContentSearchService", function done(contentSearchServiceEvent) { + info("Got newtab search event " + contentSearchServiceEvent.detail.type); + if (contentSearchServiceEvent.detail.type == "State") { + win.removeEventListener("ContentSearchService", done); + // Let gSearch respond to the event before continuing. + executeSoon(() => doSearch(win.document)); + } + }); + } + }, true); + } + } + ]; + + function nextTest() { + if (gTests.length) { + gCurrTest = gTests.shift(); + info("Running : " + gCurrTest.name); + executeSoon(gCurrTest.run); + } else { + finish(); + } + } + + let tab = gBrowser.selectedTab = gBrowser.addTab(); + + let listener = { + onStateChange: function onStateChange(webProgress, req, flags, status) { + info("onStateChange"); + // Only care about top-level document starts + let docStart = Ci.nsIWebProgressListener.STATE_IS_DOCUMENT | + Ci.nsIWebProgressListener.STATE_START; + if (!(flags & docStart) || !webProgress.isTopLevel) + return; + + if (req.originalURI.spec == "about:blank") + return; + + info("received document start"); + + ok(req instanceof Ci.nsIChannel, "req is a channel"); + is(req.originalURI.spec, gCurrTest.searchURL, "search URL was loaded"); + info("Actual URI: " + req.URI.spec); + + req.cancel(Components.results.NS_ERROR_FAILURE); + + executeSoon(nextTest); + } + } + + registerCleanupFunction(function() { + engine.alias = undefined; + gBrowser.removeProgressListener(listener); + gBrowser.removeTab(tab); + Services.search.currentEngine = previouslySelectedEngine; + }); + + tab.linkedBrowser.addEventListener("load", function load() { + tab.linkedBrowser.removeEventListener("load", load, true); + gBrowser.addProgressListener(listener); + nextTest(); + }, true); +}