From 5df4eb4eeeb7b87eb847d8ef504d0d614777f924 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Buczek?= Date: Tue, 15 Jun 2021 11:50:37 +0200 Subject: [PATCH] Fix #3641, fix #3745 add Brave Search engine, add BraveSearch default SE callbacks. (#3535) --- Client.xcodeproj/project.pbxproj | 8 ++ Client/Application/ClientPreferences.swift | 3 + Client/Assets/SearchPlugins/bravesearch.xml | 19 +++ .../Frontend/Browser/BraveSearchHelper.swift | 122 ++++++++++++++++++ .../Browser/BrowserViewController.swift | 2 + .../BrowserViewController+OpenSearch.swift | 2 +- .../Frontend/Browser/DomainUserScript.swift | 24 +++- .../Browser/Search/InitialSearchEngines.swift | 13 +- .../Frontend/Browser/Search/OpenSearch.swift | 1 + .../Browser/Search/SearchEngines.swift | 2 +- .../Frontend/Browser/UserScriptManager.swift | 2 +- .../UserScripts/BraveSearchHelper.js | 50 +++++++ ClientTests/InitialSearchEnginesTests.swift | 9 ++ 13 files changed, 246 insertions(+), 11 deletions(-) create mode 100644 Client/Assets/SearchPlugins/bravesearch.xml create mode 100644 Client/Frontend/Browser/BraveSearchHelper.swift create mode 100644 Client/Frontend/UserContent/UserScripts/BraveSearchHelper.js diff --git a/Client.xcodeproj/project.pbxproj b/Client.xcodeproj/project.pbxproj index f3eda8a63fb..7b7446c5eef 100644 --- a/Client.xcodeproj/project.pbxproj +++ b/Client.xcodeproj/project.pbxproj @@ -162,6 +162,8 @@ 0ADCD45B231973650078CC67 /* UserAgentBuilderTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0ADCD45A231973650078CC67 /* UserAgentBuilderTests.swift */; }; 0ADCD45F2319799F0078CC67 /* RollingFileLoggerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = E61453BD1B750A1700C3F9D7 /* RollingFileLoggerTests.swift */; }; 0AE16F8D251BEF8C00A688ED /* InitialSearchEnginesTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0AE16F8C251BEF8C00A688ED /* InitialSearchEnginesTests.swift */; }; + 0AE50854261C63D70099C6A3 /* BraveSearchHelper.js in Resources */ = {isa = PBXBuildFile; fileRef = 0AE50853261C63D70099C6A3 /* BraveSearchHelper.js */; }; + 0AE5086A261C6F2E0099C6A3 /* BraveSearchHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0AE50869261C6F2E0099C6A3 /* BraveSearchHelper.swift */; }; 0AE5C09922CAA01E00DFF3EE /* RewardsButton.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0AE5C09822CAA01E00DFF3EE /* RewardsButton.swift */; }; 0AE5C69124F0059D004CBC9B /* OnboardingPrivacyConsentViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0AE5C69024F0059D004CBC9B /* OnboardingPrivacyConsentViewController.swift */; }; 0AE5C69424F005F9004CBC9B /* OnboardingPrivacyConsentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0AE5C69324F005F9004CBC9B /* OnboardingPrivacyConsentView.swift */; }; @@ -1532,6 +1534,8 @@ 0ADCD4512319640F0078CC67 /* UserAgentBuilder.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserAgentBuilder.swift; sourceTree = ""; }; 0ADCD45A231973650078CC67 /* UserAgentBuilderTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserAgentBuilderTests.swift; sourceTree = ""; }; 0AE16F8C251BEF8C00A688ED /* InitialSearchEnginesTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InitialSearchEnginesTests.swift; sourceTree = ""; }; + 0AE50853261C63D70099C6A3 /* BraveSearchHelper.js */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.javascript; path = BraveSearchHelper.js; sourceTree = ""; }; + 0AE50869261C6F2E0099C6A3 /* BraveSearchHelper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BraveSearchHelper.swift; sourceTree = ""; }; 0AE5C09822CAA01E00DFF3EE /* RewardsButton.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RewardsButton.swift; sourceTree = ""; }; 0AE5C69024F0059D004CBC9B /* OnboardingPrivacyConsentViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OnboardingPrivacyConsentViewController.swift; sourceTree = ""; }; 0AE5C69324F005F9004CBC9B /* OnboardingPrivacyConsentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OnboardingPrivacyConsentView.swift; sourceTree = ""; }; @@ -4769,6 +4773,7 @@ D0FCF7E81FE44D8F004A7995 /* AllFrames */, D0FCF7E91FE44DA2004A7995 /* MainFrame */, 595E0EDA21CAD97C00813D49 /* CookieControl.js */, + 0AE50853261C63D70099C6A3 /* BraveSearchHelper.js */, F930CDAB227000F200A23FE1 /* U2F.js */, F99505FE22937E3900CC6543 /* U2F-low-level.js */, 5E3477E822D7771700B0D5F8 /* ResourceDownloader.js */, @@ -4962,6 +4967,7 @@ 0A8A6224257905E300B035F4 /* UniversalLinkManager.swift */, 4452CAEF255412800053EFE6 /* DefaultBrowserIntroCalloutViewController.swift */, 0A0A5ED025B1F080007B3E74 /* DefaultBrowserIntroManager.swift */, + 0AE50869261C6F2E0099C6A3 /* BraveSearchHelper.swift */, ); indentWidth = 4; path = Browser; @@ -6306,6 +6312,7 @@ 0A5E04F923FEADA800E5A3E9 /* LaunchScreen.storyboard in Resources */, 0AB22A8A257EADA900126ADC /* corwin-prescott_olympic.jpg in Resources */, 0AB22A8F257EADA900126ADC /* su-san-lee.jpg in Resources */, + 0AE50854261C63D70099C6A3 /* BraveSearchHelper.js in Resources */, 5DB474F1237F4CC9007B7652 /* ntp-data.json in Resources */, 595E0EDB21CAD97C00813D49 /* CookieControl.js in Resources */, 0A0D3D3C21A4BE6400BEE65B /* adblock-list.txt in Resources */, @@ -7197,6 +7204,7 @@ 0A4BEF5D221AF1910005551A /* AdblockResourceDownloader.swift in Sources */, 0BA1E02E1B046F1E007675AF /* ErrorPageHelper.swift in Sources */, 5EC2C07325BF988E005EA984 /* PlaylistCell.swift in Sources */, + 0AE5086A261C6F2E0099C6A3 /* BraveSearchHelper.swift in Sources */, D3A9949C1A3686BD008AD1AC /* BrowserViewController.swift in Sources */, 59A681BDFC95A19F05E07223 /* SearchViewController.swift in Sources */, 0A8C6993225BC7B100988715 /* ToolbarUrlActionsProtocol.swift in Sources */, diff --git a/Client/Application/ClientPreferences.swift b/Client/Application/ClientPreferences.swift index a53bd179f03..4e20bddc720 100644 --- a/Client/Application/ClientPreferences.swift +++ b/Client/Application/ClientPreferences.swift @@ -147,6 +147,9 @@ extension Preferences { static let shouldShowRecentSearches = Option(key: "search.should-show-recent-searches", default: false) /// Whether or not to show recent searches opt-in static let shouldShowRecentSearchesOptIn = Option(key: "search.should-show-recent-searches.opt-in", default: true) + /// How many times Brave Search websites has asked the user to check whether Brave Search can be set as a default + static let braveSearchDefaultBrowserPromptCount = + Option(key: "search.brave-search-default-website-prompt", default: 0) } final class Privacy { /// Forces all private tabs diff --git a/Client/Assets/SearchPlugins/bravesearch.xml b/Client/Assets/SearchPlugins/bravesearch.xml new file mode 100644 index 00000000000..41115fc448a --- /dev/null +++ b/Client/Assets/SearchPlugins/bravesearch.xml @@ -0,0 +1,19 @@ + + + +Brave Search beta +UTF-8 + + + + + + + + + +https://search.brave.com + diff --git a/Client/Frontend/Browser/BraveSearchHelper.swift b/Client/Frontend/Browser/BraveSearchHelper.swift new file mode 100644 index 00000000000..e2fe092cdd3 --- /dev/null +++ b/Client/Frontend/Browser/BraveSearchHelper.swift @@ -0,0 +1,122 @@ +// Copyright 2021 The Brave Authors. All rights reserved. +// 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/. + +import Foundation +import WebKit +import Shared +import BraveShared + +private let log = Logger.browserLogger + +class BraveSearchHelper: TabContentScript { + private weak var tab: Tab? + private let profile: Profile + + /// Tracks how many in current browsing session the user has been prompted to set Brave Search as a default + /// while on one of Brave Search websites. + private static var canSetAsDefaultCounter = 0 + /// How many times user should be shown the default browser prompt on Brave Search websites. + private let maxCountOfDefaultBrowserPromptsPerSession = 3 + /// How many times user is shown the default browser prompt in total, this does not reset between app launches. + private let maxCountOfDefaultBrowserPromptsTotal = 10 + + required init(tab: Tab, profile: Profile) { + self.tab = tab + self.profile = profile + } + + static func name() -> String { "BraveSearchHelper" } + + func scriptMessageHandlerName() -> String? { BraveSearchHelper.name() } + + private enum Method: Int { + case canSetBraveSearchAsDefault = 1 + case setBraveSearchDefault = 2 + } + + private struct MethodModel: Codable { + enum CodingKeys: String, CodingKey { + case methodId = "method_id" + } + + let methodId: Int + } + + func userContentController(_ userContentController: WKUserContentController, + didReceiveScriptMessage message: WKScriptMessage) { + let allowedHosts = ["search.brave.com", + "search-dev.brave.com", + "search-dev-local.brave.com", + "search.brave.software", + "search.bravesoftware.com"] + + guard let requestHost = message.frameInfo.request.url?.host, + allowedHosts.contains(requestHost), + message.frameInfo.isMainFrame else { + log.error("Backup search request called from disallowed host") + return + } + + guard let data = try? JSONSerialization.data(withJSONObject: message.body, options: []), + let method = try? JSONDecoder().decode(MethodModel.self, from: data).methodId else { + log.error("Failed to retrieve method id") + return + } + + switch method { + case Method.canSetBraveSearchAsDefault.rawValue: + handleCanSetBraveSearchAsDefault(methodId: method) + case Method.setBraveSearchDefault.rawValue: + handleSetBraveSearchDefault(methodId: method) + default: + break + } + } + + private func handleCanSetBraveSearchAsDefault(methodId: Int) { + + if PrivateBrowsingManager.shared.isPrivateBrowsing { + log.debug("Private mode detected, skipping setting Brave Search as a default") + callback(methodId: methodId, result: false) + return + } + + let maximumPromptCount = Preferences.Search.braveSearchDefaultBrowserPromptCount + if Self.canSetAsDefaultCounter >= maxCountOfDefaultBrowserPromptsPerSession || + maximumPromptCount.value >= maxCountOfDefaultBrowserPromptsTotal { + log.debug("Maximum number of tries of Brave Search website prompts reached") + callback(methodId: methodId, result: false) + return + } + + Self.canSetAsDefaultCounter += 1 + maximumPromptCount.value += 1 + + let defaultEngine = profile.searchEngines.defaultEngine(forType: .standard).shortName + let canSetAsDefault = defaultEngine != OpenSearchEngine.EngineNames.brave + + callback(methodId: methodId, result: canSetAsDefault) + } + + private func handleSetBraveSearchDefault(methodId: Int) { + profile.searchEngines.updateDefaultEngine(OpenSearchEngine.EngineNames.brave, forType: .standard) + callback(methodId: methodId, result: nil) + } + + private func callback(methodId: Int, result: Bool?) { + let functionName = + "window.__firefox__.BSH\(UserScriptManager.messageHandlerTokenString).resolve" + + var args: [Any] = [methodId] + if let result = result { + args.append(result) + } + + self.tab?.webView?.evaluateSafeJavaScript( + functionName: functionName, + args: args, + sandboxed: false) + } +} diff --git a/Client/Frontend/Browser/BrowserViewController.swift b/Client/Frontend/Browser/BrowserViewController.swift index e089d04f9ba..24932d77fb9 100644 --- a/Client/Frontend/Browser/BrowserViewController.swift +++ b/Client/Frontend/Browser/BrowserViewController.swift @@ -2045,6 +2045,8 @@ extension BrowserViewController: TabDelegate { tab.addContentScript(FingerprintingProtection(tab: tab), name: FingerprintingProtection.name(), sandboxed: false) tab.addContentScript(BraveGetUA(tab: tab), name: BraveGetUA.name(), sandboxed: false) + tab.addContentScript(BraveSearchHelper(tab: tab, profile: profile), + name: BraveSearchHelper.name(), sandboxed: false) if YubiKitDeviceCapabilities.supportsMFIAccessoryKey { tab.addContentScript(U2FExtensions(tab: tab), name: U2FExtensions.name(), sandboxed: false) diff --git a/Client/Frontend/Browser/BrowserViewController/OpenSearch/BrowserViewController+OpenSearch.swift b/Client/Frontend/Browser/BrowserViewController/OpenSearch/BrowserViewController+OpenSearch.swift index 5c991535195..c4dda158dfd 100644 --- a/Client/Frontend/Browser/BrowserViewController/OpenSearch/BrowserViewController+OpenSearch.swift +++ b/Client/Frontend/Browser/BrowserViewController/OpenSearch/BrowserViewController+OpenSearch.swift @@ -49,7 +49,7 @@ extension BrowserViewController { // Add Reference Object as Open Search Engine openSearchEngine = referenceObject - // Open Search guidlines requires Title to be same as Short Name but it is not enforced, + // Open Search guidelines requires Title to be same as Short Name but it is not enforced, // thus in case of yahoo.com the title is 'Yahoo Search' and Shortname is 'Yahoo' // We are checking referenceURL match to determine searchEngine is added or not // In addition we are also checking if there is another engine with same name diff --git a/Client/Frontend/Browser/DomainUserScript.swift b/Client/Frontend/Browser/DomainUserScript.swift index 16530ba1674..273827c45ed 100644 --- a/Client/Frontend/Browser/DomainUserScript.swift +++ b/Client/Frontend/Browser/DomainUserScript.swift @@ -13,6 +13,7 @@ private let log = Logger.browserLogger enum DomainUserScript: CaseIterable { case youtube case archive + case braveSearch static func get(for domain: String) -> Self? { var found: DomainUserScript? @@ -33,7 +34,7 @@ enum DomainUserScript: CaseIterable { switch self { case .youtube: return .AdblockAndTp - case .archive: + case .archive, .braveSearch: return nil } } @@ -44,6 +45,8 @@ enum DomainUserScript: CaseIterable { return .init(arrayLiteral: "youtube.com") case .archive: return .init(arrayLiteral: "archive.is", "archive.today", "archive.vn", "archive.fo") + case .braveSearch: + return .init(arrayLiteral: "brave.com") } } @@ -53,14 +56,16 @@ enum DomainUserScript: CaseIterable { return "YoutubeAdblock" case .archive: return "ArchiveIsCompat" + case .braveSearch: + return "BraveSearchHelper" } } var script: WKUserScript? { + guard let source = sourceFile else { return nil } + switch self { case .youtube: - guard let source = sourceFile else { return nil } - // Verify that the application itself is making a call to the JS script instead of other scripts on the page. // This variable will be unique amongst scripts loaded in the page. // When the script is called, the token is provided in order to access the script variable. @@ -76,8 +81,19 @@ enum DomainUserScript: CaseIterable { return WKUserScript(source: alteredSource, injectionTime: .atDocumentStart, forMainFrameOnly: false) case .archive: - guard let source = sourceFile else { return nil } return WKUserScript(source: source, injectionTime: .atDocumentStart, forMainFrameOnly: false) + case .braveSearch: + var alteredSource = source + + let securityToken = UserScriptManager.securityToken.uuidString + .replacingOccurrences(of: "-", with: "", options: .literal) + alteredSource = alteredSource + .replacingOccurrences(of: "$", + with: "BSH\(UserScriptManager.messageHandlerTokenString)", + options: .literal) + .replacingOccurrences(of: "$", with: securityToken) + + return WKUserScript(source: alteredSource, injectionTime: .atDocumentStart, forMainFrameOnly: false) } } diff --git a/Client/Frontend/Browser/Search/InitialSearchEngines.swift b/Client/Frontend/Browser/Search/InitialSearchEngines.swift index e54a8c994d1..dc212143b6f 100644 --- a/Client/Frontend/Browser/Search/InitialSearchEngines.swift +++ b/Client/Frontend/Browser/Search/InitialSearchEngines.swift @@ -10,14 +10,17 @@ import Foundation class InitialSearchEngines { /// Type of search engine available to the user. enum SearchEngineID: String { - case google, bing, duckduckgo, yandex, qwant, startpage, yahoo, ecosia + case google, braveSearch, bing, duckduckgo, yandex, qwant, startpage, yahoo, ecosia - var excludedFromOnboarding: Bool { + func excludedFromOnboarding(for locale: Locale) -> Bool { switch self { case .google, .bing, .duckduckgo, .yandex, .qwant, .startpage, .ecosia: return false case .yahoo: return true + case .braveSearch: + guard let region = locale.regionCode else { return true } + return !InitialSearchEngines.braveSearchOnboardingRegions.contains(region) } } } @@ -50,9 +53,10 @@ class InitialSearchEngines { /// Lists of engines available during onboarding. var onboardingEngines: [SearchEngine] { - engines.filter { !$0.id.excludedFromOnboarding } + engines.filter { !$0.id.excludedFromOnboarding(for: locale) } } + static let braveSearchOnboardingRegions = ["US", "CA"] static let ddgDefaultRegions = ["DE", "AU", "NZ", "IE"] static let qwantDefaultRegions = ["FR"] static let yandexDefaultRegions = ["AM", "AZ", "BY", "KG", "KZ", "MD", "RU", "TJ", "TM", "TZ"] @@ -90,7 +94,8 @@ class InitialSearchEngines { self.locale = locale // Default order and available search engines, applies to all locales - engines = [.init(id: .google), + engines = [.init(id: .braveSearch), + .init(id: .google), .init(id: .bing), .init(id: .duckduckgo), .init(id: .qwant), diff --git a/Client/Frontend/Browser/Search/OpenSearch.swift b/Client/Frontend/Browser/Search/OpenSearch.swift index a8bd0829b90..3459b8e25e2 100644 --- a/Client/Frontend/Browser/Search/OpenSearch.swift +++ b/Client/Frontend/Browser/Search/OpenSearch.swift @@ -16,6 +16,7 @@ class OpenSearchEngine: NSObject, NSSecureCoding { struct EngineNames { static let duckDuckGo = "DuckDuckGo" static let qwant = "Qwant" + static let brave = "Brave Search beta" } static let defaultSearchClientName = "brave" diff --git a/Client/Frontend/Browser/Search/SearchEngines.swift b/Client/Frontend/Browser/Search/SearchEngines.swift index 03ae56b59b4..b6e7185d2f2 100644 --- a/Client/Frontend/Browser/Search/SearchEngines.swift +++ b/Client/Frontend/Browser/Search/SearchEngines.swift @@ -295,7 +295,7 @@ class SearchEngines { let se = InitialSearchEngines() let engines = isOnboarding ? se.onboardingEngines : se.engines - let engineNames = engines.map { $0.customId ?? $0.id.rawValue } + let engineNames = engines.map { ($0.customId ?? $0.id.rawValue).lowercased() } assert(!engineNames.isEmpty, "No search engines") return engineNames.map({ (name: $0, path: pluginDirectory.appendingPathComponent("\($0).xml").path) }) diff --git a/Client/Frontend/Browser/UserScriptManager.swift b/Client/Frontend/Browser/UserScriptManager.swift index 473c3cddb19..f7b1830d9a1 100644 --- a/Client/Frontend/Browser/UserScriptManager.swift +++ b/Client/Frontend/Browser/UserScriptManager.swift @@ -233,7 +233,7 @@ class UserScriptManager { alteredSource = alteredSource.replacingOccurrences(of: "$", with: "ResourceDownloadManager\(messageHandlerTokenString)", options: .literal) return WKUserScript(source: alteredSource, injectionTime: .atDocumentEnd, forMainFrameOnly: false) - }() + }() private let WindowRenderHelperScript: WKUserScript? = { guard let path = Bundle.main.path(forResource: "WindowRenderHelper", ofType: "js"), let source = try? String(contentsOfFile: path) else { diff --git a/Client/Frontend/UserContent/UserScripts/BraveSearchHelper.js b/Client/Frontend/UserContent/UserScripts/BraveSearchHelper.js new file mode 100644 index 00000000000..aa81d71e1c5 --- /dev/null +++ b/Client/Frontend/UserContent/UserScripts/BraveSearchHelper.js @@ -0,0 +1,50 @@ +// Copyright 2021 The Brave Authors. All rights reserved. +// 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/. + +'use strict'; + +Object.defineProperty(window.__firefox__, '$', { + enumerable: false, + configurable: true, + writable: false, + value: { + id: 1, + resolution_handlers: {}, + resolve(id, data, error) { + if (error && window.__firefox__.$.resolution_handlers[id].reject) { + window.__firefox__.$.resolution_handlers[id].reject(error); + } else if (window.__firefox__.$.resolution_handlers[id].resolve) { + window.__firefox__.$.resolution_handlers[id].resolve(data); + } else if (window.__firefox__.$.resolution_handlers[id].reject) { + window.__firefox__.$.resolution_handlers[id].reject(new Error("Invalid Data!")); + } else { + console.log("Invalid Promise ID: ", id); + } + + delete window.__firefox__.$.resolution_handlers[id]; + }, + sendMessage(method_id) { + return new Promise((resolve, reject) => { + window.__firefox__.$.resolution_handlers[method_id] = { resolve, reject }; + webkit.messageHandlers.BraveSearchHelper.postMessage({ 'securitytoken': '$' ,'method_id': method_id}); + }); + } + } +}); + +Object.defineProperty(window, 'brave', { +enumerable: false, +configurable: true, +writable: false, + value: { + getCanSetDefaultSearchProvider() { + return window.__firefox__.$.sendMessage(1); + }, + + setIsDefaultSearchProvider() { + return window.__firefox__.$.sendMessage(2); + } +} +}); diff --git a/ClientTests/InitialSearchEnginesTests.swift b/ClientTests/InitialSearchEnginesTests.swift index 640f0fa74a2..45e70523b0f 100644 --- a/ClientTests/InitialSearchEnginesTests.swift +++ b/ClientTests/InitialSearchEnginesTests.swift @@ -15,6 +15,7 @@ class InitialSearchEnginesTests: XCTestCase { let engines = unknownLocaleSE.engines.map { $0.id } XCTAssertEqual(engines, [.google, + .braveSearch, .bing, .duckduckgo, .qwant, @@ -66,6 +67,7 @@ class InitialSearchEnginesTests: XCTestCase { let availableEngines = localeSE.engines.map { $0.id } XCTAssertEqual(availableEngines, [.google, + .braveSearch, .bing, .duckduckgo, .qwant, @@ -75,6 +77,7 @@ class InitialSearchEnginesTests: XCTestCase { let onboardingEngines = localeSE.onboardingEngines.map { $0.id } XCTAssertEqual(onboardingEngines, [.google, + .braveSearch, .bing, .duckduckgo, .qwant, @@ -92,6 +95,7 @@ class InitialSearchEnginesTests: XCTestCase { let availableEngines = localeSE.engines.map { $0.id } XCTAssertEqual(availableEngines, [.google, + .braveSearch, .bing, .duckduckgo, .qwant, @@ -114,6 +118,7 @@ class InitialSearchEnginesTests: XCTestCase { let localeSE = SE(locale: Locale(identifier: "en_GB")) let availableEngines = localeSE.engines.map { $0.id } XCTAssertEqual(availableEngines, [.google, + .braveSearch, .bing, .duckduckgo, .qwant, @@ -138,6 +143,7 @@ class InitialSearchEnginesTests: XCTestCase { let availableEngines = localeSE.engines.map { $0.id } XCTAssertEqual(availableEngines, [.duckduckgo, + .braveSearch, .google, .bing, .qwant, @@ -162,6 +168,7 @@ class InitialSearchEnginesTests: XCTestCase { let availableEngines = localeSE.engines.map { $0.id } XCTAssertEqual(availableEngines, [.qwant, + .braveSearch, .google, .bing, .duckduckgo, @@ -186,6 +193,7 @@ class InitialSearchEnginesTests: XCTestCase { let availableEngines = unknownLocaleSE.engines.map { $0.id } XCTAssertEqual(availableEngines, [.google, + .braveSearch, .bing, .duckduckgo, .qwant, @@ -207,6 +215,7 @@ class InitialSearchEnginesTests: XCTestCase { let availableEngines = russianLocale.engines.map { $0.id } XCTAssertEqual(availableEngines, [.yandex, + .braveSearch, .google, .bing, .duckduckgo,