Skip to content

Commit

Permalink
Add Fragment Support for Search Terms (#5847)
Browse files Browse the repository at this point in the history
* add fragment support for search engine

* refactor and re indent file

* add test case of fragment usage for search term
  • Loading branch information
Mustafa Hastürk authored and garvankeeley committed Dec 11, 2019
1 parent a81b35b commit ce33cb5
Show file tree
Hide file tree
Showing 3 changed files with 54 additions and 21 deletions.
11 changes: 11 additions & 0 deletions Client/Assets/Search/SearchPlugins/yaani.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
<!-- This Source Code Form is subject to the terms of the Mozilla Public
- License, v. 2.0. If a copy of the MPL was not distributed with this
- file, You can obtain one at http://mozilla.org/MPL/2.0/. -->

<SearchPlugin xmlns="http://www.mozilla.org/2006/browser/search/">
<ShortName>Yaani</ShortName>
<InputEncoding>UTF-8</InputEncoding>
<Image width="16" height="16"></Image>
<Url type="text/html" method="GET" template="https://tr.yaani.com.tr/?src=1#q={searchTerms}">
</Url>
</SearchPlugin>
48 changes: 32 additions & 16 deletions Client/Frontend/Browser/OpenSearch.swift
Original file line number Diff line number Diff line change
Expand Up @@ -39,8 +39,8 @@ class OpenSearchEngine: NSObject, NSCoding {
// http://stackoverflow.com/a/40034694
let isCustomEngine = aDecoder.decodeAsBool(forKey: "isCustomEngine")
guard let searchTemplate = aDecoder.decodeObject(forKey: "searchTemplate") as? String,
let shortName = aDecoder.decodeObject(forKey: "shortName") as? String,
let image = aDecoder.decodeObject(forKey: "image") as? UIImage else {
let shortName = aDecoder.decodeObject(forKey: "shortName") as? String,
let image = aDecoder.decodeObject(forKey: "image") as? UIImage else {
assertionFailure()
return nil
}
Expand Down Expand Up @@ -78,14 +78,25 @@ class OpenSearchEngine: NSObject, NSCoding {
// and have to do flaky pattern matching instead.
let placeholder = "PLACEHOLDER"
let template = searchTemplate.replacingOccurrences(of: SearchTermComponent, with: placeholder)
let components = URLComponents(string: template)
let searchTerm = components?.queryItems?.filter { item in
var components = URLComponents(string: template)

if let retVal = extractQueryArg(in: components?.queryItems, for: placeholder) {
return retVal
} else {
// Query arg may be exist inside fragment
components = URLComponents()
components?.query = URL(string: template)?.fragment
return extractQueryArg(in: components?.queryItems, for: placeholder)
}
}

fileprivate func extractQueryArg(in queryItems: [URLQueryItem]?, for placeholder: String) -> String? {
let searchTerm = queryItems?.filter { item in
return item.value == placeholder
}
guard let term = searchTerm, !term.isEmpty else { return nil }
return term[0].name
return searchTerm?.first?.name
}

/**
* check that the URL host contains the name of the search engine somewhere inside it
**/
Expand All @@ -100,13 +111,18 @@ class OpenSearchEngine: NSObject, NSCoding {
* Returns the query that was used to construct a given search URL
**/
func queryForSearchURL(_ url: URL?) -> String? {
if isSearchURLForEngine(url) {
if let key = searchQueryComponentKey,
let value = url?.getQuery()[key] {
return value.replacingOccurrences(of: "+", with: " ").removingPercentEncoding
}
guard isSearchURLForEngine(url), let key = searchQueryComponentKey else { return nil }

if let value = url?.getQuery()[key] {
return value.replacingOccurrences(of: "+", with: " ").removingPercentEncoding
} else {
// If search term could not found in query, it may be exist inside fragment
var components = URLComponents()
components.query = url?.fragment?.removingPercentEncoding

guard let value = components.url?.getQuery()[key] else { return nil }
return value.replacingOccurrences(of: "+", with: " ").removingPercentEncoding
}
return nil
}

/**
Expand Down Expand Up @@ -266,9 +282,9 @@ class OpenSearchParser {

let uiImage: UIImage
if let imageElement = largestImageElement,
let imageURL = URL(string: imageElement.stringValue),
let imageData = try? Data(contentsOf: imageURL),
let image = UIImage.imageFromDataThreadSafe(imageData) {
let imageURL = URL(string: imageElement.stringValue),
let imageData = try? Data(contentsOf: imageURL),
let image = UIImage.imageFromDataThreadSafe(imageData) {
uiImage = image
} else {
print("Error: Invalid search image data")
Expand Down
16 changes: 11 additions & 5 deletions ClientTests/SearchTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ class SearchTests: XCTestCase {
func testSuggestClient() {
let webServerBase = startMockSuggestServer()
let engine = OpenSearchEngine(engineID: "mock", shortName: "Mock engine", image: UIImage(), searchTemplate: "", suggestTemplate: "\(webServerBase)?q={searchTerms}",
isCustomEngine: false)
isCustomEngine: false)
let client = SearchSuggestClient(searchEngine: engine, userAgent: "Fx-testSuggestClient")

let query1 = self.expectation(description: "foo query")
Expand Down Expand Up @@ -100,15 +100,16 @@ class SearchTests: XCTestCase {

func testExtractingOfSearchTermsFromURL() {
let parser = OpenSearchParser(pluginMode: true)
var file = Bundle.main.path(forResource: "google-b-m", ofType: "xml", inDirectory: "SearchPlugins/")
let googleEngine: OpenSearchEngine! = parser.parse(file!, engineID: "google")
var file = Bundle.main.path(forResource: "google-b-m", ofType: "xml", inDirectory: "SearchPlugins/")!
let googleEngine: OpenSearchEngine! = parser.parse(file, engineID: "google")

// create URL
let searchTerm = "Foo Bar"
let encodedSeachTerm = searchTerm.replacingOccurrences(of: " ", with: "+")
let googleSearchURL = URL(string: "https://www.google.com/search?q=\(encodedSeachTerm)&ie=utf-8&oe=utf-8&gws_rd=cr&ei=I0UyVp_qK4HtUoytjagM")
let duckDuckGoSearchURL = URL(string: "https://duckduckgo.com/?q=\(encodedSeachTerm)&ia=about")
let invalidSearchURL = URL(string: "https://www.google.co.uk")
let yaaniSearchURL = URL(string: "https://tr.yaani.com.tr/?src=1#q=\(encodedSeachTerm)")

// check it correctly matches google search term given google config
XCTAssertEqual(searchTerm, googleEngine.queryForSearchURL(googleSearchURL))
Expand All @@ -117,15 +118,20 @@ class SearchTests: XCTestCase {
XCTAssertNil(googleEngine.queryForSearchURL(invalidSearchURL))

// check that it matches given a different configuration
file = Bundle.main.path(forResource: "duckduckgo", ofType: "xml", inDirectory: "SearchPlugins/")
let duckDuckGoEngine: OpenSearchEngine! = parser.parse(file!, engineID: "duckduckgo")
file = Bundle.main.path(forResource: "duckduckgo", ofType: "xml", inDirectory: "SearchPlugins/")!
let duckDuckGoEngine: OpenSearchEngine! = parser.parse(file, engineID: "duckduckgo")
XCTAssertEqual(searchTerm, duckDuckGoEngine.queryForSearchURL(duckDuckGoSearchURL))

// check it doesn't match search URLs for different configurations
XCTAssertNil(duckDuckGoEngine.queryForSearchURL(googleSearchURL))

// check that if you pass in a nil URL that everything works
XCTAssertNil(duckDuckGoEngine.queryForSearchURL(nil))

// check that if search engine that uses fragment matches search term
file = Bundle.main.path(forResource: "yaani", ofType: "xml", inDirectory: "SearchPlugins/")!
let yaaniEngine: OpenSearchEngine = parser.parse(file, engineID: "Yaani")!
XCTAssertEqual(searchTerm, yaaniEngine.queryForSearchURL(yaaniSearchURL))
}

fileprivate func startMockSuggestServer() -> String {
Expand Down

0 comments on commit ce33cb5

Please sign in to comment.