summaryrefslogtreecommitdiffstats
path: root/mobile/android/android-components/components/feature/search/src
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-15 03:35:49 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-15 03:35:49 +0000
commitd8bbc7858622b6d9c278469aab701ca0b609cddf (patch)
treeeff41dc61d9f714852212739e6b3738b82a2af87 /mobile/android/android-components/components/feature/search/src
parentReleasing progress-linux version 125.0.3-1~progress7.99u1. (diff)
downloadfirefox-d8bbc7858622b6d9c278469aab701ca0b609cddf.tar.xz
firefox-d8bbc7858622b6d9c278469aab701ca0b609cddf.zip
Merging upstream version 126.0.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'mobile/android/android-components/components/feature/search/src')
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/AndroidManifest.xml4
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/adsTelemetry.js82
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/manifest.template.json219
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/manifest.template.json220
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/searchTelemetry.js61
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/search/list.json923
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/search/search_telemetry_v2.json657
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-au.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-ca.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-co-uk.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-de.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-es.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-fr.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-in.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-it.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-jp.xml31
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-nl.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-se.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazondotcom.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/azerdict.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/baidu.xml25
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/bing.xml24
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ceneje.xml15
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/coccoc.xml21
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/daum-kr.xml21
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ddg.xml23
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-at.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-au.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-befr.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ca.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ch.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-co-uk.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-de.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-es.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-fr.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ie.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-it.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-nl.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-pl.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ecosia.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/faclair-beag.xml13
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-1-m.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-m.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-com-nocodes.xml14
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/gulesider-mobile-NO.xml15
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/leo_ende_de.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mapy-cz.xml14
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-ar.xml14
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-cl.xml14
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-mx.xml14
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/odpiralni.xml15
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/pazaruvaj.xml14
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/prisjakt-sv-SE.xml14
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/qwant.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/rakuten.xml16
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/reddit.xml11
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/salidzinilv.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/seznam-cz.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/vatera.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NN.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NO.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-an.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ar.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-as.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ast.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-az.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-be.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bg.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bn.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-br.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bs.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ca.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cy.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cz.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-da.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-de.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-dsb.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-el.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eo.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-es.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-et.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eu.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fa.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fi.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fr.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fy-NL.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ga-IE.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gd.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gl.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gn.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gu.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-he.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hi.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hr.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hsb.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hu.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hy-AM.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ia.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-id.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-is.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-it.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ja.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ka.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kab.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kk.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-km.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kn.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lij.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lo.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lt.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ltg.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lv.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ml.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-mr.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ms.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-my.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ne.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-nl.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-oc.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-or.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pa.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pl.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pt.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-rm.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ro.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ru.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sk.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sl.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sq.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sr.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sv-SE.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ta.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-te.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-th.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-tr.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uk.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ur.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uz.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-vi.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-wo.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-CN.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-TW.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-kn.xml18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-oc.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-or.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-ta.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-te.xml17
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp-auctions.xml16
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp.xml16
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-en.xml22
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-ru.xml21
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-tr.xml22
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.by.xml22
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.xml21
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/youtube.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/BrowserStoreSearchAdapter.kt41
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchAdapter.kt21
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchFeature.kt52
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchUseCases.kt343
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/BrowserStore.kt36
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/SearchEngine.kt148
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/internal/SearchUrlBuilder.kt93
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddleware.kt77
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/SearchMiddleware.kt397
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionManager.kt126
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionMiddleware.kt72
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorage.kt278
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/CustomSearchEnginesStorage.kt66
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineReader.kt243
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineWriter.kt115
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchMetadataStorage.kt103
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/Parser.kt79
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/SearchSuggestionClient.kt101
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetry.kt132
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ExtensionInfo.kt18
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderCookie.kt24
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderModel.kt80
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepository.kt170
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/TrackKeyInfo.kt38
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/Utils.kt116
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetry.kt124
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetry.kt86
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/AppSearchWidgetProvider.kt312
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivity.kt134
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/drawable/mozac_rounded_search_widget_background.xml9
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v1.xml20
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v2.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_large.xml45
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_medium.xml45
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small.xml27
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small_no_mic.xml19
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-am/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ar/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ast/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-azb/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-be/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-bg/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-br/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-bs/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ca/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-cak/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ckb/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-co/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-cs/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-cy/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-da/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-de/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-dsb/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-el/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-en-rCA/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-en-rGB/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-eo/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-es-rAR/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-es-rCL/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-es-rES/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-es-rMX/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-es/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-et/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-eu/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-fa/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ff/strings.xml8
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-fi/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-fr/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-fur/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-fy-rNL/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-gd/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-gl/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-gn/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-hr/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-hsb/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-hu/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-hy-rAM/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ia/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-in/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-is/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-it/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-iw/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ja/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ka/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-kaa/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-kab/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-kk/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-kmr/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ko/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-lo/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-lt/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-my/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-nb-rNO/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-night/colors.xml10
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-nl/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-nn-rNO/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-oc/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-pa-rIN/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-pa-rPK/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-pl/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-pt-rBR/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-pt-rPT/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-rm/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ru/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-sat/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-sc/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-si/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-sk/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-skr/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-sl/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-sq/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-sr/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-su/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-sv-rSE/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ta/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-te/strings.xml7
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-tg/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-th/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-tl/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-tr/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-trs/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-tt/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-ug/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-uk/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-uz/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-vi/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-yo/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-zh-rCN/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values-zh-rTW/strings.xml12
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values/colors.xml9
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values/dimens.xml7
-rw-r--r--mobile/android/android-components/components/feature/search/src/main/res/values/strings.xml16
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/BrowserStoreSeachAdapterTest.kt97
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchFeatureTest.kt134
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchUseCasesTest.kt663
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/BrowserStoreKtTest.kt117
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/SearchEngineKtTest.kt196
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddlewareTest.kt126
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/SearchMiddlewareTest.kt1921
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionManagerTest.kt146
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionMiddlewareTest.kt153
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorageTest.kt196
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/CustomSearchEngineStorageTest.kt105
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/ParseSearchPluginsTest.kt106
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineReaderTest.kt84
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineWriterTest.kt212
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/ParserTest.kt149
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/SearchSuggestionClientTest.kt110
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetryTest.kt187
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SearchProviderModelTest.kt40
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepositoryTest.kt92
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetryTest.kt271
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetryTest.kt467
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/AppSearchWidgetProviderTest.kt159
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityExtendedForTests.kt24
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityTest.kt134
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/resources/mockito-extensions/org.mockito.plugins.MockMaker2
-rw-r--r--mobile/android/android-components/components/feature/search/src/test/resources/robolectric.properties1
315 files changed, 15647 insertions, 0 deletions
diff --git a/mobile/android/android-components/components/feature/search/src/main/AndroidManifest.xml b/mobile/android/android-components/components/feature/search/src/main/AndroidManifest.xml
new file mode 100644
index 0000000000..e16cda1d34
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/AndroidManifest.xml
@@ -0,0 +1,4 @@
+<!-- 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/. -->
+<manifest />
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/adsTelemetry.js b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/adsTelemetry.js
new file mode 100644
index 0000000000..65bf306835
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/adsTelemetry.js
@@ -0,0 +1,82 @@
+/* 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/. */
+
+/**
+ * Send
+ * - current URL
+ * - cookies of this page
+ * - all links found in this page
+ * to the native application.
+ */
+function sendCurrentState() {
+ let message = {
+ 'url': document.location.href,
+ 'urls': getLinks(),
+ 'cookies': getCookies()
+ };
+ browser.runtime.sendNativeMessage("MozacBrowserAdsMessage", message);
+}
+
+/**
+ * Get all links in the current page.
+ *
+ * @return {Array<string>} containing all current links in the current page.
+ */
+function getLinks() {
+ let urls = [];
+
+ let anchors = document.getElementsByTagName("a");
+ for (let anchor of anchors) {
+ if (!anchor.href) {
+ continue;
+ }
+ urls.push(anchor.href);
+ }
+
+ return urls;
+}
+
+/**
+ * Get all cookies for the current document.
+ *
+ * @return {Array<{name: string, value: string}>} containing all cookies.
+ */
+function getCookies() {
+ let cookiesList = document.cookie.split("; ");
+ let result = [];
+
+ cookiesList.forEach(cookie => {
+ var [name, ...value] = cookie.split('=');
+ // For that special cases where the value contains '='.
+ value = value.join("=")
+
+ result.push({
+ "name" : name,
+ "value" : value
+ });
+ });
+
+ return result;
+}
+
+// Whenever a page is first accessed or when loaded from cache
+// send all needed data about the ads provider to the app.
+const events = ["pageshow", "load"];
+const eventLogger = event => {
+ switch (event.type) {
+ case "load":
+ sendCurrentState();
+ break;
+ case "pageshow":
+ if (event.persisted) {
+ sendCurrentState();
+ }
+ break;
+ default:
+ console.log('Event:', event.type);
+ }
+};
+events.forEach(eventName =>
+ window.addEventListener(eventName, eventLogger)
+);
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/manifest.template.json b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/manifest.template.json
new file mode 100644
index 0000000000..219e41b554
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/ads/manifest.template.json
@@ -0,0 +1,219 @@
+{
+ "manifest_version": 2,
+ "browser_specific_settings": {
+ "gecko": {
+ "id": "ads@mozac.org"
+ }
+ },
+ "name": "Mozilla Android Components - Ads Telemetry",
+ "version": "${version}",
+ "content_scripts": [
+ {
+ "matches": ["https://*/*"],
+ "include_globs": [
+ "https://www.google.com/search*",
+ "https://www.google.ad/search*",
+ "https://www.google.ae/search*",
+ "https://www.google.com.af/search*",
+ "https://www.google.com.ag/search*",
+ "https://www.google.com.ai/search*",
+ "https://www.google.al/search*",
+ "https://www.google.am/search*",
+ "https://www.google.co.ao/search*",
+ "https://www.google.com.ar/search*",
+ "https://www.google.as/search*",
+ "https://www.google.at/search*",
+ "https://www.google.com.au/search*",
+ "https://www.google.az/search*",
+ "https://www.google.ba/search*",
+ "https://www.google.com.bd/search*",
+ "https://www.google.be/search*",
+ "https://www.google.bf/search*",
+ "https://www.google.bg/search*",
+ "https://www.google.com.bh/search*",
+ "https://www.google.bi/search*",
+ "https://www.google.bj/search*",
+ "https://www.google.com.bn/search*",
+ "https://www.google.com.bo/search*",
+ "https://www.google.com.br/search*",
+ "https://www.google.bs/search*",
+ "https://www.google.bt/search*",
+ "https://www.google.co.bw/search*",
+ "https://www.google.by/search*",
+ "https://www.google.com.bz/search*",
+ "https://www.google.ca/search*",
+ "https://www.google.cd/search*",
+ "https://www.google.cf/search*",
+ "https://www.google.cg/search*",
+ "https://www.google.ch/search*",
+ "https://www.google.ci/search*",
+ "https://www.google.co.ck/search*",
+ "https://www.google.cl/search*",
+ "https://www.google.cm/search*",
+ "https://www.google.cn/search*",
+ "https://www.google.com.co/search*",
+ "https://www.google.co.cr/search*",
+ "https://www.google.com.cu/search*",
+ "https://www.google.cv/search*",
+ "https://www.google.com.cy/search*",
+ "https://www.google.cz/search*",
+ "https://www.google.de/search*",
+ "https://www.google.dj/search*",
+ "https://www.google.dk/search*",
+ "https://www.google.dm/search*",
+ "https://www.google.com.do/search*",
+ "https://www.google.dz/search*",
+ "https://www.google.com.ec/search*",
+ "https://www.google.ee/search*",
+ "https://www.google.com.eg/search*",
+ "https://www.google.es/search*",
+ "https://www.google.com.et/search*",
+ "https://www.google.fi/search*",
+ "https://www.google.com.fj/search*",
+ "https://www.google.fm/search*",
+ "https://www.google.fr/search*",
+ "https://www.google.ga/search*",
+ "https://www.google.ge/search*",
+ "https://www.google.gg/search*",
+ "https://www.google.com.gh/search*",
+ "https://www.google.com.gi/search*",
+ "https://www.google.gl/search*",
+ "https://www.google.gm/search*",
+ "https://www.google.gr/search*",
+ "https://www.google.com.gt/search*",
+ "https://www.google.gy/search*",
+ "https://www.google.com.hk/search*",
+ "https://www.google.hn/search*",
+ "https://www.google.hr/search*",
+ "https://www.google.ht/search*",
+ "https://www.google.hu/search*",
+ "https://www.google.co.id/search*",
+ "https://www.google.ie/search*",
+ "https://www.google.co.il/search*",
+ "https://www.google.im/search*",
+ "https://www.google.co.in/search*",
+ "https://www.google.iq/search*",
+ "https://www.google.is/search*",
+ "https://www.google.it/search*",
+ "https://www.google.je/search*",
+ "https://www.google.com.jm/search*",
+ "https://www.google.jo/search*",
+ "https://www.google.co.jp/search*",
+ "https://www.google.co.ke/search*",
+ "https://www.google.com.kh/search*",
+ "https://www.google.ki/search*",
+ "https://www.google.kg/search*",
+ "https://www.google.co.kr/search*",
+ "https://www.google.com.kw/search*",
+ "https://www.google.kz/search*",
+ "https://www.google.la/search*",
+ "https://www.google.com.lb/search*",
+ "https://www.google.li/search*",
+ "https://www.google.lk/search*",
+ "https://www.google.co.ls/search*",
+ "https://www.google.lt/search*",
+ "https://www.google.lu/search*",
+ "https://www.google.lv/search*",
+ "https://www.google.com.ly/search*",
+ "https://www.google.co.ma/search*",
+ "https://www.google.md/search*",
+ "https://www.google.me/search*",
+ "https://www.google.mg/search*",
+ "https://www.google.mk/search*",
+ "https://www.google.ml/search*",
+ "https://www.google.com.mm/search*",
+ "https://www.google.mn/search*",
+ "https://www.google.ms/search*",
+ "https://www.google.com.mt/search*",
+ "https://www.google.mu/search*",
+ "https://www.google.mv/search*",
+ "https://www.google.mw/search*",
+ "https://www.google.com.mx/search*",
+ "https://www.google.com.my/search*",
+ "https://www.google.co.mz/search*",
+ "https://www.google.com.na/search*",
+ "https://www.google.com.ng/search*",
+ "https://www.google.com.ni/search*",
+ "https://www.google.ne/search*",
+ "https://www.google.nl/search*",
+ "https://www.google.no/search*",
+ "https://www.google.com.np/search*",
+ "https://www.google.nr/search*",
+ "https://www.google.nu/search*",
+ "https://www.google.co.nz/search*",
+ "https://www.google.com.om/search*",
+ "https://www.google.com.pa/search*",
+ "https://www.google.com.pe/search*",
+ "https://www.google.com.pg/search*",
+ "https://www.google.com.ph/search*",
+ "https://www.google.com.pk/search*",
+ "https://www.google.pl/search*",
+ "https://www.google.pn/search*",
+ "https://www.google.com.pr/search*",
+ "https://www.google.ps/search*",
+ "https://www.google.pt/search*",
+ "https://www.google.com.py/search*",
+ "https://www.google.com.qa/search*",
+ "https://www.google.ro/search*",
+ "https://www.google.ru/search*",
+ "https://www.google.rw/search*",
+ "https://www.google.com.sa/search*",
+ "https://www.google.com.sb/search*",
+ "https://www.google.sc/search*",
+ "https://www.google.se/search*",
+ "https://www.google.com.sg/search*",
+ "https://www.google.sh/search*",
+ "https://www.google.si/search*",
+ "https://www.google.sk/search*",
+ "https://www.google.com.sl/search*",
+ "https://www.google.sn/search*",
+ "https://www.google.so/search*",
+ "https://www.google.sm/search*",
+ "https://www.google.sr/search*",
+ "https://www.google.st/search*",
+ "https://www.google.com.sv/search*",
+ "https://www.google.td/search*",
+ "https://www.google.tg/search*",
+ "https://www.google.co.th/search*",
+ "https://www.google.com.tj/search*",
+ "https://www.google.tl/search*",
+ "https://www.google.tm/search*",
+ "https://www.google.tn/search*",
+ "https://www.google.to/search*",
+ "https://www.google.com.tr/search*",
+ "https://www.google.tt/search*",
+ "https://www.google.com.tw/search*",
+ "https://www.google.co.tz/search*",
+ "https://www.google.com.ua/search*",
+ "https://www.google.co.ug/search*",
+ "https://www.google.co.uk/search*",
+ "https://www.google.com.uy/search*",
+ "https://www.google.co.uz/search*",
+ "https://www.google.com.vc/search*",
+ "https://www.google.co.ve/search*",
+ "https://www.google.vg/search*",
+ "https://www.google.co.vi/search*",
+ "https://www.google.com.vn/search*",
+ "https://www.google.vu/search*",
+ "https://www.google.ws/search*",
+ "https://www.google.rs/search*",
+ "https://www.google.co.za/search*",
+ "https://www.google.co.zm/search*",
+ "https://www.google.co.zw/search*",
+ "https://www.google.cat/search*",
+ "https://www.bing.com/search*",
+ "https://www.baidu.com/*",
+ "https://m.baidu.com/*",
+ "https://duckduckgo.com/*",
+ "https://www.ecosia.org/*"
+ ],
+ "js": ["adsTelemetry.js"],
+ "run_at": "document_end"
+ }
+ ],
+ "permissions": [
+ "geckoViewAddons",
+ "nativeMessaging",
+ "nativeMessagingFromContent"
+ ]
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/manifest.template.json b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/manifest.template.json
new file mode 100644
index 0000000000..ae9cf13dac
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/manifest.template.json
@@ -0,0 +1,220 @@
+{
+ "manifest_version": 2,
+ "browser_specific_settings": {
+ "gecko": {
+ "id": "cookies@mozac.org"
+ }
+ },
+ "name": "Mozilla Android Components - Search Telemetry",
+ "version": "${version}",
+ "content_scripts": [
+ {
+ "matches": ["https://*/*"],
+ "include_globs": [
+ "https://www.google.com/search*",
+ "https://www.google.ad/search*",
+ "https://www.google.ae/search*",
+ "https://www.google.com.af/search*",
+ "https://www.google.com.ag/search*",
+ "https://www.google.com.ai/search*",
+ "https://www.google.al/search*",
+ "https://www.google.am/search*",
+ "https://www.google.co.ao/search*",
+ "https://www.google.com.ar/search*",
+ "https://www.google.as/search*",
+ "https://www.google.at/search*",
+ "https://www.google.com.au/search*",
+ "https://www.google.az/search*",
+ "https://www.google.ba/search*",
+ "https://www.google.com.bd/search*",
+ "https://www.google.be/search*",
+ "https://www.google.bf/search*",
+ "https://www.google.bg/search*",
+ "https://www.google.com.bh/search*",
+ "https://www.google.bi/search*",
+ "https://www.google.bj/search*",
+ "https://www.google.com.bn/search*",
+ "https://www.google.com.bo/search*",
+ "https://www.google.com.br/search*",
+ "https://www.google.bs/search*",
+ "https://www.google.bt/search*",
+ "https://www.google.co.bw/search*",
+ "https://www.google.by/search*",
+ "https://www.google.com.bz/search*",
+ "https://www.google.ca/search*",
+ "https://www.google.cd/search*",
+ "https://www.google.cf/search*",
+ "https://www.google.cg/search*",
+ "https://www.google.ch/search*",
+ "https://www.google.ci/search*",
+ "https://www.google.co.ck/search*",
+ "https://www.google.cl/search*",
+ "https://www.google.cm/search*",
+ "https://www.google.cn/search*",
+ "https://www.google.com.co/search*",
+ "https://www.google.co.cr/search*",
+ "https://www.google.com.cu/search*",
+ "https://www.google.cv/search*",
+ "https://www.google.com.cy/search*",
+ "https://www.google.cz/search*",
+ "https://www.google.de/search*",
+ "https://www.google.dj/search*",
+ "https://www.google.dk/search*",
+ "https://www.google.dm/search*",
+ "https://www.google.com.do/search*",
+ "https://www.google.dz/search*",
+ "https://www.google.com.ec/search*",
+ "https://www.google.ee/search*",
+ "https://www.google.com.eg/search*",
+ "https://www.google.es/search*",
+ "https://www.google.com.et/search*",
+ "https://www.google.fi/search*",
+ "https://www.google.com.fj/search*",
+ "https://www.google.fm/search*",
+ "https://www.google.fr/search*",
+ "https://www.google.ga/search*",
+ "https://www.google.ge/search*",
+ "https://www.google.gg/search*",
+ "https://www.google.com.gh/search*",
+ "https://www.google.com.gi/search*",
+ "https://www.google.gl/search*",
+ "https://www.google.gm/search*",
+ "https://www.google.gr/search*",
+ "https://www.google.com.gt/search*",
+ "https://www.google.gy/search*",
+ "https://www.google.com.hk/search*",
+ "https://www.google.hn/search*",
+ "https://www.google.hr/search*",
+ "https://www.google.ht/search*",
+ "https://www.google.hu/search*",
+ "https://www.google.co.id/search*",
+ "https://www.google.ie/search*",
+ "https://www.google.co.il/search*",
+ "https://www.google.im/search*",
+ "https://www.google.co.in/search*",
+ "https://www.google.iq/search*",
+ "https://www.google.is/search*",
+ "https://www.google.it/search*",
+ "https://www.google.je/search*",
+ "https://www.google.com.jm/search*",
+ "https://www.google.jo/search*",
+ "https://www.google.co.jp/search*",
+ "https://www.google.co.ke/search*",
+ "https://www.google.com.kh/search*",
+ "https://www.google.ki/search*",
+ "https://www.google.kg/search*",
+ "https://www.google.co.kr/search*",
+ "https://www.google.com.kw/search*",
+ "https://www.google.kz/search*",
+ "https://www.google.la/search*",
+ "https://www.google.com.lb/search*",
+ "https://www.google.li/search*",
+ "https://www.google.lk/search*",
+ "https://www.google.co.ls/search*",
+ "https://www.google.lt/search*",
+ "https://www.google.lu/search*",
+ "https://www.google.lv/search*",
+ "https://www.google.com.ly/search*",
+ "https://www.google.co.ma/search*",
+ "https://www.google.md/search*",
+ "https://www.google.me/search*",
+ "https://www.google.mg/search*",
+ "https://www.google.mk/search*",
+ "https://www.google.ml/search*",
+ "https://www.google.com.mm/search*",
+ "https://www.google.mn/search*",
+ "https://www.google.ms/search*",
+ "https://www.google.com.mt/search*",
+ "https://www.google.mu/search*",
+ "https://www.google.mv/search*",
+ "https://www.google.mw/search*",
+ "https://www.google.com.mx/search*",
+ "https://www.google.com.my/search*",
+ "https://www.google.co.mz/search*",
+ "https://www.google.com.na/search*",
+ "https://www.google.com.ng/search*",
+ "https://www.google.com.ni/search*",
+ "https://www.google.ne/search*",
+ "https://www.google.nl/search*",
+ "https://www.google.no/search*",
+ "https://www.google.com.np/search*",
+ "https://www.google.nr/search*",
+ "https://www.google.nu/search*",
+ "https://www.google.co.nz/search*",
+ "https://www.google.com.om/search*",
+ "https://www.google.com.pa/search*",
+ "https://www.google.com.pe/search*",
+ "https://www.google.com.pg/search*",
+ "https://www.google.com.ph/search*",
+ "https://www.google.com.pk/search*",
+ "https://www.google.pl/search*",
+ "https://www.google.pn/search*",
+ "https://www.google.com.pr/search*",
+ "https://www.google.ps/search*",
+ "https://www.google.pt/search*",
+ "https://www.google.com.py/search*",
+ "https://www.google.com.qa/search*",
+ "https://www.google.ro/search*",
+ "https://www.google.ru/search*",
+ "https://www.google.rw/search*",
+ "https://www.google.com.sa/search*",
+ "https://www.google.com.sb/search*",
+ "https://www.google.sc/search*",
+ "https://www.google.se/search*",
+ "https://www.google.com.sg/search*",
+ "https://www.google.sh/search*",
+ "https://www.google.si/search*",
+ "https://www.google.sk/search*",
+ "https://www.google.com.sl/search*",
+ "https://www.google.sn/search*",
+ "https://www.google.so/search*",
+ "https://www.google.sm/search*",
+ "https://www.google.sr/search*",
+ "https://www.google.st/search*",
+ "https://www.google.com.sv/search*",
+ "https://www.google.td/search*",
+ "https://www.google.tg/search*",
+ "https://www.google.co.th/search*",
+ "https://www.google.com.tj/search*",
+ "https://www.google.tl/search*",
+ "https://www.google.tm/search*",
+ "https://www.google.tn/search*",
+ "https://www.google.to/search*",
+ "https://www.google.com.tr/search*",
+ "https://www.google.tt/search*",
+ "https://www.google.com.tw/search*",
+ "https://www.google.co.tz/search*",
+ "https://www.google.com.ua/search*",
+ "https://www.google.co.ug/search*",
+ "https://www.google.co.uk/search*",
+ "https://www.google.com.uy/search*",
+ "https://www.google.co.uz/search*",
+ "https://www.google.com.vc/search*",
+ "https://www.google.co.ve/search*",
+ "https://www.google.vg/search*",
+ "https://www.google.co.vi/search*",
+ "https://www.google.com.vn/search*",
+ "https://www.google.vu/search*",
+ "https://www.google.ws/search*",
+ "https://www.google.rs/search*",
+ "https://www.google.co.za/search*",
+ "https://www.google.co.zm/search*",
+ "https://www.google.co.zw/search*",
+ "https://www.google.cat/search*",
+ "https://www.baidu.com/*",
+ "https://m.baidu.com/*",
+ "https://*search.yahoo.com/search*",
+ "https://www.bing.com/search*",
+ "https://duckduckgo.com/*",
+ "https://www.ecosia.org/*"
+ ],
+ "js": ["searchTelemetry.js"],
+ "run_at": "document_end"
+ }
+ ],
+ "permissions": [
+ "geckoViewAddons",
+ "nativeMessaging",
+ "nativeMessagingFromContent"
+ ]
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/searchTelemetry.js b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/searchTelemetry.js
new file mode 100644
index 0000000000..3199335fdf
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/extensions/search/searchTelemetry.js
@@ -0,0 +1,61 @@
+/* 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/. */
+
+/**
+ * Send
+ * - current URL
+ * - cookies of this page
+ * to the native application.
+ */
+function sendCurrentState() {
+ let message = {
+ 'url': document.location.href,
+ 'cookies': getCookies()
+ };
+ browser.runtime.sendNativeMessage("MozacBrowserSearchMessage", message);
+}
+
+/**
+ * Get all cookies for the current document.
+ *
+ * @return {Array<{name: string, value: string}>} containing all cookies.
+ */
+function getCookies() {
+ let cookiesList = document.cookie.split("; ");
+ let result = [];
+
+ cookiesList.forEach(cookie => {
+ var [name, ...value] = cookie.split('=');
+ // For that special cases where the cookie value contains '='.
+ value = value.join("=");
+
+ result.push({
+ "name" : name,
+ "value" : value
+ });
+ });
+
+ return result;
+}
+
+// Whenever a page is first accessed or when loaded from cache
+// send all needed data about the search provider to the app.
+const events = ["pageshow", "load"];
+const eventLogger = event => {
+ switch (event.type) {
+ case "load":
+ sendCurrentState();
+ break;
+ case "pageshow":
+ if (event.persisted) {
+ sendCurrentState();
+ }
+ break;
+ default:
+ console.log('Event:', event.type);
+ }
+};
+events.forEach(eventName =>
+ window.addEventListener(eventName, eventLogger)
+);
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/search/list.json b/mobile/android/android-components/components/feature/search/src/main/assets/search/list.json
new file mode 100644
index 0000000000..b2bb3d4698
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/search/list.json
@@ -0,0 +1,923 @@
+{
+ "default": {
+ "searchDefault": "Google",
+ "searchOrder": ["Google", "Bing"],
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia"
+ ]
+ },
+ "regionOverrides": {
+ "US": {
+ "google-b-m": "google-b-1-m"
+ }
+ },
+ "locales": {
+ "ach": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia"
+ ]
+ }
+ },
+ "an": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-es","wikipedia-an"
+ ]
+ }
+ },
+ "ar": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-ar"
+ ]
+ }
+ },
+ "as": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-as"
+ ]
+ }
+ },
+ "ast": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "ebay-es", "wikipedia-ast"
+ ]
+ }
+ },
+ "az": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "azerdict", "wikipedia-az"
+ ]
+ }
+ },
+ "be": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-be"
+ ]
+ },
+ "BY": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-be"
+ ]
+ },
+ "KZ": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-be"
+ ]
+ },
+ "RU": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-be"
+ ]
+ },
+ "TR": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-be"
+ ]
+ }
+ },
+ "bg": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "pazaruvaj", "wikipedia-bg"
+ ]
+ }
+ },
+ "bn": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-bn"
+ ]
+ }
+ },
+ "bn-BD": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-bn"
+ ]
+ }
+ },
+ "bn-IN": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-bn"
+ ]
+ }
+ },
+ "br": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-br"
+ ]
+ }
+ },
+ "bs": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-bs"
+ ]
+ }
+ },
+ "ca": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-es", "wikipedia-ca"
+ ]
+ }
+ },
+ "cak": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-es"
+ ]
+ }
+ },
+ "cs": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "mapy-cz", "seznam-cz", "wikipedia-cz"
+ ]
+ }
+ },
+ "cy": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-co-uk", "ddg", "ebay-co-uk", "wikipedia-cy"
+ ]
+ }
+ },
+ "da": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "amazon-co-uk", "bing", "ddg", "wikipedia-da"
+ ]
+ }
+ },
+ "de": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-de", "ddg", "ecosia", "qwant", "wikipedia-de", "ebay-de"
+ ]
+ }
+ },
+ "de-AT": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-de", "ddg", "ecosia", "qwant", "wikipedia-de", "ebay-at"
+ ]
+ }
+ },
+ "dsb": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-de", "ddg", "wikipedia-dsb", "ebay-de"
+ ]
+ }
+ },
+ "el": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-el"
+ ]
+ }
+ },
+ "en-AU": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-au", "ddg", "wikipedia", "ebay-au"
+ ]
+ }
+ },
+ "en-CA": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-ca", "ddg", "wikipedia", "ebay-ca"
+ ]
+ }
+ },
+ "en-IE": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-co-uk", "ddg", "qwant", "wikipedia", "ebay-ie"
+ ]
+ }
+ },
+ "en-GB": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-co-uk", "ddg", "qwant", "wikipedia", "ebay-co-uk"
+ ]
+ },
+ "BY": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazon-co-uk", "ddg", "qwant", "wikipedia"
+ ]
+ },
+ "KZ": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazon-co-uk", "ddg", "qwant", "wikipedia"
+ ]
+ },
+ "RU": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazon-co-uk", "ddg", "qwant", "wikipedia"
+ ]
+ },
+ "TR": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazon-co-uk", "ddg", "qwant", "wikipedia"
+ ]
+ }
+ },
+ "en-US": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "ebay", "wikipedia"
+ ]
+ },
+ "BY": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazondotcom", "ddg", "wikipedia"
+ ]
+ },
+ "KZ": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazondotcom", "ddg", "wikipedia"
+ ]
+ },
+ "RU": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazondotcom", "ddg", "wikipedia"
+ ]
+ },
+ "TR": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "amazondotcom", "ddg", "wikipedia"
+ ]
+ }
+ },
+ "en-ZA": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "wikipedia"
+ ]
+ }
+ },
+ "eo": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-eo"
+ ]
+ }
+ },
+ "es-AR": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "mercadolibre-ar", "wikipedia-es"
+ ]
+ }
+ },
+ "es-CL": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "mercadolibre-cl", "wikipedia-es"
+ ]
+ }
+ },
+ "es-ES": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-es", "amazon-es", "ebay-es"
+ ]
+ }
+ },
+ "es-MX": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "mercadolibre-mx", "wikipedia-es"
+ ]
+ }
+ },
+ "et": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "amazon-co-uk", "ddg", "wikipedia-et"
+ ]
+ }
+ },
+ "eu": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-es", "wikipedia-eu"
+ ]
+ }
+ },
+ "fa": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-fa"
+ ]
+ }
+ },
+ "ff": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-fr", "ddg", "wikipedia-fr"
+ ]
+ }
+ },
+ "fi": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "amazondotcom", "bing", "ddg", "wikipedia-fi"
+ ]
+ }
+ },
+ "fr-BE": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "qwant", "wikipedia-fr", "ebay-befr"
+ ]
+ }
+ },
+ "fr-CA": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-ca", "ddg", "wikipedia-fr", "ebay-ca"
+ ]
+ }
+ },
+ "fr-FR": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "qwant", "wikipedia-fr", "amazon-fr", "ebay-fr"
+ ]
+ }
+ },
+ "fr": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-fr", "qwant", "wikipedia-fr"
+ ]
+ }
+ },
+ "fy-NL": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-nl", "wikipedia-fy-NL"
+ ]
+ }
+ },
+ "ga-IE": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "amazon-co-uk", "ddg", "ebay-ie", "wikipedia-ga-IE"
+ ]
+ }
+ },
+ "gd": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-co-uk", "faclair-beag", "wikipedia-gd"
+ ]
+ }
+ },
+ "gl": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "ebay-es", "wikipedia-gl"
+ ]
+ }
+ },
+ "gn": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-gn"
+ ]
+ }
+ },
+ "gu-IN": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-gu"
+ ]
+ }
+ },
+ "he": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-he"
+ ]
+ }
+ },
+ "hi-IN": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-hi"
+ ]
+ }
+ },
+ "hr": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-co-uk", "ddg", "wikipedia-hr"
+ ]
+ }
+ },
+ "hsb": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-de", "ddg", "wikipedia-hsb", "ebay-de"
+ ]
+ }
+ },
+ "hu": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "vatera", "wikipedia-hu"
+ ]
+ }
+ },
+ "hy-AM": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-hy-AM"
+ ]
+ }
+ },
+ "ia": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-ia"
+ ]
+ }
+ },
+ "id": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-id"
+ ]
+ }
+ },
+ "is": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-is"
+ ]
+ }
+ },
+ "it": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-it", "amazon-it", "ebay-it"
+ ]
+ }
+ },
+ "ja": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "amazon-jp", "bing", "ddg", "rakuten", "wikipedia-ja", "yahoo-jp", "yahoo-jp-auctions"
+ ]
+ }
+ },
+ "ka": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-ka"
+ ]
+ }
+ },
+ "kab": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-kab"
+ ]
+ }
+ },
+ "kk": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-kk"
+ ]
+ },
+ "KZ": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-kk"
+ ]
+ },
+ "BY": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-kk"
+ ]
+ },
+ "RU": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-kk"
+ ]
+ },
+ "TR": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "bing", "ddg", "wikipedia-kk"
+ ]
+ }
+ },
+ "km": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-km"
+ ]
+ }
+ },
+ "kn": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-kn", "wiktionary-kn"
+ ]
+ }
+ },
+ "ko": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "daum-kr"
+ ]
+ }
+ },
+ "lij": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-it", "ddg", "wikipedia-lij", "ebay-it"
+ ]
+ }
+ },
+ "lo": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-lo"
+ ]
+ }
+ },
+ "lt": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-lt"
+ ]
+ }
+ },
+ "ltg": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-co-uk", "wikipedia-ltg"
+ ]
+ }
+ },
+ "lv": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "salidzinilv", "wikipedia-lv"
+ ]
+ }
+ },
+ "mai": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-hi"
+ ]
+ }
+ },
+ "meh": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-es"
+ ]
+ }
+ },
+ "mix": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-es"
+ ]
+ }
+ },
+ "ml": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-ml"
+ ]
+ }
+ },
+ "mr": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-mr"
+ ]
+ }
+ },
+ "ms": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-ms"
+ ]
+ }
+ },
+ "my": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-my"
+ ]
+ }
+ },
+ "nb-NO": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "gulesider-mobile-NO", "wikipedia-NO"
+ ]
+ }
+ },
+ "ne-NP": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-ne"
+ ]
+ }
+ },
+ "nl-NL": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-nl", "amazon-nl", "ebay-nl"
+ ]
+ }
+ },
+ "nl": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-nl", "wikipedia-nl"
+ ]
+ }
+ },
+ "nn-NO": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "gulesider-mobile-NO", "wikipedia-NN"
+ ]
+ }
+ },
+ "oc": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-oc", "wiktionary-oc"
+ ]
+ }
+ },
+ "or": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-or", "wiktionary-or"
+ ]
+ }
+ },
+ "pa-IN": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-pa"
+ ]
+ }
+ },
+ "pl": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-pl", "ebay-pl"
+ ]
+ }
+ },
+ "pt-BR": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-pt"
+ ]
+ }
+ },
+ "pt-PT": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "wikipedia-pt"
+ ]
+ }
+ },
+ "rm": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-ch", "leo_ende_de", "wikipedia-rm"
+ ]
+ }
+ },
+ "ro": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-ro"
+ ]
+ }
+ },
+ "ru": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "wikipedia-ru"
+ ]
+ },
+ "RU": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "wikipedia-ru"
+ ]
+ },
+ "BY": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "wikipedia-ru"
+ ]
+ },
+ "KZ": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "wikipedia-ru"
+ ]
+ },
+ "TR": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "wikipedia-ru"
+ ]
+ }
+ },
+ "sk": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "wikipedia-sk"
+ ]
+ }
+ },
+ "sl": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "ceneje", "odpiralni", "wikipedia-sl"
+ ]
+ }
+ },
+ "son": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "ddg", "bing", "amazon-fr", "wikipedia-fr"
+ ]
+ }
+ },
+ "sq": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-co-uk", "ddg", "wikipedia-sq"
+ ]
+ }
+ },
+ "sr": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-sr"
+ ]
+ }
+ },
+ "sv-SE": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "prisjakt-sv-SE", "ddg", "wikipedia-sv-SE", "amazon-se", "ebay-ch"
+ ]
+ }
+ },
+ "ta": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-ta", "wiktionary-ta"
+ ]
+ }
+ },
+ "te": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-te", "wiktionary-te"
+ ]
+ }
+ },
+ "th": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-th"
+ ]
+ }
+ },
+ "tl": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg"
+ ]
+ }
+ },
+ "tr": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-tr"
+ ]
+ },
+ "TR": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "bing", "wikipedia-tr"
+ ]
+ },
+ "BY": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "bing", "wikipedia-tr"
+ ]
+ },
+ "KZ": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "bing", "wikipedia-tr"
+ ]
+ },
+ "RU": {
+ "visibleDefaultEngines": [
+ "google-com-nocodes", "ddg", "bing", "wikipedia-tr"
+ ]
+ }
+ },
+ "trs": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-es"
+ ]
+ }
+ },
+ "uk": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-uk"
+ ]
+ }
+ },
+ "ur": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazon-in", "ddg", "wikipedia-ur"
+ ]
+ }
+ },
+ "uz": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-uz"
+ ]
+ }
+ },
+ "vi": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "coccoc", "ddg", "wikipedia-vi"
+ ]
+ }
+ },
+ "wo": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "ebay-fr", "wikipedia-wo"
+ ]
+ }
+ },
+ "xh": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia"
+ ]
+ }
+ },
+ "zam": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-es"
+ ]
+ }
+ },
+ "zh-CN": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "baidu", "bing", "ddg", "wikipedia-zh-CN"
+ ]
+ },
+ "CN": {
+ "searchDefault": "百度"
+ }
+ },
+ "zh-TW": {
+ "default": {
+ "visibleDefaultEngines": [
+ "google-b-m", "bing", "ddg", "wikipedia-zh-TW"
+ ]
+ }
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/search/search_telemetry_v2.json b/mobile/android/android-components/components/feature/search/src/main/assets/search/search_telemetry_v2.json
new file mode 100644
index 0000000000..1803977187
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/search/search_telemetry_v2.json
@@ -0,0 +1,657 @@
+{
+ "data": [
+ {
+ "isSPA": true,
+ "schema": 1707523204491,
+ "components": [
+ {
+ "type": "ad_image_row",
+ "included": {
+ "parent": {
+ "selector": "[data-testid='pam.container']"
+ },
+ "children": [
+ {
+ "selector": "[data-slide-index]",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "included": {
+ "parent": {
+ "selector": "[data-testid='adResult']"
+ }
+ }
+ },
+ {
+ "type": "incontent_searchbox",
+ "topDown": true,
+ "included": {
+ "parent": {
+ "selector": "._1zdrb._1cR1n"
+ },
+ "related": {
+ "selector": "#search-suggestions"
+ },
+ "children": [
+ {
+ "selector": "input[type='search']"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "default": true
+ }
+ ],
+ "taggedCodes": [
+ "brz-moz",
+ "firefoxqwant"
+ ],
+ "telemetryId": "qwant",
+ "organicCodes": [],
+ "codeParamName": "client",
+ "queryParamName": "q",
+ "queryParamNames": [
+ "q"
+ ],
+ "searchPageRegexp": "^https://www\\.qwant\\.com/",
+ "filter_expression": "env.version|versionCompare(\"124.0a1\")>=0",
+ "followOnParamNames": [],
+ "defaultPageQueryParam": {
+ "key": "t",
+ "value": "web"
+ },
+ "extraAdServersRegexps": [
+ "^https://www\\.bing\\.com/acli?c?k",
+ "^https://api\\.qwant\\.com/v3/r/"
+ ],
+ "id": "19c434a3-d173-4871-9743-290ac92a3f6b",
+ "last_modified": 1707833261849
+ },
+ {
+ "schema": 1705948294201,
+ "components": [
+ {
+ "type": "ad_carousel",
+ "included": {
+ "parent": {
+ "selector": ".pla-exp-container"
+ },
+ "related": {
+ "selector": "g-right-button, g-left-button, .exp-button"
+ },
+ "children": [
+ {
+ "selector": "[data-dtld]",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "refined_search_buttons",
+ "topDown": true,
+ "included": {
+ "parent": {
+ "selector": "#appbar g-scrolling-carousel"
+ },
+ "related": {
+ "selector": "g-right-button, g-left-button"
+ },
+ "children": [
+ {
+ "selector": "a"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "excluded": {
+ "parent": {
+ "selector": "#rhs"
+ }
+ },
+ "included": {
+ "parent": {
+ "selector": "[data-text-ad='1']"
+ },
+ "children": [
+ {
+ "type": "ad_sitelink",
+ "selector": "[role='list']"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_sidebar",
+ "included": {
+ "parent": {
+ "selector": "#rhs"
+ },
+ "children": [
+ {
+ "selector": ".pla-unit, .mnr-c",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "incontent_searchbox",
+ "topDown": true,
+ "included": {
+ "parent": {
+ "selector": "form[role='search']"
+ },
+ "related": {
+ "selector": "div.logo + div + div"
+ },
+ "children": [
+ {
+ "selector": "input[type='text']"
+ },
+ {
+ "selector": "textarea[name='q']"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_image_row",
+ "excluded": {
+ "parent": {
+ "selector": ".pla-exp-container"
+ }
+ },
+ "included": {
+ "parent": {
+ "selector": ".top-pla-group-inner"
+ },
+ "children": [
+ {
+ "selector": "[data-dtld]",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "default": true
+ }
+ ],
+ "shoppingTab": {
+ "regexp": "&tbm=shop",
+ "selector": "div[role='navigation'] a",
+ "inspectRegexpInSERP": true
+ },
+ "taggedCodes": [
+ "firefox-a",
+ "firefox-b",
+ "firefox-b-1",
+ "firefox-b-ab",
+ "firefox-b-1-ab",
+ "firefox-b-d",
+ "firefox-b-1-d",
+ "firefox-b-e",
+ "firefox-b-1-e",
+ "firefox-b-m",
+ "firefox-b-1-m",
+ "firefox-b-o",
+ "firefox-b-1-o",
+ "firefox-b-lm",
+ "firefox-b-1-lm",
+ "firefox-b-lg",
+ "firefox-b-huawei-h1611",
+ "firefox-b-is-oem1",
+ "firefox-b-oem1",
+ "firefox-b-oem2",
+ "firefox-b-tinno",
+ "firefox-b-pn-wt",
+ "firefox-b-pn-wt-us",
+ "ubuntu",
+ "ubuntu-sn"
+ ],
+ "telemetryId": "google",
+ "organicCodes": [],
+ "codeParamName": "client",
+ "queryParamName": "q",
+ "queryParamNames": [
+ "q"
+ ],
+ "domainExtraction": {
+ "ads": [
+ {
+ "method": "data-attribute",
+ "options": {
+ "dataAttributeKey": "dtld"
+ },
+ "selectors": "[data-dtld]"
+ }
+ ],
+ "nonAds": [
+ {
+ "method": "href",
+ "selectors": "#rso div.g[jscontroller] > div > div > div > div a[data-usg]"
+ }
+ ]
+ },
+ "searchPageRegexp": "^https://www\\.google\\.(?:.+)/search",
+ "nonAdsLinkRegexps": [
+ "^https?://www\\.google\\.(?:.+)/url?(?:.+)&url="
+ ],
+ "adServerAttributes": [
+ "rw"
+ ],
+ "followOnParamNames": [
+ "oq",
+ "ved",
+ "ei"
+ ],
+ "extraAdServersRegexps": [
+ "^https?://www\\.google(?:adservices)?\\.com/(?:pagead/)?aclk"
+ ],
+ "id": "635a3325-1995-42d6-be09-dbe4b2a95453",
+ "last_modified": 1706198445460
+ },
+ {
+ "schema": 1705363206938,
+ "components": [
+ {
+ "type": "ad_carousel",
+ "included": {
+ "parent": {
+ "selector": ".module--carousel"
+ },
+ "related": {
+ "selector": ".module--carousel__left, .module--carousel__right"
+ },
+ "children": [
+ {
+ "selector": ".module--carousel__item",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "excluded": {
+ "parent": {
+ "selector": ".js-results-sidebar"
+ }
+ },
+ "included": {
+ "parent": {
+ "selector": "article[data-testid='ad']"
+ },
+ "children": [
+ {
+ "type": "ad_sitelink",
+ "selector": "ul"
+ }
+ ]
+ }
+ },
+ {
+ "type": "incontent_searchbox",
+ "topDown": true,
+ "included": {
+ "parent": {
+ "selector": "form#search_form"
+ },
+ "related": {
+ "selector": "input#search_button, .search__autocomplete"
+ },
+ "children": [
+ {
+ "selector": " input#search_form_input"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_sidebar",
+ "included": {
+ "parent": {
+ "selector": ".js-results-sidebar"
+ },
+ "children": [
+ {
+ "selector": "article[data-testid='ad']",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "default": true
+ }
+ ],
+ "shoppingTab": {
+ "regexp": "&iax=shopping&ia=shopping",
+ "selector": "#duckbar a[data-zci-link='products']"
+ },
+ "taggedCodes": [
+ "ffab",
+ "ffcm",
+ "ffhp",
+ "ffip",
+ "ffit",
+ "ffnt",
+ "ffocus",
+ "ffos",
+ "ffsb",
+ "fpas",
+ "fpsa",
+ "ftas",
+ "ftsa",
+ "lm",
+ "newext"
+ ],
+ "telemetryId": "duckduckgo",
+ "organicCodes": [],
+ "codeParamName": "t",
+ "queryParamName": "q",
+ "queryParamNames": [
+ "q"
+ ],
+ "domainExtraction": {
+ "ads": [
+ {
+ "method": "href",
+ "options": {
+ "queryParamKey": "ad_domain"
+ },
+ "selectors": ".products-carousel a.js-carousel-item-title, [data-testid='ad'] a[data-testid='result-title-a']"
+ }
+ ],
+ "nonAds": [
+ {
+ "method": "href",
+ "selectors": "[data-layout='organic'] a[data-testid='result-title-a']"
+ }
+ ]
+ },
+ "searchPageRegexp": "^https://duckduckgo\\.com/",
+ "expectedOrganicCodes": [
+ "h_",
+ "ha",
+ "hb",
+ "hc",
+ "hd",
+ "he",
+ "hf",
+ "hg",
+ "hh",
+ "hi",
+ "hj",
+ "hk",
+ "hl",
+ "hm",
+ "hn",
+ "ho",
+ "hp",
+ "hq",
+ "hr",
+ "hs",
+ "ht",
+ "hu",
+ "hv",
+ "hw",
+ "hx",
+ "hy",
+ "hz"
+ ],
+ "extraAdServersRegexps": [
+ "^https://duckduckgo.com/y\\.js?.*ad_provider\\=",
+ "^https://www\\.amazon\\.(?:[a-z.]{2,24}).*(?:tag=duckduckgo-)"
+ ],
+ "id": "9dfd626b-26f2-4913-9d0a-27db6cb7d8ca",
+ "last_modified": 1706198445456
+ },
+ {
+ "schema": 1698656464939,
+ "taggedCodes": [
+ "monline_dg",
+ "monline_3_dg",
+ "monline_4_dg",
+ "monline_7_dg"
+ ],
+ "telemetryId": "baidu",
+ "organicCodes": [],
+ "codeParamName": "tn",
+ "queryParamName": "wd",
+ "queryParamNames": [
+ "wd",
+ "word"
+ ],
+ "searchPageRegexp": "^https://(?:m|www)\\.baidu\\.com/(?:s|baidu)",
+ "followOnParamNames": [
+ "oq"
+ ],
+ "extraAdServersRegexps": [
+ "^https?://www\\.baidu\\.com/baidu\\.php?"
+ ],
+ "id": "19c434a3-d173-4871-9743-290ac92a3f6a",
+ "last_modified": 1698666532326
+ },
+ {
+ "schema": 1698656463945,
+ "components": [
+ {
+ "type": "ad_carousel",
+ "included": {
+ "parent": {
+ "selector": ".product-ads-carousel"
+ },
+ "related": {
+ "selector": ".snippet__control"
+ },
+ "children": [
+ {
+ "selector": ".product-ads-carousel__item",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "included": {
+ "parent": {
+ "selector": ".ad-result"
+ },
+ "children": [
+ {
+ "type": "ad_sitelink",
+ "selector": ".result__extra-content .deep-links--descriptions"
+ }
+ ]
+ }
+ },
+ {
+ "type": "incontent_searchbox",
+ "topDown": true,
+ "included": {
+ "parent": {
+ "selector": "form.search-form"
+ },
+ "related": {
+ "selector": ".search-form__suggestions"
+ },
+ "children": [
+ {
+ "selector": ".search-form__input, .search-form__submit"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "default": true
+ }
+ ],
+ "shoppingTab": {
+ "regexp": "/shopping?",
+ "selector": "nav li[data-test-id='search-navigation-item-shopping'] a"
+ },
+ "taggedCodes": [
+ "mzl",
+ "813cf1dd",
+ "16eeffc4"
+ ],
+ "telemetryId": "ecosia",
+ "organicCodes": [],
+ "codeParamName": "tt",
+ "queryParamName": "q",
+ "queryParamNames": [
+ "q"
+ ],
+ "searchPageRegexp": "^https://www\\.ecosia\\.org/",
+ "filter_expression": "env.version|versionCompare(\"110.0a1\")>=0",
+ "expectedOrganicCodes": [],
+ "extraAdServersRegexps": [
+ "^https://www\\.bing\\.com/acli?c?k"
+ ],
+ "id": "9a487171-3a06-4647-8866-36250ec84f3a",
+ "last_modified": 1698666532324
+ },
+ {
+ "schema": 1698656462833,
+ "components": [
+ {
+ "type": "ad_carousel",
+ "included": {
+ "parent": {
+ "selector": ".adsMvCarousel"
+ },
+ "related": {
+ "selector": ".cr"
+ },
+ "children": [
+ {
+ "selector": ".pa_item",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "excluded": {
+ "parent": {
+ "selector": "aside"
+ }
+ },
+ "included": {
+ "parent": {
+ "selector": ".sb_adTA"
+ },
+ "children": [
+ {
+ "type": "ad_sitelink",
+ "selector": ".b_vlist2col"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_sidebar",
+ "included": {
+ "parent": {
+ "selector": "aside"
+ },
+ "children": [
+ {
+ "selector": ".pa_item, .sb_adTA",
+ "countChildren": true
+ }
+ ]
+ }
+ },
+ {
+ "type": "incontent_searchbox",
+ "topDown": true,
+ "included": {
+ "parent": {
+ "selector": "form#sb_form"
+ },
+ "related": {
+ "selector": "#sw_as"
+ },
+ "children": [
+ {
+ "selector": "input[name='q']"
+ }
+ ]
+ }
+ },
+ {
+ "type": "ad_link",
+ "default": true
+ }
+ ],
+ "shoppingTab": {
+ "regexp": "^/shop?",
+ "selector": "#b-scopeListItem-shop a"
+ },
+ "taggedCodes": [
+ "MOZ2",
+ "MOZ4",
+ "MOZ5",
+ "MOZA",
+ "MOZB",
+ "MOZD",
+ "MOZE",
+ "MOZI",
+ "MOZL",
+ "MOZM",
+ "MOZO",
+ "MOZR",
+ "MOZT",
+ "MOZW",
+ "MOZX",
+ "MZSL01",
+ "MZSL02",
+ "MZSL03"
+ ],
+ "telemetryId": "bing",
+ "organicCodes": [],
+ "codeParamName": "pc",
+ "queryParamName": "q",
+ "followOnCookies": [
+ {
+ "host": "www.bing.com",
+ "name": "SRCHS",
+ "codeParamName": "PC",
+ "extraCodePrefixes": [
+ "QBRE"
+ ],
+ "extraCodeParamName": "form"
+ }
+ ],
+ "queryParamNames": [
+ "q"
+ ],
+ "searchPageRegexp": "^https://www\\.bing\\.com/search",
+ "nonAdsLinkRegexps": [
+ "^https://www.bing.com/ck/a"
+ ],
+ "extraAdServersRegexps": [
+ "^https://www\\.bing\\.com/acli?c?k"
+ ],
+ "id": "e1eec461-f1f3-40de-b94b-3b670b78108c",
+ "last_modified": 1698666532321
+ }
+ ],
+ "timestamp": 1707833261849
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-au.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-au.xml
new file mode 100644
index 0000000000..6e8801d893
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-au.xml
@@ -0,0 +1,12 @@
+<!-- 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>Amazon.com.au</ShortName>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.com.au/s">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.com.au/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-ca.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-ca.xml
new file mode 100644
index 0000000000..932f62b276
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-ca.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.ca</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.ca/s" resultdomain="amazon.ca">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.ca/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-co-uk.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-co-uk.xml
new file mode 100644
index 0000000000..5ff238cc73
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-co-uk.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.co.uk</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.co.uk/s" resultdomain="amazon.co.uk">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.co.uk/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-de.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-de.xml
new file mode 100644
index 0000000000..137abd4b94
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-de.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.de</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.de/s" resultdomain="amazon.de">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.de/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-es.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-es.xml
new file mode 100644
index 0000000000..c989b9f361
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-es.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.es</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.es/s" resultdomain="amazon.es">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.es/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-fr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-fr.xml
new file mode 100644
index 0000000000..abfb75bee5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-fr.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.fr</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.fr/s" resultdomain="amazon.fr">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.fr/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-in.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-in.xml
new file mode 100644
index 0000000000..2c1be1f73a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-in.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.in</ShortName>
+<InputEncoding>utf-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.in/s">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.in/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-it.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-it.xml
new file mode 100644
index 0000000000..805bbf0af2
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-it.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.it</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.it/s" resultdomain="amazon.it">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.it/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-jp.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-jp.xml
new file mode 100644
index 0000000000..7d57698067
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-jp.xml
@@ -0,0 +1,31 @@
+<!-- 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>Amazon.co.jp</ShortName>
+<Description>検索エンジン - Amazon.co.jp 検索</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://completion.amazon.co.jp/search/complete?q={searchTerms}&amp;client=amazon-search-ui&amp;search-alias=aps&amp;mkt=6"/>
+<Url type="text/html" method="GET" template="https://www.amazon.co.jp/exec/obidos/external-search/" resultdomain="amazon.co.jp">
+ <Param name="field-keywords" value="{searchTerms}"/>
+ <Param name="mode" value="blended"/>
+ <!--
+ <Param name="mode" value="books-jp"/>
+ <Param name="mode" value="books-us"/>
+ -->
+ <Param name="tag" value="moz-jp-mbl-22"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+ <!--
+ <Param name="sz" value="25"/>
+ <Param name="rank" value="+salesrank"/>
+ <Param name="rank" value="+pricerank"/>
+ <Param name="rank" value="+inverse-pricerank"/>
+ <Param name="rank" value="+daterank"/>
+ <Param name="rank" value="+titlerank"/>
+ <Param name="rank" value="-titlerank"/>
+ -->
+</Url>
+<SearchForm>https://www.amazon.co.jp/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-nl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-nl.xml
new file mode 100644
index 0000000000..75530332cc
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-nl.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.nl</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.nl/s">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.nl/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-se.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-se.xml
new file mode 100644
index 0000000000..98f7b2d35a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazon-se.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.se</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.se/s" resultdomain="amazon.se">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.se/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazondotcom.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazondotcom.xml
new file mode 100644
index 0000000000..2af84b936c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/amazondotcom.xml
@@ -0,0 +1,13 @@
+<!-- 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>Amazon.com</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="32" height="32"></Image>
+<Url type="text/html" method="GET" template="https://www.amazon.com/s" resultdomain="amazon.com">
+ <Param name="k" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.amazon.com/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/azerdict.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/azerdict.xml
new file mode 100644
index 0000000000..779c658f60
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/azerdict.xml
@@ -0,0 +1,17 @@
+<!-- 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>Azerdict</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://api.azerdict.com/english/autocomplete">
+ <Param name="action" value="opensearch" />
+ <Param name="query" value="{searchTerms}" />
+</Url>
+<Url type="text/html" method="GET" template="https://azerdict.com/english/" resultdomain="azerdict.com">
+ <Param name="word" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://azerdict.com/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/baidu.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/baidu.xml
new file mode 100644
index 0000000000..494d8b1b06
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/baidu.xml
@@ -0,0 +1,25 @@
+<!-- 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>百度</ShortName>
+ <InputEncoding>UTF-8</InputEncoding>
+ <Image width="16" height="16"></Image>
+ <Url type="application/x-suggestions+json" method="GET" template="https://m.baidu.com/su">
+ <Param name="wd" value="{searchTerms}"/>
+ <Param name="action" value="opensearch"/>
+ <Param name="ie" value="UTF-8"/>b
+ </Url>
+ <Url type="text/html" method="GET" template="https://m.baidu.com/s">
+ <Param name="word" value="{searchTerms}"/>
+ </Url>
+ <!-- As of Bug 861164, we can do a client-side detection to determine whether a user is using
+ tablet or a phone(relative to this case).
+ Note: The order of <URL> DOES not affect the way we pick between them.
+ -->
+ <Url type="application/x-moz-tabletsearch" method="GET" template="https://m.baidu.com/s">
+ <Param name="word" value="{searchTerms}"/>
+ </Url>
+ <SearchForm>https://m.baidu.com/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/bing.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/bing.xml
new file mode 100644
index 0000000000..0b3bc82461
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/bing.xml
@@ -0,0 +1,24 @@
+<!-- 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>Bing</ShortName>
+<Image width="32" height="32"></Image>
+<Url type="application/x-suggestions+json" template="https://www.bing.com/osjson.aspx">
+ <Param name="query" value="{searchTerms}"/>
+ <Param name="language" value="{moz:locale}"/>
+</Url>
+<!-- this is effectively x-moz-phonesearch, but search service expects a text/html entry -->
+<Url type="text/html" method="GET" template="https://www.bing.com/search">
+ <Param name="q" value="{searchTerms}" />
+ <Param name="pc" value="MOZB" />
+ <Param name="form" value="MOZMBA" />
+</Url>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://www.bing.com/search">
+ <Param name="q" value="{searchTerms}" />
+ <Param name="pc" value="MOZA" />
+ <Param name="form" value="MOZAT" />
+</Url>
+<SearchForm>http://www.bing.com</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ceneje.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ceneje.xml
new file mode 100644
index 0000000000..624c8ebde4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ceneje.xml
@@ -0,0 +1,15 @@
+<!-- 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>m.Ceneje.si</ShortName>
+<Description>Mobilni iskalnik Ceneje.si</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image height="16" width="16" type="image/x-icon"></Image>
+<Url type="text/html" method="GET" template="https://www.ceneje.si/search_new.aspx">
+ <Param name="q" value="{searchTerms}" />
+ <Param name="FF-SearchBox" value="1" />
+</Url>
+<SearchForm>https://ceneje.si</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/coccoc.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/coccoc.xml
new file mode 100644
index 0000000000..1baa2a3118
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/coccoc.xml
@@ -0,0 +1,21 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!-- 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>Cốc Cốc</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET"
+ template="https://coccoc.com/composer/autocomplete">
+ <Param name="of" value="b" />
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="s" value="ff"/>
+</Url>
+<Url type="text/html" method="GET" template="https://coccoc.com/search" resultdomain="coccoc.com">
+ <Param name="query" value="{searchTerms}"/>
+ <Param name="s" value="ff"/>
+ <Param name="utm_source" value="ffmobile"/>
+</Url>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/daum-kr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/daum-kr.xml
new file mode 100644
index 0000000000..b86cb446ee
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/daum-kr.xml
@@ -0,0 +1,21 @@
+<!-- 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>다음</ShortName>
+<InputEncoding>utf-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="http://sug.search.daum.net/search_nsuggest">
+ <Param name="mod" value="fxjson" />
+ <Param name="code" value="utf_in_out" />
+ <Param name="q" value="{searchTerms}" />
+</Url>
+
+<Url type="text/html" method="GET" template="https://m.search.daum.net/search">
+ <Param name="w" value="tot"/>
+ <Param name="nil_ch" value="ffsr"/>
+ <Param name="q" value="{searchTerms}"/>
+</Url>
+<SearchForm>http://m.search.daum.net</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ddg.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ddg.xml
new file mode 100644
index 0000000000..21364b08aa
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ddg.xml
@@ -0,0 +1,23 @@
+<!-- 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>DuckDuckGo</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="32" height="32"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ac.duckduckgo.com/ac/">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="type" value="list"/>
+</Url>
+<!-- this is effectively x-moz-phonesearch, but search service expects a text/html entry -->
+<Url type="text/html" method="GET" template="https://duckduckgo.com/">
+ <Param name="q" value="{searchTerms}" />
+ <Param name="t" value="fpas" />
+</Url>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://duckduckgo.com/">
+ <Param name="q" value="{searchTerms}" />
+ <Param name="t" value="fpas" />
+</Url>
+<SearchForm>https://duckduckgo.com</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-at.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-at.xml
new file mode 100644
index 0000000000..31ac7f6b9b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-at.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.at/autosug?sId=16&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.at/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="5221-53469-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.at/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-au.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-au.xml
new file mode 100644
index 0000000000..490b34de9e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-au.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.au/autosug?sId=15&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.com.au/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="705-53470-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.com.au/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-befr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-befr.xml
new file mode 100644
index 0000000000..db76954be3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-befr.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.befr.ebay.be/autosug?sId=23&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.befr.ebay.be/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="1553-53471-19255-0"/>
+</Url>
+<SearchForm>https://www.befr.ebay.be/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ca.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ca.xml
new file mode 100644
index 0000000000..7cf4fc531e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ca.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.ca/autosug?sId=2&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.ca/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="706-53473-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.ca/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ch.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ch.xml
new file mode 100644
index 0000000000..7557e24c6c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ch.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.ch/autosug?sId=193&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.ch/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="5222-53480-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.ch/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-co-uk.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-co-uk.xml
new file mode 100644
index 0000000000..06e8f22820
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-co-uk.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.co.uk/autosug?sId=3&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.co.uk/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="710-53481-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.co.uk/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-de.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-de.xml
new file mode 100644
index 0000000000..8e05c24fe8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-de.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.de/autosug?sId=77&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.de/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="707-53477-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.de/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-es.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-es.xml
new file mode 100644
index 0000000000..6e7d644d1d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-es.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.es/autosug?sId=186&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.es/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="1185-53479-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.es/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-fr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-fr.xml
new file mode 100644
index 0000000000..a060af1ca4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-fr.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.fr/autosug?sId=71&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.fr/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="709-53476-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.fr/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ie.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ie.xml
new file mode 100644
index 0000000000..2397a28e79
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-ie.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.ie/autosug?sId=205&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.ie/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="5282-53468-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.ie/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-it.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-it.xml
new file mode 100644
index 0000000000..8fe7c4313b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-it.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.it/autosug?sId=101&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.it/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="724-53478-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.it/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-nl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-nl.xml
new file mode 100644
index 0000000000..f265d912a6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-nl.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.nl/autosug?sId=146&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.nl/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="1346-53482-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.nl/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-pl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-pl.xml
new file mode 100644
index 0000000000..66ec476113
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay-pl.xml
@@ -0,0 +1,18 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="text/html" method="GET" template="https://www.ebay.pl/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="4908-226936-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.pl/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay.xml
new file mode 100644
index 0000000000..67c9251ef5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ebay.xml
@@ -0,0 +1,19 @@
+<!-- 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>eBay</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="68" height="68"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://autosug.ebay.com/autosug?sId=0&amp;fmt=osr&amp;kwd={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.ebay.com/sch/">
+ <Param name="kw" value="{searchTerms}"/>
+ <Param name="toolid" value="20004"/>
+ <Param name="campid" value="5338791379"/>
+ <Param name="mkevt" value="1"/>
+ <Param name="mkcid" value="1"/>
+ <Param name="mkrid" value="711-53200-19255-0"/>
+</Url>
+<SearchForm>https://www.ebay.com/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ecosia.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ecosia.xml
new file mode 100644
index 0000000000..81d2aa4856
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/ecosia.xml
@@ -0,0 +1,18 @@
+<!-- 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>Ecosia</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ac.ecosia.org/autocomplete">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="type" value="list"/>
+</Url>
+<Url type="text/html" method="GET" template="https://www.ecosia.org/search">
+ <Param name="tt" value="813cf1dd"/>
+ <Param name="q" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.ecosia.org/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/faclair-beag.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/faclair-beag.xml
new file mode 100644
index 0000000000..a2fcef2fe2
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/faclair-beag.xml
@@ -0,0 +1,13 @@
+<!-- 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>Am Faclair Beag</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.faclair.com/m" resultdomain="faclair.com">
+ <Param name="txtSearch" value="{searchTerms}" />
+</Url>
+<SearchForm>https://www.faclair.com/m</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-1-m.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-1-m.xml
new file mode 100644
index 0000000000..ed4ee58ad3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-1-m.xml
@@ -0,0 +1,17 @@
+<!-- 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>Google</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://www.google.com/complete/search?client=firefox&amp;q={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.google.com/search">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="ie" value="utf-8"/>
+ <Param name="oe" value="utf-8"/>
+ <Param name="client" value="firefox-b-1-m"/>
+</Url>
+<SearchForm>https://www.google.com</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-m.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-m.xml
new file mode 100644
index 0000000000..232bbce214
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-b-m.xml
@@ -0,0 +1,17 @@
+<!-- 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>Google</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://www.google.com/complete/search?client=firefox&amp;q={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.google.com/search">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="ie" value="utf-8"/>
+ <Param name="oe" value="utf-8"/>
+ <Param name="client" value="firefox-b-m"/>
+</Url>
+<SearchForm>https://www.google.com</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-com-nocodes.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-com-nocodes.xml
new file mode 100644
index 0000000000..2df8e9d277
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/google-com-nocodes.xml
@@ -0,0 +1,14 @@
+<!-- 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>Google</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://www.google.com/complete/search?client=firefox&amp;q={searchTerms}"/>
+<Url type="text/html" method="GET" template="https://www.google.com/search">
+ <Param name="q" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.google.com</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/gulesider-mobile-NO.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/gulesider-mobile-NO.xml
new file mode 100644
index 0000000000..bfd642190a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/gulesider-mobile-NO.xml
@@ -0,0 +1,15 @@
+<!-- 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>Gule sider mobil</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.gulesider.no/search">
+ <Param name="what" value="all" />
+ <Param name="cmpid" value="fre_partner_fire_gssbtop"/>
+ <Param name="q" value="{searchTerms}"/>
+</Url>
+<SearchForm>http://m.gulesider.no/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/leo_ende_de.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/leo_ende_de.xml
new file mode 100644
index 0000000000..5e8eb8681f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/leo_ende_de.xml
@@ -0,0 +1,18 @@
+<!-- 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>LEO Eng-Tud</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="get" template="https://dict.leo.org/dictQuery/m-query/conf/ende/query.conf/strlist.json">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="sort" value="PLa"/>
+ <Param name="shortQuery"/>
+ <Param name="noDescription"/>
+ <Param name="noQueryURLs"/>
+</Url>
+<Url type="text/html" method="GET" template="https://dict.leo.org/englisch-deutsch/{searchTerms}" resultdomain="leo.org"/>
+<SearchForm>https://dict.leo.org</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mapy-cz.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mapy-cz.xml
new file mode 100644
index 0000000000..6f198bf526
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mapy-cz.xml
@@ -0,0 +1,14 @@
+<!-- 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>Mapy.cz</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.mapy.cz/" resultdomain="mapy.cz">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="sourceid" value="Searchmodule_3"/>
+</Url>
+<SearchForm>https://www.mapy.cz/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-ar.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-ar.xml
new file mode 100644
index 0000000000..f886dad815
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-ar.xml
@@ -0,0 +1,14 @@
+<!-- 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>MercadoLibre Argentina</ShortName>
+<InputEncoding>ISO-8859-1</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.mercadolibre.com.ar/jm/search">
+ <Param name="as_word" value="{searchTerms}"/>
+ <Param name="source" value="firefox_box"/>
+</Url>
+<SearchForm>https://www.mercadolibre.com.ar/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-cl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-cl.xml
new file mode 100644
index 0000000000..3881fc348f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-cl.xml
@@ -0,0 +1,14 @@
+<!-- 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>MercadoLibre Chile</ShortName>
+<InputEncoding>ISO-8859-1</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.mercadolibre.cl/jm/search">
+ <Param name="as_word" value="{searchTerms}"/>
+ <Param name="source" value="firefox_box"/>
+</Url>
+<SearchForm>https://www.mercadolibre.cl/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-mx.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-mx.xml
new file mode 100644
index 0000000000..ae628ab554
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/mercadolibre-mx.xml
@@ -0,0 +1,14 @@
+<!-- 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>MercadoLibre Mexico</ShortName>
+<InputEncoding>ISO-8859-1</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.mercadolibre.com.mx/jm/search">
+ <Param name="as_word" value="{searchTerms}"/>
+ <Param name="source" value="firefox_box"/>
+</Url>
+<SearchForm>https://www.mercadolibre.com.mx/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/odpiralni.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/odpiralni.xml
new file mode 100644
index 0000000000..5ad1167292
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/odpiralni.xml
@@ -0,0 +1,15 @@
+<!-- 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>Odpiralni Časi</ShortName>
+<Description>Odpiralni Časi v Sloveniji</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<UpdateUrl>https://www.odpiralnicasi.com/opensearch/description.xml</UpdateUrl>
+<Url type="text/html" method="GET" template="https://www.odpiralnicasi.com/spots">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="source" value="1"/>
+</Url>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/pazaruvaj.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/pazaruvaj.xml
new file mode 100644
index 0000000000..8e79095eec
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/pazaruvaj.xml
@@ -0,0 +1,14 @@
+<!-- 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>Pazaruvaj</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.pazaruvaj.com/CategorySearch.php"
+ resultdomain="pazaruvaj.com">
+ <Param name="st" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.pazaruvaj.com/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/prisjakt-sv-SE.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/prisjakt-sv-SE.xml
new file mode 100644
index 0000000000..1027b39dad
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/prisjakt-sv-SE.xml
@@ -0,0 +1,14 @@
+<!-- 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>Prisjakt</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://www.prisjakt.nu/plugins/opensearch/suggestions.php">
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://m.prisjakt.nu/search/{searchTerms}"/>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://www.prisjakt.nu/#rparams=ss={searchTerms}"/>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/qwant.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/qwant.xml
new file mode 100644
index 0000000000..e5b9dfb328
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/qwant.xml
@@ -0,0 +1,18 @@
+<!-- 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>Qwant</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://api.qwant.com/api/suggest/">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="client" value="ff_android"/>
+</Url>
+<Url type="text/html" method="GET" template="https://www.qwant.com/">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="client" value="ff_android"/>
+</Url>
+<SearchForm>https://www.qwant.com/</SearchForm>
+</SearchPlugin> \ No newline at end of file
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/rakuten.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/rakuten.xml
new file mode 100644
index 0000000000..1b6d0e18e8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/rakuten.xml
@@ -0,0 +1,16 @@
+<!-- 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>楽天市場</ShortName>
+ <Description>楽天市場 商品検索</Description>
+ <InputEncoding>EUC-JP</InputEncoding>
+ <Image height="16" width="16"></Image>
+ <Url method="GET" template="https://pt.afl.rakuten.co.jp/c/013ca98b.cd7c5f0c/" type="text/html">
+ <Param name="sv" value="2" />
+ <Param name="p" value="0" />
+ <Param name="sitem" value="{searchTerms}" />
+ </Url>
+ <SearchForm>https://www.rakuten.co.jp/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/reddit.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/reddit.xml
new file mode 100644
index 0000000000..4f761ba236
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/reddit.xml
@@ -0,0 +1,11 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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>Reddit</ShortName>
+ <Description>Search Reddit</Description>
+ <LongName>Reddit Search</LongName>
+ <Image width="16" height="16"></Image>
+ <Url type="text/html" method="get" template="https://www.reddit.com/search/?q={searchTerms}"/>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/salidzinilv.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/salidzinilv.xml
new file mode 100644
index 0000000000..436261f8f5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/salidzinilv.xml
@@ -0,0 +1,18 @@
+<!-- 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>Salidzini.lv</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://www.salidzini.lv/search.php">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="utm_source" value="firefox_mobile"/>
+</Url>
+<Url type="application/x-suggestions+json" method="GET" template="https://www.salidzini.lv/suggested_search.php">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="utm_source" value="firefox_mobile"/>
+</Url>
+<SearchForm>https://salidzini.lv</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/seznam-cz.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/seznam-cz.xml
new file mode 100644
index 0000000000..98132cd2a5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/seznam-cz.xml
@@ -0,0 +1,17 @@
+<!-- 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>Seznam</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://suggest.seznam.cz/fulltext_ff">
+ <Param name="phrase" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://search.seznam.cz/">
+ <Param name="q" value="{searchTerms}"/>
+ <Param name="sourceid" value="SearchBox"/>
+</Url>
+<SearchForm>http://search.seznam.cz/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/vatera.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/vatera.xml
new file mode 100644
index 0000000000..8e0929c0b4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/vatera.xml
@@ -0,0 +1,17 @@
+<!-- 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>Vatera.hu</ShortName>
+ <Language>hu</Language>
+ <OutputEncoding>ISO-8859-2</OutputEncoding>
+ <InputEncoding>ISO-8859-2</InputEncoding>
+ <Image width="16" height="16"></Image>
+ <Url type="text/html" method="GET" template="https://www.vatera.hu/listings/index.php">
+ <Param name="c" value="0" />
+ <Param name="td" value="on" />
+ <Param name="q" value="{searchTerms}" />
+ </Url>
+ <SearchForm>http://m.vatera.hu/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NN.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NN.xml
new file mode 100644
index 0000000000..f876b15112
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NN.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (nn)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://nn.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://nn.wikipedia.org/wiki/Spesial:Søk">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://nn.wikipedia.org/wiki/Spesial:Søk</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NO.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NO.xml
new file mode 100644
index 0000000000..75bb1768f6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-NO.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (no)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://no.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://no.wikipedia.org/wiki/Spesial:Søk">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://no.wikipedia.org/wiki/Spesial:Søk</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-an.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-an.xml
new file mode 100644
index 0000000000..dbaddfcb5c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-an.xml
@@ -0,0 +1,18 @@
+<!-- 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>Biquipedia (an)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://an.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://an.wikipedia.org/wiki/Especial:Mirar">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://an.wikipedia.org/wiki/Especial:Mirar</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ar.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ar.xml
new file mode 100644
index 0000000000..78c63e61c9
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ar.xml
@@ -0,0 +1,18 @@
+<!-- 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>ويكيبيديا (ar)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ar.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ar.wikipedia.org/wiki/خاص:بحث">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ar.wikipedia.org/wiki/خاص:بحث</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-as.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-as.xml
new file mode 100644
index 0000000000..914e513f0e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-as.xml
@@ -0,0 +1,18 @@
+<!-- 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>অসমীয়া ৱিকিপিডিয়া (as)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://as.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://as.wikipedia.org/wiki/বিশেষ:সন্ধান">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://as.wikipedia.org/wiki/বিশেষ:সন্ধান</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ast.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ast.xml
new file mode 100644
index 0000000000..0a32de3219
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ast.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (ast)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ast.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ast.wikipedia.org/wiki/Especial:Gueta">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ast.wikipedia.org/wiki/Especial:Gueta</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-az.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-az.xml
new file mode 100644
index 0000000000..dd87a40cc6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-az.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vikipediya (az)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://az.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://az.wikipedia.org/wiki/Xüsusi:Axtar">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://az.wikipedia.org/wiki/Xüsusi:Axtar</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-be.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-be.xml
new file mode 100644
index 0000000000..9aa6bd3651
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-be.xml
@@ -0,0 +1,18 @@
+<!-- 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>Вікіпедыя (be)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://be.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://be.wikipedia.org/wiki/Адмысловае:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://be.wikipedia.org/wiki/Адмысловае:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bg.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bg.xml
new file mode 100644
index 0000000000..4b98497add
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bg.xml
@@ -0,0 +1,18 @@
+<!-- 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>Уикипедия (bg)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://bg.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://bg.wikipedia.org/wiki/Специални:Търсене">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://bg.wikipedia.org/wiki/Специални:Търсене</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bn.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bn.xml
new file mode 100644
index 0000000000..b564438636
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bn.xml
@@ -0,0 +1,18 @@
+<!-- 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>উইকিপিডিয়া (bn)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://bn.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://bn.wikipedia.org/wiki/বিশেষ:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://bn.wikipedia.org/wiki/বিশেষ:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-br.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-br.xml
new file mode 100644
index 0000000000..41bf921eb4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-br.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (br)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://br.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://br.wikipedia.org/wiki/Dibar:Klask">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://br.wikipedia.org/wiki/Dibar:Klask</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bs.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bs.xml
new file mode 100644
index 0000000000..f54aed3b54
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-bs.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (bs)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://bs.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://bs.wikipedia.org/wiki/Posebno:Pretraga">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://bs.wikipedia.org/wiki/Posebno:Pretraga</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ca.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ca.xml
new file mode 100644
index 0000000000..3b08c30c29
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ca.xml
@@ -0,0 +1,18 @@
+<!-- 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>Viquipèdia (ca)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ca.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ca.wikipedia.org/wiki/Especial:Cerca">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ca.wikipedia.org/wiki/Especial:Cerca</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cy.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cy.xml
new file mode 100644
index 0000000000..72c526eee4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cy.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wicipedia (cy)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://cy.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://cy.wikipedia.org/wiki/Arbennig:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://cy.wikipedia.org/wiki/Arbennig:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cz.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cz.xml
new file mode 100644
index 0000000000..9fa115ce52
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-cz.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedie (cs)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://cs.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://cs.wikipedia.org/wiki/Speciální:Hledání">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://cs.wikipedia.org/wiki/Speciální:Hledání</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-da.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-da.xml
new file mode 100644
index 0000000000..7ac52b86f6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-da.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (da)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://da.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://da.wikipedia.org/wiki/Speciel:Søgning">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://da.wikipedia.org/wiki/Speciel:Søgning</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-de.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-de.xml
new file mode 100644
index 0000000000..1052b85311
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-de.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (de)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://de.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://de.wikipedia.org/wiki/Spezial:Suche">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://de.wikipedia.org/wiki/Spezial:Suche</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-dsb.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-dsb.xml
new file mode 100644
index 0000000000..52ce1e97b3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-dsb.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedija (dsb)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://dsb.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://dsb.wikipedia.org/wiki/Specialne:Pytaś">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://dsb.wikipedia.org/wiki/Specialne:Pytaś</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-el.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-el.xml
new file mode 100644
index 0000000000..db902d9813
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-el.xml
@@ -0,0 +1,18 @@
+<!-- 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>Βικιπαίδεια (el)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://el.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://el.wikipedia.org/wiki/Ειδικό:Αναζήτηση">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://el.wikipedia.org/wiki/Ειδικό:Αναζήτηση</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eo.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eo.xml
new file mode 100644
index 0000000000..b42dd74658
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eo.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vikipedio (eo)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://eo.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://eo.wikipedia.org/wiki/Specialaĵo:Serĉi">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://eo.wikipedia.org/wiki/Specialaĵo:Serĉi</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-es.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-es.xml
new file mode 100644
index 0000000000..41dac576b7
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-es.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (es)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://es.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://es.wikipedia.org/wiki/Especial:Buscar">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://es.wikipedia.org/wiki/Especial:Buscar</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-et.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-et.xml
new file mode 100644
index 0000000000..c499e90397
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-et.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vikipeedia (et)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://et.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://et.wikipedia.org/wiki/Eri:Otsimine">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://et.wikipedia.org/wiki/Eri:Otsimine</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eu.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eu.xml
new file mode 100644
index 0000000000..9928360286
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-eu.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (eu)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://eu.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://eu.wikipedia.org/wiki/Berezi:Bilatu">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://eu.wikipedia.org/wiki/Berezi:Bilatu</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fa.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fa.xml
new file mode 100644
index 0000000000..757c4cc575
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fa.xml
@@ -0,0 +1,18 @@
+<!-- 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>ویکی‌پدیا (fa)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://fa.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://fa.wikipedia.org/wiki/ویژه:جستجو">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://fa.wikipedia.org/wiki/ویژه:جستجو</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fi.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fi.xml
new file mode 100644
index 0000000000..2612d72193
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fi.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (fi)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://fi.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://fi.wikipedia.org/wiki/Toiminnot:Haku">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://fi.wikipedia.org/wiki/Toiminnot:Haku</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fr.xml
new file mode 100644
index 0000000000..6f44da1fff
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fr.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipédia (fr)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://fr.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://fr.wikipedia.org/wiki/Spécial:Recherche">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://fr.wikipedia.org/wiki/Spécial:Recherche</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fy-NL.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fy-NL.xml
new file mode 100644
index 0000000000..9bfb0a97a0
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-fy-NL.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (fy)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://fy.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://fy.wikipedia.org/wiki/Wiki:Sykje">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://fy.wikipedia.org/wiki/Wiki:Sykje</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ga-IE.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ga-IE.xml
new file mode 100644
index 0000000000..d954b45615
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ga-IE.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vicipéid (ga)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ga.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ga.wikipedia.org/wiki/Speisialta:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ga.wikipedia.org/wiki/Speisialta:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gd.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gd.xml
new file mode 100644
index 0000000000..4b87ed7450
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gd.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (gd)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://gd.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://gd.wikipedia.org/wiki/Sònraichte:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://gd.wikipedia.org/wiki/Sònraichte:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gl.xml
new file mode 100644
index 0000000000..6b354639dc
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gl.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (gl)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://gl.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://gl.wikipedia.org/wiki/Especial:Procurar">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://gl.wikipedia.org/wiki/Especial:Procurar</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gn.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gn.xml
new file mode 100644
index 0000000000..17a12cf997
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gn.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vikipetã (gn)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://gn.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://gn.wikipedia.org/wiki/Mba'echĩchĩ:Buscar">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://gn.wikipedia.org/wiki/Mba'echĩchĩ:Buscar</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gu.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gu.xml
new file mode 100644
index 0000000000..0fd505d003
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-gu.xml
@@ -0,0 +1,18 @@
+<!-- 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>વિકિપીડિયા (gu)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://gu.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://gu.wikipedia.org/wiki/વિશેષ:શોધ">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://gu.wikipedia.org/wiki/વિશેષ:શોધ</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-he.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-he.xml
new file mode 100644
index 0000000000..2611582d7c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-he.xml
@@ -0,0 +1,18 @@
+<!-- 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>ויקיפדיה</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://he.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://he.wikipedia.org/wiki/מיוחד:חיפוש">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://he.wikipedia.org/wiki/מיוחד:חיפוש</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hi.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hi.xml
new file mode 100644
index 0000000000..e3a28a278c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hi.xml
@@ -0,0 +1,18 @@
+<!-- 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>विकिपीडिया (hi)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://hi.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://hi.wikipedia.org/wiki/विशेष:खोज">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://hi.wikipedia.org/wiki/विशेष:खोज</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hr.xml
new file mode 100644
index 0000000000..02e514dbd7
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hr.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedija (hr)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://hr.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://hr.wikipedia.org/wiki/Posebno:Traži">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://hr.wikipedia.org/wiki/Posebno:Traži</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hsb.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hsb.xml
new file mode 100644
index 0000000000..f72d268f9b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hsb.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedija (hsb)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://hsb.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://hsb.wikipedia.org/wiki/Specialnje:Pytać">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://hsb.wikipedia.org/wiki/Specialnje:Pytać</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hu.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hu.xml
new file mode 100644
index 0000000000..7d39f2c500
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hu.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipédia (hu)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://hu.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://hu.wikipedia.org/wiki/Speciális:Keresés">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://hu.wikipedia.org/wiki/Speciális:Keresés</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hy-AM.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hy-AM.xml
new file mode 100644
index 0000000000..bff51bbe49
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-hy-AM.xml
@@ -0,0 +1,18 @@
+<!-- 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>Վիքիպեդիա (hy)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://hy.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://hy.wikipedia.org/wiki/Սպասարկող:Որոնել">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://hy.wikipedia.org/wiki/Սպասարկող:Որոնել</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ia.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ia.xml
new file mode 100644
index 0000000000..4994148c75
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ia.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (ia)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ia.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ia.wikipedia.org/wiki/Special:Recerca">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ia.wikipedia.org/wiki/Special:Recerca</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-id.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-id.xml
new file mode 100644
index 0000000000..0e82cbcb8f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-id.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (id)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://id.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://id.wikipedia.org/wiki/Istimewa:Pencarian">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://id.wikipedia.org/wiki/Istimewa:Pencarian</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-is.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-is.xml
new file mode 100644
index 0000000000..093f45cc77
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-is.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (is)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://is.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://is.wikipedia.org/wiki/Kerfissíða:Leit">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://is.wikipedia.org/wiki/Kerfissíða:Leit</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-it.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-it.xml
new file mode 100644
index 0000000000..a04aeb42f8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-it.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (it)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://it.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://it.wikipedia.org/wiki/Speciale:Ricerca">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://it.wikipedia.org/wiki/Speciale:Ricerca</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ja.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ja.xml
new file mode 100644
index 0000000000..c27327138b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ja.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (ja)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ja.wikipedia.org/wiki">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ja.wikipedia.org/wiki/特別:検索">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ja.wikipedia.org/wiki/特別:検索</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ka.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ka.xml
new file mode 100644
index 0000000000..15afd0174f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ka.xml
@@ -0,0 +1,18 @@
+<!-- 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>ვიკიპედია (ka)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ka.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ka.wikipedia.org/wiki/სპეციალური:ძიება">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ka.wikipedia.org/wiki/სპეციალური:ძიება</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kab.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kab.xml
new file mode 100644
index 0000000000..7ef24e7aba
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kab.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (kab)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://kab.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://kab.wikipedia.org/wiki/Uslig:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://kab.wikipedia.org/wiki/Uslig:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kk.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kk.xml
new file mode 100644
index 0000000000..328edd40c0
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kk.xml
@@ -0,0 +1,18 @@
+<!-- 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>Уикипедия (kk)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://kk.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://kk.wikipedia.org/wiki/Арнайы:Іздеу">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://kk.wikipedia.org/wiki/Арнайы:Іздеу</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-km.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-km.xml
new file mode 100644
index 0000000000..a69f99b34d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-km.xml
@@ -0,0 +1,18 @@
+<!-- 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>វិគីភីឌា (km)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://km.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://km.wikipedia.org/wiki/ពិសេស:ស្វែងរក">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://km.wikipedia.org/wiki/ពិសេស:ស្វែងរក</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kn.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kn.xml
new file mode 100644
index 0000000000..155b515fed
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-kn.xml
@@ -0,0 +1,18 @@
+<!-- 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>ವಿಕಿಪೀಡಿಯ (kn)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://kn.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://kn.wikipedia.org/wiki/ವಿಶೇಷ:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://kn.wikipedia.org/wiki/ವಿಶೇಷ:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lij.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lij.xml
new file mode 100644
index 0000000000..1b75c36c79
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lij.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (lij)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://lij.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://lij.wikipedia.org/wiki/Speçiale:Riçerca">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://lij.wikipedia.org/wiki/Speçiale:Riçerca</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lo.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lo.xml
new file mode 100644
index 0000000000..64b236617b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lo.xml
@@ -0,0 +1,18 @@
+<!-- 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>ວິກິພີເດຍ (lo)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://lo.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://lo.wikipedia.org/wiki/ພິເສດ:ຊອກຫາ">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://lo.wikipedia.org/wiki/ພິເສດ:ຊອກຫາ</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lt.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lt.xml
new file mode 100644
index 0000000000..8430cb6591
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lt.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (lt)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://lt.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://lt.wikipedia.org/wiki/Specialus:Paieška">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://lt.wikipedia.org/wiki/Specialus:Paieška</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ltg.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ltg.xml
new file mode 100644
index 0000000000..384d206107
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ltg.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vikipedeja (ltg)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ltg.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ltg.wikipedia.org/wiki/Seviškuo:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ltg.wikipedia.org/wiki/Seviškuo:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lv.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lv.xml
new file mode 100644
index 0000000000..e25485901a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-lv.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vikipēdija (lv)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://lv.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://lv.wikipedia.org/wiki/Special:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://lv.wikipedia.org/wiki/Special:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ml.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ml.xml
new file mode 100644
index 0000000000..1d8a5df23c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ml.xml
@@ -0,0 +1,18 @@
+<!-- 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>വിക്കിപീഡിയ (ml)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ml.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ml.wikipedia.org/wiki/പ്രത്യേകം:അന്വേഷണം">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ml.wikipedia.org/wiki/പ്രത്യേകം:അന്വേഷണം</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-mr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-mr.xml
new file mode 100644
index 0000000000..a73dac3d99
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-mr.xml
@@ -0,0 +1,18 @@
+<!-- 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>विकिपीडिया (mr)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://mr.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://mr.wikipedia.org/wiki/विशेष:शोधा">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://mr.wikipedia.org/wiki/विशेष:शोधा</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ms.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ms.xml
new file mode 100644
index 0000000000..5b892124cf
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ms.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (ms)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ms.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ms.wikipedia.org/wiki/Khas:Gelintar">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ms.wikipedia.org/wiki/Khas:Gelintar</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-my.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-my.xml
new file mode 100644
index 0000000000..cc781dee19
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-my.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (my)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://my.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://my.wikipedia.org/wiki/Special:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://my.wikipedia.org/wiki/Special:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ne.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ne.xml
new file mode 100644
index 0000000000..93118b3057
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ne.xml
@@ -0,0 +1,18 @@
+<!-- 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>विकिपीडिया (ne)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ne.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ne.wikipedia.org/wiki/विशेष:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ne.wikipedia.org/wiki/विशेष:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-nl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-nl.xml
new file mode 100644
index 0000000000..36b41dd48a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-nl.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (nl)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://nl.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://nl.wikipedia.org/wiki/Speciaal:Zoeken">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://nl.wikipedia.org/wiki/Speciaal:Zoeken</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-oc.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-oc.xml
new file mode 100644
index 0000000000..6493a0b1ed
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-oc.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipèdia (oc)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://oc.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://oc.wikipedia.org/wiki/Especial:Recèrca">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://oc.wikipedia.org/wiki/Especial:Recèrca</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-or.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-or.xml
new file mode 100644
index 0000000000..109e9e0fb6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-or.xml
@@ -0,0 +1,18 @@
+<!-- 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>ଉଇକିପିଡ଼ିଆ (or)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://or.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://or.wikipedia.org/wiki/ବିଶେଷ:ଖୋଜନ୍ତୁ">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://or.wikipedia.org/wiki/ବିଶେଷ:ଖୋଜନ୍ତୁ</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pa.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pa.xml
new file mode 100644
index 0000000000..b423d887da
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pa.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (pa)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://pa.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://pa.wikipedia.org/wiki/ਖ਼ਾਸ:ਖੋਜੋ">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://pa.wikipedia.org/wiki/ਖ਼ਾਸ:ਖੋਜੋ</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pl.xml
new file mode 100644
index 0000000000..e8fb1aafb3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pl.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (pl)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://pl.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://pl.wikipedia.org/wiki/Specjalna:Szukaj">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://pl.wikipedia.org/wiki/Specjalna:Szukaj</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pt.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pt.xml
new file mode 100644
index 0000000000..1595395e25
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-pt.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (pt)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://pt.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://pt.wikipedia.org/wiki/Especial:Pesquisar">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://pt.wikipedia.org/wiki/Especial:Pesquisar</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-rm.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-rm.xml
new file mode 100644
index 0000000000..aca3ffdfb6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-rm.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (rm)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://rm.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://rm.wikipedia.org/wiki/Spezial:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://rm.wikipedia.org/wiki/Spezial:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ro.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ro.xml
new file mode 100644
index 0000000000..f91f665dc1
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ro.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (ro)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ro.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ro.wikipedia.org/wiki/Special:Căutare">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ro.wikipedia.org/wiki/Special:Căutare</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ru.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ru.xml
new file mode 100644
index 0000000000..33a9a541bd
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ru.xml
@@ -0,0 +1,18 @@
+<!-- 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>Википедия (ru)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ru.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ru.wikipedia.org/wiki/Служебная:Поиск">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ru.wikipedia.org/wiki/Служебная:Поиск</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sk.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sk.xml
new file mode 100644
index 0000000000..44c3f0326a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sk.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipédia (sk)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://sk.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://sk.wikipedia.org/wiki/Špeciálne:Hľadanie">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://sk.wikipedia.org/wiki/Špeciálne:Hľadanie</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sl.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sl.xml
new file mode 100644
index 0000000000..7ff7347b0a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sl.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedija (sl)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://sl.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://sl.wikipedia.org/wiki/Posebno:Iskanje">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://sl.wikipedia.org/wiki/Posebno:Iskanje</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sq.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sq.xml
new file mode 100644
index 0000000000..c100012f37
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sq.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (sq)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://sq.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://sq.wikipedia.org/wiki/Speciale:Kërkim">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://sq.wikipedia.org/wiki/Speciale:Kërkim</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sr.xml
new file mode 100644
index 0000000000..3368eea67e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sr.xml
@@ -0,0 +1,18 @@
+<!-- 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>Википедија (sr)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://sr.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://sr.wikipedia.org/wiki/Посебно:Претражи">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://sr.wikipedia.org/wiki/Посебно:Претражи</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sv-SE.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sv-SE.xml
new file mode 100644
index 0000000000..3595d01bab
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-sv-SE.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (sv)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://sv.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://sv.wikipedia.org/wiki/Special:Sök">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://sv.wikipedia.org/wiki/Special:Sök</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ta.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ta.xml
new file mode 100644
index 0000000000..778e39f733
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ta.xml
@@ -0,0 +1,18 @@
+<!-- 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>விக்கிப்பீடியா (ta)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ta.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ta.wikipedia.org/wiki/சிறப்பு:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ta.wikipedia.org/wiki/சிறப்பு:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-te.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-te.xml
new file mode 100644
index 0000000000..79c2b04e8c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-te.xml
@@ -0,0 +1,18 @@
+<!-- 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>వికీపీడియా (te)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://te.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://te.wikipedia.org/wiki/ప్రత్యేక:అన్వేషణ">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://te.wikipedia.org/wiki/ప్రత్యేక:అన్వేషణ</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-th.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-th.xml
new file mode 100644
index 0000000000..cbcf5319c9
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-th.xml
@@ -0,0 +1,18 @@
+<!-- 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>วิกิพีเดีย</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://th.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://th.wikipedia.org/wiki/พิเศษ:ค้นหา">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://th.wikipedia.org/wiki/พิเศษ:ค้นหา</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-tr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-tr.xml
new file mode 100644
index 0000000000..4bc9e189d3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-tr.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (tr)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://tr.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://tr.wikipedia.org/wiki/Özel:Ara">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://tr.wikipedia.org/wiki/Özel:Ara</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uk.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uk.xml
new file mode 100644
index 0000000000..bc2f4741a8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uk.xml
@@ -0,0 +1,18 @@
+<!-- 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>Вікіпедія</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://uk.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://uk.wikipedia.org/wiki/Спеціальна:Пошук">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://uk.wikipedia.org/wiki/Спеціальна:Пошук</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ur.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ur.xml
new file mode 100644
index 0000000000..e4b0c8a616
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-ur.xml
@@ -0,0 +1,18 @@
+<!-- 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>ویکیپیڈیا (ur)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ur.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://ur.wikipedia.org/wiki/خاص:تلاش">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://ur.wikipedia.org/wiki/خاص:تلاش</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uz.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uz.xml
new file mode 100644
index 0000000000..a33e6a4d82
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-uz.xml
@@ -0,0 +1,18 @@
+<!-- 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>Vikipediya (uz)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://uz.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://uz.wikipedia.org/wiki/Maxsus:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://uz.wikipedia.org/wiki/Maxsus:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-vi.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-vi.xml
new file mode 100644
index 0000000000..e3ac224fd9
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-vi.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (vi)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="get" template="https://vi.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="get" template="https://vi.wikipedia.org/wiki/Đặc_biệt:Tìm_kiếm">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://vi.wikipedia.org/wiki/Đặc_biệt:Tìm_kiếm</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-wo.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-wo.xml
new file mode 100644
index 0000000000..39caa9839b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-wo.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia (wo)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://wo.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://wo.wikipedia.org/wiki/Jagleel:Ceet">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://wo.wikipedia.org/wiki/Jagleel:Ceet</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-CN.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-CN.xml
new file mode 100644
index 0000000000..a168d71dea
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-CN.xml
@@ -0,0 +1,18 @@
+<!-- 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>维基百科</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://zh.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://zh.wikipedia.org/wiki/Special:搜索">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://zh.wikipedia.org/wiki/Special:搜索</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-TW.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-TW.xml
new file mode 100644
index 0000000000..aeaabe6811
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia-zh-TW.xml
@@ -0,0 +1,19 @@
+<!-- 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>Wikipedia (zh)</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://zh.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://zh.wikipedia.org/wiki/Special:搜索">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+ <Param name="variant" value="zh-tw"/>
+</Url>
+<SearchForm>https://zh.wikipedia.org/wiki/Special:搜索</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia.xml
new file mode 100644
index 0000000000..ab1d0521e3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wikipedia.xml
@@ -0,0 +1,18 @@
+<!-- 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>Wikipedia</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="32" height="32"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://en.wikipedia.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://en.wikipedia.org/wiki/Special:Search">
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="sourceid" value="Mozilla-search"/>
+</Url>
+<SearchForm>https://en.wikipedia.org/wiki/Special:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-kn.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-kn.xml
new file mode 100644
index 0000000000..7e39c48ee2
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-kn.xml
@@ -0,0 +1,18 @@
+<!-- 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>ವಿಕ್ಷನರಿ (kn)</ShortName>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="get" template="https://kn.wiktionary.org/wiki/ವಿಶೇಷ:Search">
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<Url type="application/x-suggestions+json" method="GET" template="https://kn.wiktionary.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="format" value="xml"/>
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="namespace" value="0"/>
+</Url>
+<SearchForm>https://kn.wiktionary.org/wiki/ವಿಶೇಷ:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-oc.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-oc.xml
new file mode 100644
index 0000000000..82bd9724c1
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-oc.xml
@@ -0,0 +1,17 @@
+<!-- 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>Wikiccionari (oc)</ShortName>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://oc.wiktionary.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="namespace" value="0"/>
+</Url>
+<Url type="text/html" method="get" template="https://oc.wiktionary.org/wiki/Especial:Recèrca">
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://oc.wiktionary.org/wiki/Especial:Recèrca</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-or.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-or.xml
new file mode 100644
index 0000000000..bb475cafb5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-or.xml
@@ -0,0 +1,17 @@
+<!-- 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>Wiktionary (or)</ShortName>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://or.wiktionary.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="namespace" value="0"/>
+</Url>
+<Url type="text/html" method="get" template="https://or.wiktionary.org/wiki/ବିଶେଷ:ଖୋଜନ୍ତୁ">
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://or.wiktionary.org/wiki/ବିଶେଷ:ଖୋଜନ୍ତୁ</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-ta.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-ta.xml
new file mode 100644
index 0000000000..5b4662a29e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-ta.xml
@@ -0,0 +1,17 @@
+<!-- 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>விக்சனரி (ta)</ShortName>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://ta.wiktionary.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="namespace" value="0"/>
+</Url>
+<Url type="text/html" method="get" template="https://ta.wiktionary.org/wiki/சிறப்பு:Search">
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://ta.wiktionary.org/wiki/சிறப்பு:Search</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-te.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-te.xml
new file mode 100644
index 0000000000..1dd499dc37
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/wiktionary-te.xml
@@ -0,0 +1,17 @@
+<!-- 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>విక్షనరీ (te)</ShortName>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://te.wiktionary.org/w/api.php">
+ <Param name="action" value="opensearch"/>
+ <Param name="search" value="{searchTerms}"/>
+ <Param name="namespace" value="0"/>
+</Url>
+<Url type="text/html" method="get" template="https://te.wiktionary.org/wiki/ప్రత్యేక:అన్వేషణ">
+ <Param name="search" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://te.wiktionary.org/wiki/ప్రత్యేక:అన్వేషణ</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp-auctions.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp-auctions.xml
new file mode 100644
index 0000000000..6a3666954d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp-auctions.xml
@@ -0,0 +1,16 @@
+<!-- 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>Yahoo!オークション</ShortName>
+ <Description>ヤフオク! 検索</Description>
+ <InputEncoding>EUC-JP</InputEncoding>
+ <Image height="16" width="16"></Image>
+ <Url method="GET" template="https://auctions.yahoo.co.jp/search/search" type="text/html">
+ <Param name="p" value="{searchTerms}" />
+ <Param name="ei" value="EUC-JP" />
+ <Param name="fr" value="mozff" />
+ </Url>
+ <SearchForm>https://auctions.yahoo.co.jp/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp.xml
new file mode 100644
index 0000000000..2196d37b61
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yahoo-jp.xml
@@ -0,0 +1,16 @@
+<!-- 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>Yahoo! JAPAN</ShortName>
+<Description>検索エンジン - Yahoo!検索</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="text/html" method="GET" template="https://search.yahoo.co.jp/search">
+ <Param name="p" value="{searchTerms}"/>
+ <Param name="ei" value="UTF-8"/>
+ <Param name="fr" value="mozff" />
+</Url>
+<SearchForm>https://search.yahoo.co.jp/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-en.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-en.xml
new file mode 100644
index 0000000000..75ba859ca3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-en.xml
@@ -0,0 +1,22 @@
+<!-- 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>Yandex</ShortName>
+<Description>Use Yandex to search the Internet.</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="http://suggest.yandex.com/suggest-ff.cgi">
+ <Param name="part" value="{searchTerms}"/>
+ <Param name="uil" value="tr"/>
+</Url>
+<Url type="text/html" method="GET" template="https://yandex.com/search">
+ <Param name="clid" value="2186727"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://yandex.com/search">
+ <Param name="clid" value="2186733"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-ru.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-ru.xml
new file mode 100644
index 0000000000..68d5f6bbf8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-ru.xml
@@ -0,0 +1,21 @@
+<!-- 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>Яндекс</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://suggest.yandex.net/suggest-ff.cgi">
+ <Param name="part" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://yandex.ru/search">
+ <Param name="clid" value="2186727"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://yandex.ru/search">
+ <Param name="clid" value="2186733"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.yandex.ru/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-tr.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-tr.xml
new file mode 100644
index 0000000000..5d798c746c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex-tr.xml
@@ -0,0 +1,22 @@
+<!-- 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>Yandex</ShortName>
+<Description>Yandex Türkiye arama motoru</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="http://suggest.yandex.com.tr/suggest-ff.cgi">
+ <Param name="part" value="{searchTerms}"/>
+ <Param name="uil" value="tr"/>
+</Url>
+<Url type="text/html" method="GET" template="https://yandex.com.tr/search">
+ <Param name="clid" value="2186727"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://yandex.com.tr/search">
+ <Param name="clid" value="2186733"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.by.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.by.xml
new file mode 100644
index 0000000000..a4c6ea7a55
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.by.xml
@@ -0,0 +1,22 @@
+<!-- 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>Яндекс</ShortName>
+<Description>Пошук з дапамогаю Яндекс</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://suggest.yandex.by/suggest-ff.cgi">
+ <Param name="part" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://yandex.by/search">
+ <Param name="clid" value="2186727"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://yandex.by/search">
+ <Param name="clid" value="2186733"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.yandex.by/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.xml
new file mode 100644
index 0000000000..801ff5cbef
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/yandex.xml
@@ -0,0 +1,21 @@
+<!-- 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>Яндекс</ShortName>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET" template="https://suggest.yandex.net/suggest-ff.cgi">
+ <Param name="part" value="{searchTerms}"/>
+</Url>
+<Url type="text/html" method="GET" template="https://yandex.kz/search">
+ <Param name="clid" value="2186727"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<Url type="application/x-moz-tabletsearch" method="GET" template="https://yandex.kz/search">
+ <Param name="clid" value="2186733"/>
+ <Param name="text" value="{searchTerms}"/>
+</Url>
+<SearchForm>https://www.yandex.kz/</SearchForm>
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/youtube.xml b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/youtube.xml
new file mode 100644
index 0000000000..60a7897ae4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/assets/searchplugins/youtube.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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>YouTube</ShortName>
+ <Description>Search for videos on YouTube</Description>
+ <Tags>youtube video</Tags>
+ <Image height="16" width="16"></Image>
+ <Url type="text/html" template="https://www.youtube.com/results?search_query={searchTerms}&amp;page={startPage?}&amp;utm_source=opensearch" />
+ <Query role="example" searchTerms="cat" />
+</SearchPlugin>
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/BrowserStoreSearchAdapter.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/BrowserStoreSearchAdapter.kt
new file mode 100644
index 0000000000..ffc6179d84
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/BrowserStoreSearchAdapter.kt
@@ -0,0 +1,41 @@
+/* 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/. */
+
+package mozilla.components.feature.search
+
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.selector.findTabOrCustomTabOrSelectedTab
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.search.SearchRequest
+
+/**
+ * Adapter which wraps a [browserStore] in order to fulfill the [SearchAdapter] interface.
+ *
+ * This class uses the [browserStore] to determine when private mode is active, and updates the
+ * [browserStore] whenever a new search has been requested.
+ *
+ * NOTE: this will add [SearchRequest]s to [BrowserStore.state] when [sendSearch] is called. Client
+ * code is responsible for consuming these requests and displaying something to the user.
+ *
+ * NOTE: client code is responsible for sending [ContentAction.ConsumeSearchRequestAction]s
+ * after consuming events. See [SearchFeature] for a component that will handle this for you.
+ *
+ * @param browserStore The application's [BrowserStore].
+ * @param tabId ID of the tab that requests the search, or null to use the selected tab.
+ */
+class BrowserStoreSearchAdapter(
+ private val browserStore: BrowserStore,
+ private val tabId: String? = null,
+) : SearchAdapter {
+
+ override fun sendSearch(isPrivate: Boolean, text: String) {
+ val selectedTabId = tabId ?: browserStore.state.selectedTabId ?: return
+ browserStore.dispatch(
+ ContentAction.UpdateSearchRequestAction(selectedTabId, SearchRequest(isPrivate, text)),
+ )
+ }
+
+ override fun isPrivateSession(): Boolean =
+ browserStore.state.findTabOrCustomTabOrSelectedTab(tabId)?.content?.private ?: false
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchAdapter.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchAdapter.kt
new file mode 100644
index 0000000000..0d06911936
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchAdapter.kt
@@ -0,0 +1,21 @@
+/* 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/. */
+
+package mozilla.components.feature.search
+
+/**
+ * May be implemented by client code in order to allow a component to start searches.
+ */
+interface SearchAdapter {
+
+ /**
+ * Called by the component to indicate that the user should be shown a search.
+ */
+ fun sendSearch(isPrivate: Boolean, text: String)
+
+ /**
+ * Called by the component to check whether or not the currently selected session is private.
+ */
+ fun isPrivateSession(): Boolean
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchFeature.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchFeature.kt
new file mode 100644
index 0000000000..61d89a2844
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchFeature.kt
@@ -0,0 +1,52 @@
+/* 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/. */
+
+package mozilla.components.feature.search
+
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.cancel
+import kotlinx.coroutines.flow.distinctUntilChangedBy
+import kotlinx.coroutines.flow.map
+import kotlinx.coroutines.flow.mapNotNull
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.selector.findTabOrCustomTabOrSelectedTab
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.search.SearchRequest
+import mozilla.components.lib.state.ext.flowScoped
+import mozilla.components.support.base.feature.LifecycleAwareFeature
+import mozilla.components.support.utils.ext.toNullablePair
+
+/**
+ * Lifecycle aware feature that forwards [SearchRequest]s from the [store] to [performSearch], and
+ * then consumes them.
+ *
+ * NOTE: that this only consumes SearchRequests, and will not hook up the [store] to a source of
+ * SearchRequests. See [mozilla.components.concept.engine.selection.SelectionActionDelegate] for use
+ * in generating SearchRequests.
+ */
+class SearchFeature(
+ private val store: BrowserStore,
+ private val tabId: String? = null,
+ private val performSearch: (SearchRequest, tabId: String) -> Unit,
+) : LifecycleAwareFeature {
+
+ private var scope: CoroutineScope? = null
+
+ override fun start() {
+ scope = store.flowScoped { flow ->
+ flow.map { state -> state.findTabOrCustomTabOrSelectedTab(tabId) }
+ .distinctUntilChangedBy { it?.content?.searchRequest }
+ // Do nothing if searchRequest or sessionId is null
+ .mapNotNull { tab -> Pair(tab?.content?.searchRequest, tab?.id).toNullablePair() }
+ .collect { (searchRequest, sessionId) ->
+ performSearch(searchRequest, sessionId)
+ store.dispatch(ContentAction.ConsumeSearchRequestAction(sessionId))
+ }
+ }
+ }
+
+ override fun stop() {
+ scope?.cancel()
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchUseCases.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchUseCases.kt
new file mode 100644
index 0000000000..b3e8a1763b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchUseCases.kt
@@ -0,0 +1,343 @@
+/* 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/. */
+
+package mozilla.components.feature.search
+
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.action.EngineAction
+import mozilla.components.browser.state.action.SearchAction
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.selector.findTabOrCustomTab
+import mozilla.components.browser.state.state.SessionState
+import mozilla.components.browser.state.state.selectedOrDefaultSearchEngine
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.EngineSession
+import mozilla.components.feature.search.ext.buildSearchUrl
+import mozilla.components.feature.session.SessionUseCases
+import mozilla.components.feature.tabs.TabsUseCases
+import mozilla.components.support.base.log.logger.Logger
+
+/**
+ * Contains use cases related to the search feature.
+ */
+class SearchUseCases(
+ store: BrowserStore,
+ tabsUseCases: TabsUseCases,
+ sessionUseCases: SessionUseCases,
+) {
+ interface SearchUseCase {
+ /**
+ * Triggers a search.
+ */
+ fun invoke(
+ searchTerms: String,
+ searchEngine: SearchEngine? = null,
+ parentSessionId: String? = null,
+ )
+ }
+
+ class DefaultSearchUseCase(
+ private val store: BrowserStore,
+ private val tabsUseCases: TabsUseCases,
+ private val sessionUseCases: SessionUseCases,
+ ) : SearchUseCase {
+ private val logger = Logger("DefaultSearchUseCase")
+
+ /**
+ * Triggers a search in the currently selected session.
+ */
+ override fun invoke(
+ searchTerms: String,
+ searchEngine: SearchEngine?,
+ parentSessionId: String?,
+ ) {
+ invoke(searchTerms, store.state.selectedTabId, searchEngine)
+ }
+
+ /**
+ * Triggers a search using the default search engine for the provided search terms.
+ *
+ * @param searchTerms the search terms.
+ * @param sessionId the ID of the session/tab to use, or null if the currently selected tab
+ * should be used.
+ * @param searchEngine Search Engine to use, or the default search engine if none is provided
+ * @param flags Flags that will be used when loading the URL.
+ * @param additionalHeaders The extra headers to use when loading the URL.
+ */
+ operator fun invoke(
+ searchTerms: String,
+ sessionId: String? = store.state.selectedTabId,
+ searchEngine: SearchEngine? = null,
+ flags: EngineSession.LoadUrlFlags = EngineSession.LoadUrlFlags.none(),
+ additionalHeaders: Map<String, String>? = null,
+ ) {
+ val searchUrl = searchEngine?.let {
+ searchEngine.buildSearchUrl(searchTerms)
+ } ?: store.state.search.selectedOrDefaultSearchEngine?.buildSearchUrl(searchTerms)
+
+ if (searchUrl == null) {
+ logger.warn("No default search engine available to perform search")
+ return
+ }
+
+ val id = if (sessionId == null) {
+ // If no `sessionId` was passed in then create a new tab
+ tabsUseCases.addTab(
+ url = searchUrl,
+ flags = flags,
+ isSearch = true,
+ searchEngineName = searchEngine?.name,
+ additionalHeaders = additionalHeaders,
+ )
+ } else {
+ // If we got a `sessionId` then try to find the tab and load the search URL in it
+ val existingTab = store.state.findTabOrCustomTab(sessionId)
+ if (existingTab != null) {
+ store.dispatch(ContentAction.UpdateIsSearchAction(existingTab.id, true, searchEngine?.name))
+ store.dispatch(
+ EngineAction.LoadUrlAction(
+ tabId = existingTab.id,
+ url = searchUrl,
+ flags = flags,
+ additionalHeaders = additionalHeaders,
+ ),
+ )
+ existingTab.id
+ } else {
+ // If the tab with the provided id was not found then create a new tab
+ tabsUseCases.addTab(
+ url = searchUrl,
+ isSearch = true,
+ searchEngineName = searchEngine?.name,
+ flags = flags,
+ additionalHeaders = additionalHeaders,
+ )
+ }
+ }
+
+ store.dispatch(ContentAction.UpdateSearchTermsAction(id, searchTerms))
+ }
+ }
+
+ class NewTabSearchUseCase(
+ private val store: BrowserStore,
+ private val tabsUseCases: TabsUseCases,
+ private val isPrivate: Boolean,
+ ) : SearchUseCase {
+ private val logger = Logger("NewTabSearchUseCase")
+
+ override fun invoke(
+ searchTerms: String,
+ searchEngine: SearchEngine?,
+ parentSessionId: String?,
+ ) {
+ invoke(
+ searchTerms,
+ source = SessionState.Source.Internal.None,
+ selected = true,
+ searchEngine = searchEngine,
+ parentSessionId = parentSessionId,
+ )
+ }
+
+ /**
+ * Triggers a search on a new session, using the default search engine for the provided search terms.
+ *
+ * @param searchTerms the search terms.
+ * @param selected whether or not the new session should be selected, defaults to true.
+ * @param source the source of the new session.
+ * @param searchEngine Search Engine to use, or the default search engine if none is provided
+ * @param parentSessionId optional parent session to attach this new search session to
+ * @param flags Flags that will be used when loading the URL.
+ * @param additionalHeaders The extra headers to use when loading the URL.
+ */
+ operator fun invoke(
+ searchTerms: String,
+ source: SessionState.Source,
+ selected: Boolean = true,
+ searchEngine: SearchEngine? = null,
+ parentSessionId: String? = null,
+ flags: EngineSession.LoadUrlFlags = EngineSession.LoadUrlFlags.none(),
+ additionalHeaders: Map<String, String>? = null,
+ ) {
+ val searchUrl = searchEngine?.let {
+ searchEngine.buildSearchUrl(searchTerms)
+ } ?: store.state.search.selectedOrDefaultSearchEngine?.buildSearchUrl(searchTerms)
+
+ if (searchUrl == null) {
+ logger.warn("No default search engine available to perform search")
+ return
+ }
+
+ val id = tabsUseCases.addTab(
+ url = searchUrl,
+ parentId = parentSessionId,
+ flags = flags,
+ source = source,
+ selectTab = selected,
+ private = isPrivate,
+ isSearch = true,
+ additionalHeaders = additionalHeaders,
+ )
+
+ store.dispatch(ContentAction.UpdateSearchTermsAction(id, searchTerms))
+ }
+ }
+
+ /**
+ * Adds a new search engine to the list of search engines the user can use for searches.
+ */
+ class AddNewSearchEngineUseCase(
+ private val store: BrowserStore,
+ ) {
+ /**
+ * Adds the given [searchEngine] to the list of search engines the user can use for searches.
+ */
+ operator fun invoke(
+ searchEngine: SearchEngine,
+ ) {
+ when (searchEngine.type) {
+ SearchEngine.Type.BUNDLED -> store.dispatch(
+ SearchAction.ShowSearchEngineAction(searchEngine.id),
+ )
+
+ SearchEngine.Type.BUNDLED_ADDITIONAL -> store.dispatch(
+ SearchAction.AddAdditionalSearchEngineAction(searchEngine.id),
+ )
+
+ SearchEngine.Type.CUSTOM -> store.dispatch(
+ SearchAction.UpdateCustomSearchEngineAction(searchEngine),
+ )
+
+ SearchEngine.Type.APPLICATION -> { /* Do nothing */ }
+ }
+ }
+ }
+
+ /**
+ * Removes a search engine from the list of search engines the user can use for searches.
+ */
+ class RemoveExistingSearchEngineUseCase(
+ private val store: BrowserStore,
+ ) {
+ /**
+ * Removes the given [searchEngine] from the list of search engines the user can use for
+ * searches.
+ */
+ operator fun invoke(
+ searchEngine: SearchEngine,
+ ) {
+ when (searchEngine.type) {
+ SearchEngine.Type.BUNDLED -> store.dispatch(
+ SearchAction.HideSearchEngineAction(searchEngine.id),
+ )
+
+ SearchEngine.Type.BUNDLED_ADDITIONAL -> store.dispatch(
+ SearchAction.RemoveAdditionalSearchEngineAction(searchEngine.id),
+ )
+
+ SearchEngine.Type.CUSTOM -> store.dispatch(
+ SearchAction.RemoveCustomSearchEngineAction(searchEngine.id),
+ )
+
+ SearchEngine.Type.APPLICATION -> { /* Do nothing */ }
+ }
+ }
+ }
+
+ /**
+ * Marks a search engine as "selected" by the user to be the default search engine to perform
+ * searches with.
+ */
+ class SelectSearchEngineUseCase(
+ private val store: BrowserStore,
+ ) {
+ /**
+ * Marks the given [searchEngine] as "selected" by the user to be the default search engine
+ * to perform searches with.
+ */
+ operator fun invoke(
+ searchEngine: SearchEngine,
+ ) {
+ val name = if (searchEngine.type == SearchEngine.Type.BUNDLED) {
+ // For bundled search engines we additionally save the name of the search engine.
+ // We do this because with "home" region changes the previous search plugin/id
+ // may no longer be available, but there may be a clone of the search engine with
+ // a different plugin/id using the same name.
+ // This should be safe to do since Fenix as well as Fennec only kept the name of
+ // the default search engine.
+ // For all other cases (e.g. custom search engines) we only care about the ID and
+ // do not want to switch to a different search engine based on its name once it is
+ // gone.
+ searchEngine.name
+ } else {
+ null
+ }
+
+ store.dispatch(
+ SearchAction.SelectSearchEngineAction(searchEngine.id, name),
+ )
+ }
+ }
+
+ /**
+ * Updates the list of unselected shortcuts, to be hidden from the quick search menus.
+ */
+ class UpdateDisabledSearchEngineIdsUseCase(private val store: BrowserStore) {
+ /**
+ * Updates the list of unselected shortcuts with the given [searchEngineId], to be hidden from
+ * the quick search menus.
+ */
+ operator fun invoke(
+ searchEngineId: String,
+ isEnabled: Boolean,
+ ) {
+ store.dispatch(SearchAction.UpdateDisabledSearchEngineIdsAction(searchEngineId, isEnabled))
+ }
+ }
+
+ /**
+ * Restores bundled search engines that may have been removed.
+ */
+ class RestoreHiddenSearchEnginesUseCase(private val store: BrowserStore) {
+ /**
+ * Restores all hidden engines back to the bundled engine list.
+ */
+ operator fun invoke() {
+ store.dispatch(SearchAction.RestoreHiddenSearchEnginesAction)
+ }
+ }
+
+ val defaultSearch: DefaultSearchUseCase by lazy {
+ DefaultSearchUseCase(store, tabsUseCases, sessionUseCases)
+ }
+
+ val newTabSearch: NewTabSearchUseCase by lazy {
+ NewTabSearchUseCase(store, tabsUseCases, false)
+ }
+
+ val newPrivateTabSearch: NewTabSearchUseCase by lazy {
+ NewTabSearchUseCase(store, tabsUseCases, true)
+ }
+
+ val addSearchEngine: AddNewSearchEngineUseCase by lazy {
+ AddNewSearchEngineUseCase(store)
+ }
+
+ val removeSearchEngine: RemoveExistingSearchEngineUseCase by lazy {
+ RemoveExistingSearchEngineUseCase(store)
+ }
+
+ val selectSearchEngine: SelectSearchEngineUseCase by lazy {
+ SelectSearchEngineUseCase(store)
+ }
+
+ val updateDisabledSearchEngineIds: UpdateDisabledSearchEngineIdsUseCase by lazy {
+ UpdateDisabledSearchEngineIdsUseCase(store)
+ }
+
+ val restoreHiddenSearchEngines: RestoreHiddenSearchEnginesUseCase by lazy {
+ RestoreHiddenSearchEnginesUseCase(store)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/BrowserStore.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/BrowserStore.kt
new file mode 100644
index 0000000000..2fce10dfe8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/BrowserStore.kt
@@ -0,0 +1,36 @@
+/* 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/. */
+
+package mozilla.components.feature.search.ext
+
+import mozilla.components.browser.state.action.BrowserAction
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.selectedOrDefaultSearchEngine
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.lib.state.Store
+
+/**
+ * Waits (asynchronously, non-blocking) for the search state to be loaded from disk (when using
+ * `RegionMiddleware` and `SearchMiddleware`) and invokes [block] with the default search engine
+ * (or `null` if no default could be loaded).
+ */
+fun BrowserStore.waitForSelectedOrDefaultSearchEngine(
+ block: (mozilla.components.browser.state.search.SearchEngine?) -> Unit,
+) {
+ // Did we already load the search state? In that case we can invoke `block` immediately.
+ if (state.search.complete) {
+ block(state.search.selectedOrDefaultSearchEngine)
+ return
+ }
+
+ // Otherwise: Wait for the search state to be loaded and then invoke `block`.
+ var subscription: Store.Subscription<BrowserState, BrowserAction>? = null
+ subscription = observeManually { state ->
+ if (state.search.complete) {
+ block(state.search.selectedOrDefaultSearchEngine)
+ subscription!!.unsubscribe()
+ }
+ }
+ subscription.resume()
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/SearchEngine.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/SearchEngine.kt
new file mode 100644
index 0000000000..08615935f4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/SearchEngine.kt
@@ -0,0 +1,148 @@
+/* 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/. */
+
+package mozilla.components.feature.search.ext
+
+import android.graphics.Bitmap
+import android.net.Uri
+import androidx.annotation.VisibleForTesting
+import mozilla.components.browser.state.search.OS_SEARCH_ENGINE_TERMS_PARAM
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.state.SearchState
+import mozilla.components.browser.state.state.searchEngines
+import mozilla.components.browser.state.state.selectedOrDefaultSearchEngine
+import mozilla.components.feature.search.internal.SearchUrlBuilder
+import mozilla.components.feature.search.storage.SearchEngineReader
+import java.io.InputStream
+import java.lang.IllegalArgumentException
+import java.util.UUID
+
+/**
+ * Creates a custom [SearchEngine].
+ */
+fun createSearchEngine(
+ name: String,
+ url: String,
+ icon: Bitmap,
+ inputEncoding: String? = null,
+ suggestUrl: String? = null,
+ isGeneral: Boolean = false,
+): SearchEngine {
+ if (!url.contains(OS_SEARCH_ENGINE_TERMS_PARAM)) {
+ throw IllegalArgumentException("URL does not contain search terms placeholder")
+ }
+
+ return SearchEngine(
+ id = UUID.randomUUID().toString(),
+ name = name,
+ icon = icon,
+ inputEncoding = inputEncoding,
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf(url),
+ suggestUrl = suggestUrl,
+ isGeneral = isGeneral,
+ )
+}
+
+/**
+ * Creates an application [SearchEngine].
+ */
+fun createApplicationSearchEngine(
+ id: String? = null,
+ name: String,
+ url: String,
+ inputEncoding: String? = null,
+ icon: Bitmap,
+ suggestUrl: String? = null,
+): SearchEngine {
+ return SearchEngine(
+ id = id ?: UUID.randomUUID().toString(),
+ name = name,
+ icon = icon,
+ inputEncoding = inputEncoding,
+ type = SearchEngine.Type.APPLICATION,
+ resultUrls = listOf(url),
+ suggestUrl = suggestUrl,
+ )
+}
+
+/**
+ * Whether this [SearchEngine] has a [SearchEngine.suggestUrl] set and can provide search
+ * suggestions.
+ */
+val SearchEngine.canProvideSearchSuggestions: Boolean
+ get() = suggestUrl != null
+
+/**
+ * Creates an URL to retrieve search suggestions for the provided [query].
+ */
+fun SearchEngine.buildSuggestionsURL(query: String): String? {
+ val builder = SearchUrlBuilder(this)
+ return builder.buildSuggestionUrl(query)
+}
+
+/**
+ * Builds a URL to search for the given search terms with this search engine.
+ */
+fun SearchEngine.buildSearchUrl(searchTerm: String): String {
+ val builder = SearchUrlBuilder(this)
+ return builder.buildSearchUrl(searchTerm)
+}
+
+/**
+ * Parses a [SearchEngine] from the given [stream].
+ */
+@Deprecated("Only for migrating legacy search engines. Will eventually be removed.")
+fun parseLegacySearchEngine(id: String, stream: InputStream): SearchEngine {
+ val reader = SearchEngineReader(SearchEngine.Type.CUSTOM)
+ return reader.loadStream(id, stream)
+}
+
+/**
+ * Given a [SearchState], determine if the passed-in [url] is a known search results page url
+ * and what are the associated search terms.
+ * @return Search terms if [url] is a known search results page, `null` otherwise.
+ */
+fun SearchState.parseSearchTerms(url: String): String? {
+ val parsedUrl = Uri.parse(url)
+ // Default/selected engine is the most likely to match, check it first.
+ val currentEngine = this.selectedOrDefaultSearchEngine
+ // Or go through the rest of known engines.
+ val fallback: () -> String? = fallback@{
+ this.searchEngines.forEach { searchEngine ->
+ searchEngine.parseSearchTerms(parsedUrl)?.let { return@fallback it }
+ }
+ return@fallback null
+ }
+ return currentEngine?.parseSearchTerms(parsedUrl) ?: fallback()
+}
+
+/**
+ * Given a [SearchEngine], determine if the passed-in [url] matches its results template,
+ * and what are the associated search terms.
+ * @return Search terms if [url] matches the results page template, `null` otherwise.
+ */
+@VisibleForTesting
+fun SearchEngine.parseSearchTerms(url: Uri): String? {
+ // Basic approach:
+ // - look at the "base" of the template url; if there's a match, continue
+ // - see if the GET parameter for the search terms is present in the url
+ // - if that param present, its value is our answer if it's non-empty
+ val searchResultsRoot = this.resultsUrl.authority + this.resultsUrl.path
+ val urlRoot = url.authority + url.path
+
+ return if (searchResultsRoot == urlRoot) {
+ val searchTerms = try {
+ this.searchParameterName?.let {
+ url.getQueryParameter(it)
+ }
+ } catch (e: UnsupportedOperationException) {
+ // Non-hierarchical url.
+ null
+ }
+ searchTerms.takeUnless { it.isNullOrEmpty() }
+ } else {
+ null
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/internal/SearchUrlBuilder.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/internal/SearchUrlBuilder.kt
new file mode 100644
index 0000000000..9fc9f61dd5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/internal/SearchUrlBuilder.kt
@@ -0,0 +1,93 @@
+/* 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/. */
+
+package mozilla.components.feature.search.internal
+
+import android.net.Uri
+import android.text.TextUtils
+import mozilla.components.browser.state.search.OS_SEARCH_ENGINE_TERMS_PARAM
+import mozilla.components.browser.state.search.SearchEngine
+import java.io.UnsupportedEncodingException
+import java.net.URLEncoder
+import java.util.Locale
+
+// We are using string concatenation here to avoid the Kotlin compiler interpreting this
+// as string templates. It is possible to escape the string accordingly. But this seems to
+// be inconsistent between Kotlin versions. So to be safe we avoid this completely by
+// constructing the strings manually.
+
+// Parameters copied from nsSearchService.js
+private const val MOZ_PARAM_LOCALE = "{" + "moz:locale" + "}"
+private const val MOZ_PARAM_DIST_ID = "{" + "moz:distributionID" + "}"
+private const val MOZ_PARAM_OFFICIAL = "{" + "moz:official" + "}"
+
+// Supported OpenSearch parameters
+// See http://opensearch.a9.com/spec/1.1/querysyntax/#core
+private const val OS_PARAM_USER_DEFINED = OS_SEARCH_ENGINE_TERMS_PARAM
+private const val OS_PARAM_INPUT_ENCODING = "{" + "inputEncoding" + "}"
+private const val OS_PARAM_LANGUAGE = "{" + "language" + "}"
+private const val OS_PARAM_OUTPUT_ENCODING = "{" + "outputEncoding" + "}"
+private const val OS_PARAM_OPTIONAL = "\\{" + "(?:\\w+:)?\\w+?" + "\\}"
+
+internal class SearchUrlBuilder(
+ private val searchEngine: SearchEngine,
+) {
+ fun buildSearchUrl(searchTerms: String): String {
+ // The parser should have put the best URL for this device at the beginning of the list.
+ val template = searchEngine.resultUrls[0]
+ return buildUrl(template, searchTerms)
+ }
+
+ fun buildSuggestionUrl(searchTerms: String): String? {
+ val template = searchEngine.suggestUrl ?: return null
+ return buildUrl(template, searchTerms)
+ }
+
+ private fun buildUrl(template: String, searchTerms: String): String {
+ val templateUri = Uri.decode(template)
+ val inputEncoding = searchEngine.inputEncoding ?: "UTF-8"
+ val query = try {
+ // Although android.net.Uri.encode convert space (U+0x20) to "%20", java.net.URLEncoder convert it to "+".
+ URLEncoder.encode(searchTerms, inputEncoding).replace("+", "%20")
+ } catch (e: UnsupportedEncodingException) {
+ Uri.encode(searchTerms)
+ }
+ val urlWithSubstitutions = paramSubstitution(templateUri, query, inputEncoding)
+ return normalize(urlWithSubstitutions) // User-entered search engines may need normalization.
+ }
+}
+
+/**
+ * Formats template string with proper parameters. Modeled after ParamSubstitution in nsSearchService.js
+ */
+private fun paramSubstitution(template: String, query: String, inputEncoding: String): String {
+ var result = template
+ val locale = Locale.getDefault().toString()
+
+ result = result.replace(MOZ_PARAM_LOCALE, locale)
+ result = result.replace(MOZ_PARAM_DIST_ID, "")
+ result = result.replace(MOZ_PARAM_OFFICIAL, "unofficial")
+
+ result = result.replace(OS_PARAM_USER_DEFINED, query)
+ result = result.replace(OS_PARAM_INPUT_ENCODING, inputEncoding)
+
+ result = result.replace(OS_PARAM_LANGUAGE, locale)
+ result = result.replace(OS_PARAM_OUTPUT_ENCODING, "UTF-8")
+
+ // Replace any optional parameters
+ result = result.replace(OS_PARAM_OPTIONAL.toRegex(), "")
+
+ return result
+}
+
+private fun normalize(input: String): String {
+ val trimmedInput = input.trim { it <= ' ' }
+ var uri = Uri.parse(trimmedInput)
+
+ if (TextUtils.isEmpty(uri.scheme)) {
+ uri = Uri.parse("http://$trimmedInput")
+ }
+
+ return uri.toString()
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddleware.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddleware.kt
new file mode 100644
index 0000000000..5139af8740
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddleware.kt
@@ -0,0 +1,77 @@
+/* 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/. */
+
+package mozilla.components.feature.search.middleware
+
+import androidx.annotation.VisibleForTesting
+import mozilla.components.browser.state.action.BrowserAction
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.selector.findTab
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry
+import mozilla.components.lib.state.Middleware
+import mozilla.components.lib.state.MiddlewareContext
+import mozilla.components.support.base.log.logger.Logger
+
+/**
+ * [BrowserStore] middleware to be used alongside with [AdsTelemetry] to check when an ad shown
+ * in search results is clicked.
+ */
+class AdsTelemetryMiddleware(
+ private val adsTelemetry: AdsTelemetry,
+) : Middleware<BrowserState, BrowserAction> {
+ @VisibleForTesting
+ internal val redirectChain = mutableMapOf<String, RedirectChain>()
+ private val logger = Logger("AdsTelemetryMiddleware")
+
+ @Suppress("TooGenericExceptionCaught")
+ override fun invoke(
+ context: MiddlewareContext<BrowserState, BrowserAction>,
+ next: (BrowserAction) -> Unit,
+ action: BrowserAction,
+ ) {
+ when (action) {
+ is ContentAction.UpdateLoadRequestAction -> {
+ context.state.findTab(action.sessionId)?.let { tab ->
+ // Collect all load requests in between location changes
+ if (!redirectChain.containsKey(action.sessionId) && action.loadRequest.url != tab.content.url) {
+ redirectChain[action.sessionId] = RedirectChain(tab.content.url)
+ }
+
+ redirectChain[action.sessionId]?.add(action.loadRequest.url)
+ }
+ }
+ is ContentAction.UpdateUrlAction -> {
+ redirectChain[action.sessionId]?.let {
+ // Record ads telemetry providing all redirects
+ try {
+ adsTelemetry.checkIfAddWasClicked(it.root, it.chain)
+ } catch (t: Throwable) {
+ logger.info("Failed to record search telemetry", t)
+ } finally {
+ redirectChain.remove(action.sessionId)
+ }
+ }
+ }
+ else -> {
+ // no-op
+ }
+ }
+
+ next(action)
+ }
+}
+
+/**
+ * Utility to collect URLs / load requests in between location changes.
+ */
+@VisibleForTesting
+internal class RedirectChain(val root: String) {
+ val chain = mutableListOf<String>()
+
+ fun add(url: String) {
+ chain.add(url)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/SearchMiddleware.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/SearchMiddleware.kt
new file mode 100644
index 0000000000..c748ce9c21
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/middleware/SearchMiddleware.kt
@@ -0,0 +1,397 @@
+/* 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/. */
+
+package mozilla.components.feature.search.middleware
+
+import android.content.Context
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.Dispatchers
+import kotlinx.coroutines.async
+import kotlinx.coroutines.launch
+import mozilla.components.browser.state.action.BrowserAction
+import mozilla.components.browser.state.action.SearchAction
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.SearchState
+import mozilla.components.feature.search.storage.BundledSearchEnginesStorage
+import mozilla.components.feature.search.storage.CustomSearchEngineStorage
+import mozilla.components.feature.search.storage.SearchMetadataStorage
+import mozilla.components.lib.state.Middleware
+import mozilla.components.lib.state.MiddlewareContext
+import mozilla.components.lib.state.Store
+import mozilla.components.support.base.log.logger.Logger
+import java.util.Locale
+import kotlin.coroutines.CoroutineContext
+
+/**
+ * Holds data for the search extra params.
+ */
+data class SearchExtraParams(
+ val searchEngineName: String,
+ val featureEnablerName: String?,
+ val featureEnablerParam: String?,
+ val channelIdName: String,
+ val channelIdParam: String,
+)
+
+/**
+ * [Middleware] implementation for loading and saving [SearchEngine]s whenever the state changes.
+ *
+ * @param additionalBundledSearchEngineIds List of (bundled) search engine IDs that will be loaded
+ * in addition to the search engines for the user's region and made available through
+ * [SearchState.additionalSearchEngines] and [SearchState.additionalSearchEngines].
+ * @param migration Interface for a class that can provide data from a legacy system to be imported into the
+ * storage used by the middleware.
+ * @param customStorage A storage for custom search engines of the user.
+ * @param bundleStorage A storage for loading bundled search engines.
+ * @param metadataStorage A storage for saving additional metadata related to search.
+ * @param searchExtraParams Optional search extra params.
+ * @param ioDispatcher The coroutine dispatcher to be used when loading.
+ */
+class SearchMiddleware(
+ context: Context,
+ private val additionalBundledSearchEngineIds: List<String> = emptyList(),
+ private val migration: Migration? = null,
+ private val customStorage: CustomStorage = CustomSearchEngineStorage(context),
+ private val bundleStorage: BundleStorage = BundledSearchEnginesStorage(context),
+ private val metadataStorage: MetadataStorage = SearchMetadataStorage(
+ context,
+ additionalBundledSearchEngineIds.toSet(),
+ ),
+ private val searchExtraParams: SearchExtraParams? = null,
+ private val ioDispatcher: CoroutineContext = Dispatchers.IO,
+) : Middleware<BrowserState, BrowserAction> {
+ private val logger = Logger("SearchMiddleware")
+ private val scope = CoroutineScope(ioDispatcher)
+
+ override fun invoke(
+ context: MiddlewareContext<BrowserState, BrowserAction>,
+ next: (BrowserAction) -> Unit,
+ action: BrowserAction,
+ ) {
+ when (action) {
+ is SearchAction.SetRegionAction -> loadSearchEngines(context.store, action.regionState, action.distribution)
+ is SearchAction.UpdateCustomSearchEngineAction -> saveCustomSearchEngine(action)
+ is SearchAction.RemoveCustomSearchEngineAction -> removeCustomSearchEngine(action)
+ is SearchAction.SelectSearchEngineAction -> updateSearchEngineSelection(action)
+ else -> {
+ // no-op
+ }
+ }
+
+ next(action)
+
+ when (action) {
+ is SearchAction.ShowSearchEngineAction, is SearchAction.HideSearchEngineAction,
+ is SearchAction.RestoreHiddenSearchEnginesAction,
+ -> updateHiddenSearchEngines(context.state.search.hiddenSearchEngines)
+ is SearchAction.AddAdditionalSearchEngineAction, is SearchAction.RemoveAdditionalSearchEngineAction ->
+ updateAdditionalSearchEngines(context.state.search.additionalSearchEngines)
+ is SearchAction.UpdateDisabledSearchEngineIdsAction -> updateDisabledSearchEngineIds(
+ context.store,
+ action,
+ )
+ else -> {
+ // no-op
+ }
+ }
+ }
+
+ private fun loadSearchEngines(
+ store: Store<BrowserState, BrowserAction>,
+ region: RegionState,
+ distribution: String? = null,
+ ) = scope.launch {
+ val migrationValues = migration?.getValuesToMigrate()
+ performCustomSearchEnginesMigration(migrationValues)
+
+ val regionBundle = async(ioDispatcher) {
+ bundleStorage.load(
+ region = region,
+ distribution = distribution,
+ searchExtraParams = searchExtraParams,
+ coroutineContext = ioDispatcher,
+ )
+ }
+ val customSearchEngines = async(ioDispatcher) { customStorage.loadSearchEngineList() }
+ val hiddenSearchEngineIds = async(ioDispatcher) { metadataStorage.getHiddenSearchEngines() }
+ val disabledSearchEngineIds = async(ioDispatcher) { metadataStorage.getDisabledSearchEngineIds() }
+ val additionalSearchEngineIds = async(ioDispatcher) { metadataStorage.getAdditionalSearchEngines() }
+ val allAdditionalSearchEngines = async(ioDispatcher) {
+ bundleStorage.load(
+ ids = additionalBundledSearchEngineIds,
+ searchExtraParams = searchExtraParams,
+ coroutineContext = ioDispatcher,
+ )
+ }
+
+ val hiddenSearchEngines = mutableListOf<SearchEngine>()
+ val filteredRegionSearchEngines = regionBundle.await().list.filter { searchEngine ->
+ if (hiddenSearchEngineIds.await().contains(searchEngine.id)) {
+ hiddenSearchEngines.add(searchEngine)
+ false
+ } else {
+ true
+ }
+ }
+
+ val regionSearchEngineIds = regionBundle.await().list.map { searchEngine -> searchEngine.id }
+
+ val additionalSearchEngines = allAdditionalSearchEngines.await().filter { searchEngine ->
+ searchEngine.id in additionalSearchEngineIds.await() &&
+ searchEngine.id !in regionSearchEngineIds
+ }
+
+ val additionalAvailableSearchEngines = allAdditionalSearchEngines.await().filter { searchEngine ->
+ searchEngine.id !in additionalSearchEngineIds.await() &&
+ searchEngine.id !in regionSearchEngineIds
+ }
+
+ performDefaultSearchEngineMigration(
+ migrationValues,
+ filteredRegionSearchEngines + customSearchEngines.await() + additionalSearchEngines,
+ )
+ val userChoice = async(ioDispatcher) { metadataStorage.getUserSelectedSearchEngine() }
+
+ val action = SearchAction.SetSearchEnginesAction(
+ regionSearchEngines = filteredRegionSearchEngines,
+ regionDefaultSearchEngineId = regionBundle.await().defaultSearchEngineId,
+ userSelectedSearchEngineId = userChoice.await()?.searchEngineId,
+ userSelectedSearchEngineName = userChoice.await()?.searchEngineName,
+ customSearchEngines = customSearchEngines.await(),
+ hiddenSearchEngines = hiddenSearchEngines,
+ disabledSearchEngineIds = disabledSearchEngineIds.await(),
+ additionalSearchEngines = additionalSearchEngines,
+ additionalAvailableSearchEngines = additionalAvailableSearchEngines,
+ regionSearchEnginesOrder = regionSearchEngineIds,
+ )
+ store.dispatch(action)
+ }
+
+ private fun updateSearchEngineSelection(
+ action: SearchAction.SelectSearchEngineAction,
+ ) = scope.launch {
+ metadataStorage.setUserSelectedSearchEngine(
+ action.searchEngineId,
+ action.searchEngineName,
+ )
+ }
+
+ private fun removeCustomSearchEngine(
+ action: SearchAction.RemoveCustomSearchEngineAction,
+ ) = scope.launch {
+ customStorage.removeSearchEngine(action.searchEngineId)
+ }
+
+ private fun saveCustomSearchEngine(
+ action: SearchAction.UpdateCustomSearchEngineAction,
+ ) = scope.launch {
+ customStorage.saveSearchEngine(action.searchEngine)
+ }
+
+ private fun updateHiddenSearchEngines(
+ hiddenSearchEngines: List<SearchEngine>,
+ ) = scope.launch {
+ metadataStorage.setHiddenSearchEngines(
+ hiddenSearchEngines.map { searchEngine -> searchEngine.id },
+ )
+ }
+
+ private fun updateAdditionalSearchEngines(
+ additionalSearchEngines: List<SearchEngine>,
+ ) = scope.launch {
+ metadataStorage.setAdditionalSearchEngines(
+ additionalSearchEngines.map { searchEngine -> searchEngine.id },
+ )
+ }
+
+ private fun updateDisabledSearchEngineIds(
+ store: Store<BrowserState, BrowserAction>,
+ action: SearchAction.UpdateDisabledSearchEngineIdsAction,
+ ) = scope.launch {
+ val disabledIds = store.state.search.disabledSearchEngineIds
+ val updatedList = if (action.isEnabled) {
+ disabledIds - action.searchEngineId
+ } else {
+ disabledIds + action.searchEngineId
+ }
+ metadataStorage.setDisabledSearchEngineIds(updatedList)
+ }
+
+ private suspend fun performCustomSearchEnginesMigration(values: Migration.MigrationValues?) {
+ if (values == null) {
+ return
+ }
+
+ values.customSearchEngines.forEach { searchEngine ->
+ customStorage.saveSearchEngine(searchEngine)
+ }
+ }
+
+ private suspend fun performDefaultSearchEngineMigration(
+ values: Migration.MigrationValues?,
+ engines: List<SearchEngine>,
+ ) {
+ if (values == null) {
+ return
+ }
+
+ val name = values.defaultSearchEngineName ?: return
+
+ val default = engines.find { searchEngine ->
+ searchEngine.name == name
+ }
+
+ if (default == null) {
+ logger.error("Could not find migrated default search engine ($name)")
+ return
+ }
+
+ metadataStorage.setUserSelectedSearchEngine(
+ id = default.id,
+ name = if (default.type == SearchEngine.Type.BUNDLED) {
+ default.name
+ } else {
+ null
+ },
+ )
+ }
+
+ /**
+ * A storage for custom search engines of the user.
+ */
+ interface CustomStorage {
+ /**
+ * Loads the list of search engines from the storage.
+ */
+ suspend fun loadSearchEngineList(): List<SearchEngine>
+
+ /**
+ * Removes the search engine with the specified [identifier] from the storage.
+ */
+ suspend fun removeSearchEngine(identifier: String)
+
+ /**
+ * Saves the given [searchEngine] to the storage. May replace an already existing search
+ * engine with the same ID.
+ */
+ suspend fun saveSearchEngine(searchEngine: SearchEngine): Boolean
+ }
+
+ /**
+ * A storage for loading bundled search engines.
+ */
+ interface BundleStorage {
+ /**
+ * Loads the bundled search engines for the given [locale] and [region].
+ *
+ * If [distribution] is not null then attempt to load the bundled search engine for the
+ * [distribution] in the specified [locale] and [region] if available.
+ */
+ suspend fun load(
+ region: RegionState,
+ locale: Locale = Locale.getDefault(),
+ distribution: String? = null,
+ searchExtraParams: SearchExtraParams? = null,
+ coroutineContext: CoroutineContext = Dispatchers.IO,
+ ): Bundle
+
+ /**
+ * Loads the bundled search engines with the given [ids].
+ */
+ suspend fun load(
+ ids: List<String>,
+ searchExtraParams: SearchExtraParams? = null,
+ coroutineContext: CoroutineContext = Dispatchers.IO,
+ ): List<SearchEngine>
+
+ /**
+ * A loaded bundle containing the list of search engines and the ID of the default for
+ * the region.
+ */
+ data class Bundle(
+ val list: List<SearchEngine>,
+ val defaultSearchEngineId: String,
+ )
+ }
+
+ /**
+ * A storage for saving additional metadata related to search.
+ */
+ interface MetadataStorage {
+ /**
+ * Gets the ID (and optinally name) of the default search engine the user has picked. Returns
+ * `null` if the user has not made a choice.
+ */
+ suspend fun getUserSelectedSearchEngine(): UserChoice?
+
+ /**
+ * Sets the ID (and optionally name) of the default search engine the user has picked.
+ */
+ suspend fun setUserSelectedSearchEngine(id: String, name: String?)
+
+ /**
+ * Sets the list of IDs of hidden search engines.
+ */
+ suspend fun setHiddenSearchEngines(ids: List<String>)
+
+ /**
+ * Gets the list of IDs of hidden search engines.
+ */
+ suspend fun getHiddenSearchEngines(): List<String>
+
+ /**
+ * Gets the list of IDs of additional search engines that the user explicitly added.
+ */
+ suspend fun getAdditionalSearchEngines(): List<String>
+
+ /**
+ * Sets the list of IDs of additional search engines that the user explicitly added.
+ */
+ suspend fun setAdditionalSearchEngines(ids: List<String>)
+
+ /**
+ * Gets the list of IDs of disabled search engine shortcuts.
+ */
+ suspend fun getDisabledSearchEngineIds(): List<String>
+
+ /**
+ * Sets the list of IDs of disabled search engine shortcuts.
+ */
+ suspend fun setDisabledSearchEngineIds(ids: List<String>)
+
+ /**
+ * Data class holding the ID and name of the selected search engine of the user.
+ */
+ data class UserChoice(
+ val searchEngineId: String,
+ val searchEngineName: String?,
+ )
+ }
+
+ /**
+ * Interface for a class that can provide data from a legacy system to be imported into the
+ * storage used by the middleware.
+ */
+ interface Migration {
+ /**
+ * Returns the values to be migrated. It is expected that the application returns the values
+ * only once. Afterwards the data is assumed to be migrated and should not be provided again.
+ */
+ fun getValuesToMigrate(): MigrationValues?
+
+ /**
+ * Holder data class for values to be migrated.
+ *
+ * @param customSearchEngines List of custom search engines that should be imported.
+ * @param defaultSearchEngineName Name of the default search engine that the user had
+ * selected. Or `null` if the user has not made any choice.
+ */
+ data class MigrationValues(
+ val customSearchEngines: List<SearchEngine>,
+ val defaultSearchEngineName: String?,
+ )
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionManager.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionManager.kt
new file mode 100644
index 0000000000..33f75a8a17
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionManager.kt
@@ -0,0 +1,126 @@
+/* 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/. */
+
+package mozilla.components.feature.search.region
+
+import android.content.Context
+import android.content.SharedPreferences
+import kotlinx.coroutines.CoroutineDispatcher
+import kotlinx.coroutines.Dispatchers
+import kotlinx.coroutines.delay
+import kotlinx.coroutines.withContext
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.service.location.LocationService
+
+// The amount of time (in seconds) we need to be in a new
+// location before we update the home region.
+// Currently set to 2 weeks.
+// https://searchfox.org/mozilla-central/rev/89d33e1c3b0a57a9377b4815c2f4b58d933b7c32/toolkit/modules/Region.jsm#82-85
+private const val UPDATE_INTERVAL_MS = 14 * 24 * 60 * 60 * 1000
+
+// The maximum number of times we retry fetching the region from
+// the location service until we give up. We will try again on
+// the next app start.
+private const val MAX_RETRIES = 3
+
+// Timeout until we try to fetch the region again after a failure.
+private const val RETRY_TIMEOUT_MS: Long = 10L * 60L * 1000L
+
+private const val PREFERENCE_FILE = "mozac_feature_search_region"
+private const val PREFERENCE_KEY_HOME_REGION = "region.home"
+private const val PREFERENCE_KEY_CURRENT_REGION = "region.current"
+private const val PREFERENCE_KEY_REGION_FIRST_SEEN = "region.first_seen"
+
+/**
+ * Internal RegionManager for keeping track of the "current" and "home" region of a user. Used by
+ * [RegionMiddleware].
+ */
+internal class RegionManager(
+ context: Context,
+ private val locationService: LocationService,
+ private val currentTime: () -> Long = { System.currentTimeMillis() },
+ private val preferences: Lazy<SharedPreferences> = lazy {
+ context.getSharedPreferences(
+ PREFERENCE_FILE,
+ Context.MODE_PRIVATE,
+ )
+ },
+ private val dispatcher: CoroutineDispatcher = Dispatchers.IO,
+) {
+ private var homeRegion: String?
+ get() = preferences.value.getString(PREFERENCE_KEY_HOME_REGION, null)
+ set(value) = preferences.value.edit().putString(PREFERENCE_KEY_HOME_REGION, value).apply()
+
+ private var currentRegion: String?
+ get() = preferences.value.getString(PREFERENCE_KEY_CURRENT_REGION, null)
+ set(value) = preferences.value.edit().putString(PREFERENCE_KEY_CURRENT_REGION, value).apply()
+
+ private var firstSeen: Long?
+ get() = preferences.value.getLong(PREFERENCE_KEY_REGION_FIRST_SEEN, 0)
+ set(value) = if (value == null) {
+ preferences.value.edit().remove(PREFERENCE_KEY_REGION_FIRST_SEEN).apply()
+ } else {
+ preferences.value.edit().putLong(PREFERENCE_KEY_REGION_FIRST_SEEN, value).apply()
+ }
+
+ fun region(): RegionState? {
+ return homeRegion?.let { region ->
+ RegionState(
+ region,
+ currentRegion ?: region,
+ )
+ }
+ }
+
+ suspend fun update(): RegionState? {
+ val region = fetchRegionWithRetry()?.countryCode ?: return null
+
+ if (homeRegion == null) {
+ // If we do not have a value for the home region yet, then we can set it immediately.
+ homeRegion = region
+ return RegionState(home = region, current = region)
+ }
+
+ return when (region) {
+ homeRegion -> {
+ // If we are in the home region (again) then we can clear a previously seen different
+ // "current" region.
+ currentRegion = null
+ firstSeen = null
+ null
+ }
+
+ currentRegion -> {
+ val now = currentTime()
+ if (now > (firstSeen ?: 0) + UPDATE_INTERVAL_MS) {
+ // We have been in the "current" region longer than the specified "interval".
+ // So we will set the "current" region as our new home region.
+ homeRegion = region
+ RegionState(home = region, current = region)
+ } else {
+ null
+ }
+ }
+
+ else -> {
+ // This region is neither the home region nor the current region. We set it as the
+ // new "current" region and remember when we saw it the first time.
+ firstSeen = currentTime()
+ currentRegion = region
+ null
+ }
+ }
+ }
+
+ private suspend fun fetchRegionWithRetry(): LocationService.Region? = withContext(dispatcher) {
+ repeat(MAX_RETRIES) {
+ val region = locationService.fetchRegion(readFromCache = true)
+ if (region != null) {
+ return@withContext region
+ }
+ delay(RETRY_TIMEOUT_MS)
+ }
+ return@withContext null
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionMiddleware.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionMiddleware.kt
new file mode 100644
index 0000000000..7bf4893c0b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/region/RegionMiddleware.kt
@@ -0,0 +1,72 @@
+/* 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/. */
+
+package mozilla.components.feature.search.region
+
+import android.content.Context
+import androidx.annotation.VisibleForTesting
+import kotlinx.coroutines.CoroutineDispatcher
+import kotlinx.coroutines.DelicateCoroutinesApi
+import kotlinx.coroutines.Dispatchers
+import kotlinx.coroutines.GlobalScope
+import kotlinx.coroutines.Job
+import kotlinx.coroutines.launch
+import mozilla.components.browser.state.action.BrowserAction
+import mozilla.components.browser.state.action.InitAction
+import mozilla.components.browser.state.action.SearchAction
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.lib.state.Middleware
+import mozilla.components.lib.state.MiddlewareContext
+import mozilla.components.lib.state.Store
+import mozilla.components.service.location.LocationService
+
+/**
+ * [Middleware] implementation for updating the [RegionState] using the provided [LocationService].
+ */
+class RegionMiddleware(
+ context: Context,
+ locationService: LocationService,
+ private val ioDispatcher: CoroutineDispatcher = Dispatchers.IO,
+) : Middleware<BrowserState, BrowserAction> {
+ @VisibleForTesting
+ internal var regionManager = RegionManager(context, locationService, dispatcher = ioDispatcher)
+
+ @VisibleForTesting
+ @Volatile
+ internal var updateJob: Job? = null
+
+ override fun invoke(
+ context: MiddlewareContext<BrowserState, BrowserAction>,
+ next: (BrowserAction) -> Unit,
+ action: BrowserAction,
+ ) {
+ if (action is InitAction || action is SearchAction.RefreshSearchEnginesAction) {
+ updateJob = determineRegion(context.store)
+ }
+
+ next(action)
+ }
+
+ @OptIn(DelicateCoroutinesApi::class)
+ private fun determineRegion(
+ store: Store<BrowserState, BrowserAction>,
+ ) = GlobalScope.launch(ioDispatcher) {
+ // Get the region state from the RegionManager. If there's none then dispatch the default
+ // region to be used.
+ val region = regionManager.region()
+ if (region != null) {
+ store.dispatch(SearchAction.SetRegionAction(region))
+ } else {
+ store.dispatch(SearchAction.SetRegionAction(RegionState.Default))
+ }
+
+ // Ask the RegionManager to perform an update. If the "home" region changed then it will
+ // return a new RegionState.
+ val update = regionManager.update()
+ if (update != null) {
+ store.dispatch(SearchAction.SetRegionAction(update))
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorage.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorage.kt
new file mode 100644
index 0000000000..eaa196739d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorage.kt
@@ -0,0 +1,278 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.content.Context
+import android.content.res.AssetManager
+import kotlinx.coroutines.Deferred
+import kotlinx.coroutines.DelicateCoroutinesApi
+import kotlinx.coroutines.GlobalScope
+import kotlinx.coroutines.async
+import kotlinx.coroutines.withContext
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.feature.search.middleware.SearchExtraParams
+import mozilla.components.feature.search.middleware.SearchMiddleware
+import mozilla.components.support.base.log.logger.Logger
+import mozilla.components.support.ktx.android.content.res.readJSONObject
+import mozilla.components.support.ktx.android.org.json.toList
+import mozilla.components.support.ktx.android.org.json.tryGetString
+import org.json.JSONArray
+import org.json.JSONObject
+import java.util.Locale
+import kotlin.coroutines.CoroutineContext
+
+private val logger = Logger("BundledSearchEnginesStorage")
+
+/**
+ * A storage implementation for reading bundled [SearchEngine]s from the app's assets.
+ */
+internal class BundledSearchEnginesStorage(
+ private val context: Context,
+) : SearchMiddleware.BundleStorage {
+ /**
+ * Load the [SearchMiddleware.BundleStorage.Bundle] for the given [region] and [locale].
+ */
+ override suspend fun load(
+ region: RegionState,
+ locale: Locale,
+ distribution: String?,
+ searchExtraParams: SearchExtraParams?,
+ coroutineContext: CoroutineContext,
+ ): SearchMiddleware.BundleStorage.Bundle = withContext(coroutineContext) {
+ val localizedConfiguration = loadAndFilterConfiguration(context, region, locale, distribution)
+ val searchEngineIdentifiers = localizedConfiguration.visibleSearchEngines
+
+ val searchEngines = loadSearchEnginesFromList(
+ context = context,
+ searchEngineIdentifiers = searchEngineIdentifiers.distinct(),
+ type = SearchEngine.Type.BUNDLED,
+ searchExtraParams = searchExtraParams,
+ coroutineContext = coroutineContext,
+ )
+
+ // Reorder the list of search engines according to the configuration.
+ // Note: we're using the name of the search engine, not the id, so we can only do this
+ // after we've loaded the search engine from the XML
+ val searchOrder = localizedConfiguration.searchOrder
+ val orderedList = searchOrder
+ .map { name ->
+ searchEngines.filter { it.name == name }
+ }
+ .flatten()
+
+ val unorderedRest = searchEngines
+ .filter {
+ !searchOrder.contains(it.name)
+ }
+
+ val defaultEngine = localizedConfiguration.searchDefault?.let { name ->
+ searchEngines.find { it.name == name }
+ } ?: throw IllegalStateException("No default engine for configuration: locale=$locale, region=$region")
+
+ SearchMiddleware.BundleStorage.Bundle(
+ list = orderedList + unorderedRest,
+ defaultSearchEngineId = defaultEngine.id,
+ )
+ }
+
+ override suspend fun load(
+ ids: List<String>,
+ searchExtraParams: SearchExtraParams?,
+ coroutineContext: CoroutineContext,
+ ): List<SearchEngine> = withContext(coroutineContext) {
+ if (ids.isEmpty()) {
+ emptyList()
+ } else {
+ loadSearchEnginesFromList(
+ context = context,
+ searchEngineIdentifiers = ids.distinct(),
+ type = SearchEngine.Type.BUNDLED_ADDITIONAL,
+ searchExtraParams = searchExtraParams,
+ coroutineContext = coroutineContext,
+ )
+ }
+ }
+}
+
+private data class SearchEngineListConfiguration(
+ val visibleSearchEngines: List<String>,
+ val searchOrder: List<String>,
+ val searchDefault: String?,
+)
+
+private fun loadAndFilterConfiguration(
+ context: Context,
+ region: RegionState,
+ locale: Locale,
+ distribution: String?,
+): SearchEngineListConfiguration {
+ val config = context.assets.readJSONObject("search/list.json")
+
+ val configBlocks = pickConfigurationBlocks(locale, config)
+ val jsonSearchEngineIdentifiers =
+ getSearchEngineIdentifiersFromBlock(region, locale, distribution, configBlocks)
+
+ val searchOrder = getSearchOrderFromBlock(region, configBlocks)
+ val searchDefault = getSearchDefaultFromBlock(region, configBlocks)
+
+ return SearchEngineListConfiguration(
+ applyOverridesIfNeeded(region, config, jsonSearchEngineIdentifiers),
+ searchOrder.toList(),
+ searchDefault,
+ )
+}
+
+private fun pickConfigurationBlocks(
+ locale: Locale,
+ config: JSONObject,
+): Array<JSONObject> {
+ val localesConfig = config.getJSONObject("locales")
+
+ val localizedConfig = when {
+ // First try (Locale): locales/xx_XX/
+ localesConfig.has(locale.languageTag) ->
+ localesConfig.getJSONObject(locale.languageTag)
+
+ // Second try (Language): locales/xx/
+ localesConfig.has(locale.language) ->
+ localesConfig.getJSONObject(locale.language)
+
+ // Give up, and fallback to defaults
+ else -> null
+ }
+
+ return localizedConfig?.let {
+ arrayOf(it, config)
+ } ?: arrayOf(config)
+}
+
+private fun getSearchEngineIdentifiersFromBlock(
+ region: RegionState,
+ locale: Locale,
+ distribution: String?,
+ configBlocks: Array<JSONObject>,
+): JSONArray {
+ // Now test if there's an override for the distribution or region (if it's set)
+ return distribution?.let { getArrayFromBlock(region, distribution, configBlocks) }
+ ?: getArrayFromBlock(region, "visibleDefaultEngines", configBlocks)
+ ?: throw IllegalStateException("No visibleDefaultEngines using region $region and locale $locale")
+}
+
+private fun getSearchDefaultFromBlock(
+ region: RegionState,
+ configBlocks: Array<JSONObject>,
+): String? = getValueFromBlock(region, configBlocks) {
+ it.tryGetString("searchDefault")
+}
+
+private fun getSearchOrderFromBlock(
+ region: RegionState,
+ configBlocks: Array<JSONObject>,
+): JSONArray? = getArrayFromBlock(region, "searchOrder", configBlocks)
+
+private fun getArrayFromBlock(
+ region: RegionState,
+ key: String,
+ blocks: Array<JSONObject>,
+): JSONArray? = getValueFromBlock(region, blocks) {
+ it.optJSONArray(key)
+}
+
+/**
+ * This looks for a JSONObject in the config blocks it is passed that is able to be transformed
+ * into a value. It tries the permutations of locale and region from most specific to least
+ * specific.
+ *
+ * This has to be done on a value basis, not a configBlock basis, as the configuration for a
+ * given locale/region is not grouped into one object, but spread across the json file,
+ * according to these rules.
+ */
+private fun <T : Any> getValueFromBlock(
+ region: RegionState,
+ blocks: Array<JSONObject>,
+ transform: (JSONObject) -> T?,
+): T? {
+ val regions = arrayOf(region.home, "default")
+
+ return blocks
+ .flatMap { block ->
+ regions.mapNotNull { region -> block.optJSONObject(region) }
+ }
+ .mapNotNull(transform)
+ .firstOrNull()
+}
+
+private fun applyOverridesIfNeeded(
+ region: RegionState,
+ config: JSONObject,
+ jsonSearchEngineIdentifiers: JSONArray,
+): List<String> {
+ val overrides = config.getJSONObject("regionOverrides")
+ val searchEngineIdentifiers = mutableListOf<String>()
+ val regionOverrides = if (overrides.has(region.home)) {
+ overrides.getJSONObject(region.home)
+ } else {
+ null
+ }
+
+ for (i in 0 until jsonSearchEngineIdentifiers.length()) {
+ var identifier = jsonSearchEngineIdentifiers.getString(i)
+ if (regionOverrides != null && regionOverrides.has(identifier)) {
+ identifier = regionOverrides.getString(identifier)
+ }
+ searchEngineIdentifiers.add(identifier)
+ }
+
+ return searchEngineIdentifiers
+}
+
+@OptIn(DelicateCoroutinesApi::class)
+private suspend fun loadSearchEnginesFromList(
+ context: Context,
+ searchEngineIdentifiers: List<String>,
+ type: SearchEngine.Type,
+ searchExtraParams: SearchExtraParams?,
+ coroutineContext: CoroutineContext,
+): List<SearchEngine> {
+ val assets = context.assets
+ val reader = SearchEngineReader(type, searchExtraParams)
+
+ val deferredSearchEngines = mutableListOf<Deferred<SearchEngine?>>()
+
+ searchEngineIdentifiers.forEach { identifier ->
+ deferredSearchEngines.add(
+ GlobalScope.async(coroutineContext) {
+ loadSearchEngine(assets, reader, identifier)
+ },
+ )
+ }
+
+ return deferredSearchEngines.mapNotNull { it.await() }
+}
+
+@Suppress("TooGenericExceptionCaught")
+private fun loadSearchEngine(
+ assets: AssetManager,
+ reader: SearchEngineReader,
+ identifier: String,
+): SearchEngine? = try {
+ assets.open("searchplugins/$identifier.xml").use { stream ->
+ reader.loadStream(identifier, stream)
+ }
+} catch (e: Exception) {
+ // Handling all exceptions here (instead of just IOExceptions) as we're
+ // seeing crashes we can't explain currently. Letting the app launch
+ // will eventually help us understand the root cause:
+ // https://github.com/mozilla-mobile/android-components/issues/12304
+
+ // We should also consider logging these errors to Sentry:
+ // https://github.com/mozilla-mobile/android-components/issues/12313
+ logger.error("Could not load additional search engine with ID $identifier", e)
+ null
+}
+
+private val Locale.languageTag: String
+ get() = "$language-$country"
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/CustomSearchEnginesStorage.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/CustomSearchEnginesStorage.kt
new file mode 100644
index 0000000000..f0961d6b63
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/CustomSearchEnginesStorage.kt
@@ -0,0 +1,66 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.content.Context
+import android.util.AtomicFile
+import android.util.Base64
+import androidx.annotation.VisibleForTesting
+import kotlinx.coroutines.Dispatchers
+import kotlinx.coroutines.withContext
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.feature.search.middleware.SearchMiddleware
+import java.io.File
+import kotlin.coroutines.CoroutineContext
+
+internal const val SEARCH_FILE_EXTENSION = ".xml"
+internal const val SEARCH_DIR_NAME = "search-engines"
+
+/**
+ * A storage implementation for organizing [SearchEngine]s. Its primary use case is for persisting
+ * custom search engines added by users.
+ */
+internal class CustomSearchEngineStorage(
+ private val context: Context,
+ private val coroutineContext: CoroutineContext = Dispatchers.IO,
+) : SearchMiddleware.CustomStorage {
+ private val reader = SearchEngineReader(SearchEngine.Type.CUSTOM)
+ private val writer = SearchEngineWriter()
+
+ override suspend fun loadSearchEngineList(): List<SearchEngine> = withContext(coroutineContext) {
+ val searchEngineList = mutableListOf<SearchEngine>()
+ getFileDirectory().listFiles()?.forEach {
+ val filename = it.name.removeSuffix(SEARCH_FILE_EXTENSION)
+ val identifier = String(Base64.decode(filename, Base64.NO_WRAP or Base64.URL_SAFE))
+ searchEngineList.add(loadSearchEngine(identifier))
+ }
+ searchEngineList.toList()
+ }
+
+ suspend fun loadSearchEngine(identifier: String): SearchEngine = withContext(coroutineContext) {
+ reader.loadFile(identifier, getSearchFile(identifier))
+ }
+
+ override suspend fun saveSearchEngine(searchEngine: SearchEngine): Boolean = withContext(coroutineContext) {
+ writer.saveSearchEngineXML(searchEngine, getSearchFile(searchEngine.id))
+ }
+
+ override suspend fun removeSearchEngine(identifier: String) = withContext(coroutineContext) {
+ getSearchFile(identifier).delete()
+ }
+
+ @VisibleForTesting(otherwise = VisibleForTesting.PRIVATE)
+ internal fun getSearchFile(identifier: String): AtomicFile {
+ val encodedId = Base64.encodeToString(identifier.toByteArray(), Base64.NO_WRAP or Base64.URL_SAFE)
+ return AtomicFile(File(getFileDirectory(), encodedId + SEARCH_FILE_EXTENSION))
+ }
+
+ private fun getFileDirectory(): File =
+ File(context.filesDir, SEARCH_DIR_NAME).also {
+ if (!it.exists()) {
+ it.mkdirs()
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineReader.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineReader.kt
new file mode 100644
index 0000000000..51d2ccec52
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineReader.kt
@@ -0,0 +1,243 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.graphics.Bitmap
+import android.graphics.BitmapFactory
+import android.net.Uri
+import android.util.AtomicFile
+import android.util.Base64
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.feature.search.middleware.SearchExtraParams
+import org.xmlpull.v1.XmlPullParser
+import org.xmlpull.v1.XmlPullParserException
+import org.xmlpull.v1.XmlPullParserFactory
+import java.io.IOException
+import java.io.InputStream
+import java.io.InputStreamReader
+import java.nio.charset.StandardCharsets
+
+internal const val URL_TYPE_SUGGEST_JSON = "application/x-suggestions+json"
+internal const val URL_TYPE_SEARCH_HTML = "text/html"
+internal const val URL_REL_MOBILE = "mobile"
+internal const val IMAGE_URI_PREFIX = "data:image/png;base64,"
+internal const val GOOGLE_ID = "google"
+
+// List of general search engine ids, taken from
+// https://searchfox.org/mozilla-central/rev/ef0aa879e94534ffd067a3748d034540a9fc10b0/toolkit/components/search/SearchUtils.sys.mjs#200
+internal val GENERAL_SEARCH_ENGINE_IDS = setOf(
+ GOOGLE_ID,
+ "ddg",
+ "bing",
+ "baidu",
+ "ecosia",
+ "qwant",
+ "yahoo-jp",
+ "seznam-cz",
+ "coccoc",
+ "baidu",
+)
+
+/**
+ * A simple XML reader for search engine plugins.
+ *
+ * @param type the [SearchEngine.Type] that the read [SearchEngine]s will get assigned.
+ * @param searchExtraParams Optional search extra params.
+ */
+internal class SearchEngineReader(
+ private val type: SearchEngine.Type,
+ private val searchExtraParams: SearchExtraParams? = null,
+) {
+ private class SearchEngineBuilder(
+ private val type: SearchEngine.Type,
+ private val identifier: String,
+ ) {
+ var resultsUrls: MutableList<String> = mutableListOf()
+ var suggestUrl: String? = null
+ var name: String? = null
+ var icon: Bitmap? = null
+ var inputEncoding: String? = null
+
+ fun toSearchEngine() = SearchEngine(
+ id = identifier,
+ name = name!!,
+ icon = icon!!,
+ type = type,
+ resultUrls = resultsUrls,
+ suggestUrl = suggestUrl,
+ inputEncoding = inputEncoding,
+ isGeneral = isGeneralSearchEngine(identifier, type),
+ )
+
+ /**
+ * Returns true if the provided [type] is a custom search engine or the [identifier] is
+ * included in [GENERAL_SEARCH_ENGINE_IDS].
+ */
+ private fun isGeneralSearchEngine(identifier: String, type: SearchEngine.Type): Boolean =
+ type == SearchEngine.Type.CUSTOM ||
+ identifier.startsWith(GOOGLE_ID) ||
+ GENERAL_SEARCH_ENGINE_IDS.contains(identifier)
+ }
+
+ /**
+ * Loads [SearchEngine] from a provided [file]
+ */
+ fun loadFile(identifier: String, file: AtomicFile): SearchEngine {
+ return loadStream(identifier, file.openRead())
+ }
+
+ /**
+ * Loads a <code>SearchEngine</code> from the given <code>stream</code> and assigns it the given
+ * <code>identifier</code>.
+ */
+ @Throws(IOException::class, XmlPullParserException::class)
+ fun loadStream(identifier: String, stream: InputStream): SearchEngine {
+ val builder = SearchEngineBuilder(type, identifier)
+
+ val parser = XmlPullParserFactory.newInstance().newPullParser()
+ parser.setInput(InputStreamReader(stream, StandardCharsets.UTF_8))
+ parser.next()
+
+ readSearchPlugin(parser, builder)
+
+ return builder.toSearchEngine()
+ }
+
+ @Throws(XmlPullParserException::class, IOException::class)
+ @Suppress("ComplexMethod")
+ private fun readSearchPlugin(parser: XmlPullParser, builder: SearchEngineBuilder) {
+ if (XmlPullParser.START_TAG != parser.eventType) {
+ throw XmlPullParserException("Expected start tag: " + parser.positionDescription)
+ }
+
+ val name = parser.name
+ if ("SearchPlugin" != name && "OpenSearchDescription" != name) {
+ throw XmlPullParserException(
+ "Expected <SearchPlugin> or <OpenSearchDescription> as root tag: ${parser.positionDescription}",
+ )
+ }
+
+ while (parser.next() != XmlPullParser.END_TAG) {
+ if (parser.eventType != XmlPullParser.START_TAG) {
+ continue
+ }
+
+ when (parser.name) {
+ "ShortName" -> readShortName(parser, builder)
+ "Url" -> readUrl(parser, builder)
+ "Image" -> readImage(parser, builder)
+ "InputEncoding" -> readInputEncoding(parser, builder)
+ else -> skip(parser)
+ }
+ }
+ }
+
+ @Throws(XmlPullParserException::class, IOException::class)
+ private fun readUrl(parser: XmlPullParser, builder: SearchEngineBuilder) {
+ parser.require(XmlPullParser.START_TAG, null, "Url")
+
+ val type = parser.getAttributeValue(null, "type")
+ val template = parser.getAttributeValue(null, "template")
+ val rel = parser.getAttributeValue(null, "rel")
+
+ val url = buildString {
+ append(readUri(parser, template))
+ searchExtraParams?.let {
+ with(it) {
+ if (builder.name == searchEngineName) {
+ featureEnablerParam?.let { append("&$featureEnablerName=$it") }
+ append("&$channelIdName=$channelIdParam")
+ }
+ }
+ }
+ }
+
+ if (type == URL_TYPE_SEARCH_HTML) {
+ // Prefer mobile URIs.
+ if (rel != null && rel == URL_REL_MOBILE) {
+ builder.resultsUrls.add(0, url)
+ } else {
+ builder.resultsUrls.add(url)
+ }
+ } else if (type == URL_TYPE_SUGGEST_JSON) {
+ builder.suggestUrl = url
+ }
+ }
+
+ @Throws(XmlPullParserException::class, IOException::class)
+ private fun readUri(parser: XmlPullParser, template: String): Uri {
+ var uri = Uri.parse(template)
+
+ while (parser.next() != XmlPullParser.END_TAG) {
+ if (parser.eventType != XmlPullParser.START_TAG) {
+ continue
+ }
+
+ if (parser.name == "Param") {
+ val name = parser.getAttributeValue(null, "name")
+ val value = parser.getAttributeValue(null, "value")
+ uri = uri.buildUpon().appendQueryParameter(name, value).build()
+ parser.nextTag()
+ } else {
+ skip(parser)
+ }
+ }
+
+ return uri
+ }
+
+ @Throws(XmlPullParserException::class, IOException::class)
+ private fun skip(parser: XmlPullParser) {
+ if (parser.eventType != XmlPullParser.START_TAG) {
+ throw IllegalStateException()
+ }
+ var depth = 1
+ while (depth != 0) {
+ when (parser.next()) {
+ XmlPullParser.END_TAG -> depth--
+ XmlPullParser.START_TAG -> depth++
+ // else: Do nothing - we're skipping content
+ }
+ }
+ }
+
+ @Throws(IOException::class, XmlPullParserException::class)
+ private fun readShortName(parser: XmlPullParser, builder: SearchEngineBuilder) {
+ parser.require(XmlPullParser.START_TAG, null, "ShortName")
+ if (parser.next() == XmlPullParser.TEXT) {
+ builder.name = parser.text
+ parser.nextTag()
+ }
+ }
+
+ @Throws(IOException::class, XmlPullParserException::class)
+ private fun readImage(parser: XmlPullParser, builder: SearchEngineBuilder) {
+ parser.require(XmlPullParser.START_TAG, null, "Image")
+
+ if (parser.next() != XmlPullParser.TEXT) {
+ return
+ }
+
+ val uri = parser.text
+ if (!uri.startsWith(IMAGE_URI_PREFIX)) {
+ return
+ }
+
+ val raw = Base64.decode(uri.substring(IMAGE_URI_PREFIX.length), Base64.DEFAULT)
+
+ builder.icon = BitmapFactory.decodeByteArray(raw, 0, raw.size)
+
+ parser.nextTag()
+ }
+
+ @Throws(IOException::class, XmlPullParserException::class)
+ private fun readInputEncoding(parser: XmlPullParser, builder: SearchEngineBuilder) {
+ parser.require(XmlPullParser.START_TAG, null, "InputEncoding")
+ if (parser.next() == XmlPullParser.TEXT) {
+ builder.inputEncoding = parser.text
+ parser.nextTag()
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineWriter.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineWriter.kt
new file mode 100644
index 0000000000..15d7f17212
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchEngineWriter.kt
@@ -0,0 +1,115 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.graphics.Bitmap
+import android.util.AtomicFile
+import android.util.Base64
+import androidx.annotation.VisibleForTesting
+import mozilla.components.browser.state.search.SearchEngine
+import org.w3c.dom.DOMException
+import org.w3c.dom.Document
+import java.io.ByteArrayOutputStream
+import java.io.File
+import javax.xml.parsers.DocumentBuilderFactory
+import javax.xml.parsers.ParserConfigurationException
+import javax.xml.transform.TransformerConfigurationException
+import javax.xml.transform.TransformerException
+import javax.xml.transform.TransformerFactory
+import javax.xml.transform.dom.DOMSource
+import javax.xml.transform.stream.StreamResult
+
+/**
+ * A simple XML writer for search engine plugins.
+ */
+internal class SearchEngineWriter {
+ /**
+ * Builds and save the XML document of [SearchEngine] to the provided [File].
+ *
+ * @param searchEngine the search engine to build XML with.
+ * @param file the file instance to save the search engine XML.
+ * @param document the document instance to build search engine XML with.
+ * @return true if the XML is built and saved successfully, false otherwise.
+ */
+ fun saveSearchEngineXML(
+ searchEngine: SearchEngine,
+ file: AtomicFile,
+ document: Document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument(),
+ ): Boolean {
+ return try {
+ buildSearchEngineXML(searchEngine, document)
+ saveXMLDocumentToFile(document, file)
+ true
+ } catch (e: ParserConfigurationException) {
+ false
+ } catch (e: DOMException) {
+ false
+ } catch (e: TransformerConfigurationException) {
+ false
+ } catch (e: TransformerException) {
+ false
+ }
+ }
+
+ @VisibleForTesting(otherwise = VisibleForTesting.PRIVATE)
+ @Throws(ParserConfigurationException::class, DOMException::class)
+ internal fun buildSearchEngineXML(
+ searchEngine: SearchEngine,
+ xmlDocument: Document,
+ ) {
+ val rootElement = xmlDocument.createElement("OpenSearchDescription")
+ rootElement.setAttribute("xmlns", "http://a9.com/-/spec/opensearch/1.1/")
+ rootElement.setAttribute("xmlns:moz", "http://www.mozilla.org/2006/browser/search/")
+ xmlDocument.appendChild(rootElement)
+
+ val shortNameElement = xmlDocument.createElement("ShortName")
+ shortNameElement.textContent = searchEngine.name
+ rootElement.appendChild(shortNameElement)
+
+ val descriptionElement = xmlDocument.createElement("Description")
+ descriptionElement.textContent = searchEngine.name
+ rootElement.appendChild(descriptionElement)
+
+ val imageElement = xmlDocument.createElement("Image")
+ imageElement.setAttribute("width", "16")
+ imageElement.setAttribute("height", "16")
+ imageElement.textContent = searchEngine.icon.toBase64()
+ rootElement.appendChild(imageElement)
+
+ searchEngine.inputEncoding?.let { inputEncoding ->
+ val inputEncodingElement = xmlDocument.createElement("InputEncoding")
+ inputEncodingElement.textContent = inputEncoding
+ rootElement.appendChild(inputEncodingElement)
+ }
+
+ searchEngine.resultUrls.forEach { url ->
+ val urlElement = xmlDocument.createElement("Url")
+ urlElement.setAttribute("type", URL_TYPE_SEARCH_HTML)
+ urlElement.setAttribute("template", url)
+ rootElement.appendChild(urlElement)
+ }
+
+ searchEngine.suggestUrl?.let { url ->
+ val urlElement = xmlDocument.createElement("Url")
+ urlElement.setAttribute("type", URL_TYPE_SUGGEST_JSON)
+ val templateSearchString = url.replace("%s", "{searchTerms}")
+ urlElement.setAttribute("template", templateSearchString)
+ rootElement.appendChild(urlElement)
+ }
+ }
+
+ @VisibleForTesting(otherwise = VisibleForTesting.PRIVATE)
+ @Throws(TransformerConfigurationException::class, TransformerException::class)
+ internal fun saveXMLDocumentToFile(doc: Document, file: AtomicFile) =
+ TransformerFactory.newInstance().newTransformer().transform(DOMSource(doc), StreamResult(file.baseFile))
+}
+
+private const val BITMAP_COMPRESS_QUALITY = 100
+private fun Bitmap.toBase64(): String {
+ val stream = ByteArrayOutputStream()
+ compress(Bitmap.CompressFormat.PNG, BITMAP_COMPRESS_QUALITY, stream)
+ val encodedImage = Base64.encodeToString(stream.toByteArray(), Base64.DEFAULT)
+ return "data:image/png;base64,$encodedImage"
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchMetadataStorage.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchMetadataStorage.kt
new file mode 100644
index 0000000000..b835f5ab37
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/storage/SearchMetadataStorage.kt
@@ -0,0 +1,103 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.content.Context
+import android.content.SharedPreferences
+import mozilla.components.feature.search.middleware.SearchMiddleware
+
+private const val PREFERENCE_FILE = "mozac_feature_search_metadata"
+
+private const val PREFERENCE_KEY_USER_SELECTED_SEARCH_ENGINE_ID = "user_selected_search_engine_id"
+private const val PREFERENCE_KEY_USER_SELECTED_SEARCH_ENGINE_NAME = "user_selected_search_engine_name"
+private const val PREFERENCE_KEY_HIDDEN_SEARCH_ENGINES = "hidden_search_engines"
+private const val PREFERENCE_KEY_ADDITIONAL_SEARCH_ENGINES = "additional_search_engines"
+private const val PREFERENCE_KEY_DISABLED_SEARCH_ENGINE_ID = "preference_key_disabled_search_engine_id"
+
+/**
+ * Storage for saving additional search related metadata.
+ */
+internal class SearchMetadataStorage(
+ context: Context,
+ private val disabledByDefaultSearchEngineIds: Set<String> = emptySet(),
+ private val preferences: Lazy<SharedPreferences> = lazy {
+ context.getSharedPreferences(
+ PREFERENCE_FILE,
+ Context.MODE_PRIVATE,
+ )
+ },
+) : SearchMiddleware.MetadataStorage {
+ /**
+ * Gets the ID (and optinally name) of the default search engine the user has picked. Returns
+ * `null` if the user has not made a choice.
+ */
+ override suspend fun getUserSelectedSearchEngine(): SearchMiddleware.MetadataStorage.UserChoice? {
+ val id = preferences.value.getString(PREFERENCE_KEY_USER_SELECTED_SEARCH_ENGINE_ID, null)
+ ?: return null
+
+ return SearchMiddleware.MetadataStorage.UserChoice(
+ id,
+ preferences.value.getString(PREFERENCE_KEY_USER_SELECTED_SEARCH_ENGINE_NAME, null),
+ )
+ }
+
+ /**
+ * Sets the ID (and optionally name) of the default search engine the user has picked.
+ */
+ override suspend fun setUserSelectedSearchEngine(id: String, name: String?) {
+ preferences.value.edit()
+ .putString(PREFERENCE_KEY_USER_SELECTED_SEARCH_ENGINE_ID, id)
+ .putString(PREFERENCE_KEY_USER_SELECTED_SEARCH_ENGINE_NAME, name)
+ .apply()
+ }
+
+ /**
+ * Sets the list of IDs of hidden search engines.
+ */
+ override suspend fun setHiddenSearchEngines(ids: List<String>) {
+ preferences.value.edit()
+ .putStringSet(PREFERENCE_KEY_HIDDEN_SEARCH_ENGINES, ids.toSet())
+ .apply()
+ }
+
+ /**
+ * Gets the list of IDs of hidden search engines.
+ */
+ override suspend fun getHiddenSearchEngines(): List<String> {
+ return preferences.value
+ .getStringSet(PREFERENCE_KEY_HIDDEN_SEARCH_ENGINES, emptySet())
+ ?.toList() ?: emptyList()
+ }
+
+ /**
+ * Gets the list of IDs of additional search engines that the user explicitly added.
+ */
+ override suspend fun getAdditionalSearchEngines(): List<String> {
+ return preferences.value
+ .getStringSet(PREFERENCE_KEY_ADDITIONAL_SEARCH_ENGINES, emptySet())
+ ?.toList() ?: emptyList()
+ }
+
+ override suspend fun getDisabledSearchEngineIds(): List<String> {
+ return preferences.value
+ .getStringSet(PREFERENCE_KEY_DISABLED_SEARCH_ENGINE_ID, disabledByDefaultSearchEngineIds)
+ ?.toList() ?: emptyList()
+ }
+
+ override suspend fun setDisabledSearchEngineIds(ids: List<String>) {
+ preferences.value.edit()
+ .putStringSet(PREFERENCE_KEY_DISABLED_SEARCH_ENGINE_ID, ids.toSet())
+ .apply()
+ }
+
+ /**
+ * Sets the list of IDs of additional search engines that the user explicitly added.
+ */
+ override suspend fun setAdditionalSearchEngines(ids: List<String>) {
+ preferences.value.edit()
+ .putStringSet(PREFERENCE_KEY_ADDITIONAL_SEARCH_ENGINES, ids.toSet())
+ .apply()
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/Parser.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/Parser.kt
new file mode 100644
index 0000000000..c52fdf1bdb
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/Parser.kt
@@ -0,0 +1,79 @@
+/* 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/. */
+
+package mozilla.components.feature.search.suggestions
+
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.support.ktx.android.org.json.asSequence
+import org.json.JSONArray
+import org.json.JSONObject
+
+/**
+ * The Parser is a function that takes a JSON Response and maps
+ * it to a Suggestion list.
+ */
+typealias JSONResponse = String
+typealias ResponseParser = (JSONResponse) -> List<String>
+
+/**
+ * Builds a Parser that pulls suggestions out of a given index
+ */
+private fun buildJSONArrayParser(resultsIndex: Int): ResponseParser {
+ return { input ->
+ JSONArray(input)
+ .getJSONArray(resultsIndex)
+ .asSequence()
+ .map { it as? String }
+ .filterNotNull()
+ .toList()
+ }
+}
+
+/**
+ * Builds a Parser that pulls suggestions out of a JSON object with the given key
+ */
+private fun buildJSONObjectParser(resultsKey: String): ResponseParser {
+ return { input ->
+ JSONObject(input)
+ .getJSONArray(resultsKey)
+ .asSequence()
+ .map { it as? String }
+ .filterNotNull()
+ .toList()
+ }
+}
+
+/**
+ * Builds a custom parser for Qwant
+ */
+private fun buildQwantParser(): ResponseParser {
+ return { input ->
+ JSONObject(input)
+ .getJSONObject("data")
+ .getJSONArray("items")
+ .asSequence()
+ .map { it as? JSONObject }
+ .map { it?.getString("value") }
+ .filterNotNull()
+ .toList()
+ }
+}
+
+/**
+ * The available Parsers
+ */
+internal val defaultResponseParser = buildJSONArrayParser(1)
+internal val azerdictResponseParser = buildJSONObjectParser("suggestions")
+internal val daumResponseParser = buildJSONObjectParser("items")
+internal val qwantResponseParser = buildQwantParser()
+
+/**
+ * Selects a Parser based on a SearchEngine
+ */
+internal fun selectResponseParser(searchEngine: SearchEngine): ResponseParser = when (searchEngine.name) {
+ "Azerdict" -> azerdictResponseParser
+ "다음지도" -> daumResponseParser
+ "Qwant" -> qwantResponseParser
+ else -> defaultResponseParser
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/SearchSuggestionClient.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/SearchSuggestionClient.kt
new file mode 100644
index 0000000000..d4a0b5434b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/suggestions/SearchSuggestionClient.kt
@@ -0,0 +1,101 @@
+/* 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/. */
+
+package mozilla.components.feature.search.suggestions
+
+import android.content.Context
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.state.selectedOrDefaultSearchEngine
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.feature.search.ext.buildSuggestionsURL
+import mozilla.components.feature.search.ext.canProvideSearchSuggestions
+import mozilla.components.support.base.log.logger.Logger
+import org.json.JSONException
+import java.io.IOException
+
+/**
+ * Async function responsible for taking a URL and returning the results
+ */
+typealias SearchSuggestionFetcher = suspend (url: String) -> String?
+
+/**
+ * Provides an interface to get search suggestions from a given SearchEngine.
+ */
+class SearchSuggestionClient {
+ private val context: Context?
+ private val fetcher: SearchSuggestionFetcher
+ private val logger = Logger("SearchSuggestionClient")
+
+ val store: BrowserStore?
+ var searchEngine: SearchEngine? = null
+ private set
+
+ internal constructor(
+ context: Context?,
+ store: BrowserStore?,
+ searchEngine: SearchEngine?,
+ fetcher: SearchSuggestionFetcher,
+ ) {
+ this.context = context
+ this.store = store
+ this.searchEngine = searchEngine
+ this.fetcher = fetcher
+ }
+
+ constructor(searchEngine: SearchEngine, fetcher: SearchSuggestionFetcher) :
+ this (null, null, searchEngine, fetcher)
+
+ constructor(
+ context: Context,
+ store: BrowserStore,
+ fetcher: SearchSuggestionFetcher,
+ ) : this (context, store, null, fetcher)
+
+ /**
+ * Exception types for errors caught while getting a list of suggestions
+ */
+ class FetchException : Exception("There was a problem fetching suggestions")
+ class ResponseParserException : Exception("There was a problem parsing the suggestion response")
+
+ /**
+ * Gets search suggestions for a given query
+ */
+ suspend fun getSuggestions(query: String): List<String>? {
+ val searchEngine = searchEngine ?: run {
+ requireNotNull(store)
+ requireNotNull(context)
+
+ val searchEngine = store.state.search.selectedOrDefaultSearchEngine
+ if (searchEngine == null) {
+ logger.warn("No default search engine for fetching suggestions")
+ return emptyList()
+ } else {
+ this.searchEngine = searchEngine
+ searchEngine
+ }
+ }
+
+ if (!searchEngine.canProvideSearchSuggestions) {
+ // This search engine doesn't support suggestions. Let's only return a default suggestion
+ // for the entered text.
+ return emptyList()
+ }
+
+ val suggestionsURL = searchEngine.buildSuggestionsURL(query)
+
+ val parser = selectResponseParser(searchEngine)
+
+ val suggestionResults = try {
+ suggestionsURL?.let { fetcher(it) }
+ } catch (_: IOException) {
+ throw FetchException()
+ }
+
+ return try {
+ suggestionResults?.let(parser)
+ } catch (_: JSONException) {
+ throw ResponseParserException()
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetry.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetry.kt
new file mode 100644
index 0000000000..971e41583a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetry.kt
@@ -0,0 +1,132 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+import androidx.annotation.VisibleForTesting
+import kotlinx.coroutines.flow.Flow
+import kotlinx.coroutines.flow.map
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.Engine
+import mozilla.components.concept.engine.EngineSession
+import mozilla.components.concept.engine.webextension.MessageHandler
+import mozilla.components.concept.engine.webextension.WebExtension
+import mozilla.components.lib.state.ext.flowScoped
+import mozilla.components.support.base.Component
+import mozilla.components.support.base.facts.Action.INTERACTION
+import mozilla.components.support.base.facts.Fact
+import mozilla.components.support.base.facts.collect
+import mozilla.components.support.base.log.logger.Logger
+import mozilla.components.support.ktx.kotlinx.coroutines.flow.filterChanged
+import org.json.JSONObject
+
+/**
+ * Main configuration and functionality for tracking ads / web searches with specific providers.
+ */
+abstract class BaseSearchTelemetry {
+ var providerList: List<SearchProviderModel>? = emptyList()
+
+ @VisibleForTesting(otherwise = VisibleForTesting.PROTECTED)
+ internal fun setProviderList(providerListSerp: List<SearchProviderModel>) {
+ providerList = providerListSerp
+ }
+
+ /**
+ * Finds provider among list of providers that matches regex in url.
+ * This may additionally return null if the provider list is still being initialized.
+ */
+ @VisibleForTesting(otherwise = VisibleForTesting.PROTECTED)
+ internal fun getProviderForUrl(url: String): SearchProviderModel? =
+ providerList?.find { provider -> provider.searchPageRegexp.containsMatchIn(url) }
+
+ /**
+ * Install the web extensions that this functionality is based on and start listening for updates.
+ */
+ abstract suspend fun install(
+ engine: Engine,
+ store: BrowserStore,
+ providerList: List<SearchProviderModel>,
+ )
+
+ @VisibleForTesting(otherwise = VisibleForTesting.PROTECTED)
+ internal fun installWebExtension(
+ engine: Engine,
+ store: BrowserStore,
+ extensionInfo: ExtensionInfo,
+ ) {
+ engine.installBuiltInWebExtension(
+ id = extensionInfo.id,
+ url = extensionInfo.resourceUrl,
+ onSuccess = { extension ->
+ store.flowScoped { flow ->
+ subscribeToUpdates(flow, extension, extensionInfo)
+ }
+ },
+ onError = { throwable ->
+ Logger.error("Could not install ${extensionInfo.id} extension", throwable)
+ },
+ )
+ }
+
+ protected fun emitFact(
+ event: String,
+ value: String,
+ metadata: Map<String, Any>? = null,
+ ) {
+ Fact(
+ Component.FEATURE_SEARCH,
+ INTERACTION,
+ event,
+ value,
+ metadata,
+ ).collect()
+ }
+
+ protected sealed class Action
+
+ private suspend fun subscribeToUpdates(
+ flow: Flow<BrowserState>,
+ extension: WebExtension,
+ extensionInfo: ExtensionInfo,
+ ) {
+ // Whenever we see a new EngineSession in the store then we register our content message
+ // handler if it has not been added yet.
+ flow.map { it.tabs }
+ .filterChanged { it.engineState.engineSession }
+ .collect { state ->
+ val engineSession = state.engineState.engineSession ?: return@collect
+
+ if (extension.hasContentMessageHandler(engineSession, extensionInfo.messageId)) {
+ return@collect
+ }
+ extension.registerContentMessageHandler(
+ engineSession,
+ extensionInfo.messageId,
+ SearchTelemetryMessageHandler(),
+ )
+ }
+ }
+
+ /**
+ * This method is used to process any valid json message coming from a web-extension.
+ */
+ @VisibleForTesting
+ internal abstract fun processMessage(message: JSONObject)
+
+ @VisibleForTesting
+ internal inner class SearchTelemetryMessageHandler : MessageHandler {
+
+ @Throws(IllegalStateException::class)
+ override fun onMessage(message: Any, source: EngineSession?): Any {
+ if (message is JSONObject) {
+ processMessage(message)
+ } else {
+ throw IllegalStateException("Received unexpected message: $message")
+ }
+
+ return Unit
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ExtensionInfo.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ExtensionInfo.kt
new file mode 100644
index 0000000000..492bdf66f8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ExtensionInfo.kt
@@ -0,0 +1,18 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+/**
+ * Configuration data of web extensions used for the search / ads telemetry.
+ *
+ * @property id webExtension unique id.
+ * @property resourceUrl location of the webextension (may be local or web hosted).
+ * @property messageId message key used for communicating from the extension to the native app.
+ */
+internal data class ExtensionInfo(
+ val id: String,
+ val resourceUrl: String,
+ val messageId: String,
+)
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderCookie.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderCookie.kt
new file mode 100644
index 0000000000..eceae777f2
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderCookie.kt
@@ -0,0 +1,24 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+/**
+ * Cookie details used to identify follow-on searches.
+ *
+ * @property extraCodeParamName the query parameter name in the URL that indicates
+ * this might be a follow-on search.
+ * @property extraCodePrefixes possible values for the query parameter in the URL that indicates
+ * this might be a follow-on search.
+ * @property host the hostname on which the cookie is stored.
+ * @property name the name of the cookie to check.
+ * @property codeParamName the name of parameter within the cookie.
+ */
+data class SearchProviderCookie(
+ val extraCodeParamName: String,
+ val extraCodePrefixes: List<String>,
+ val host: String,
+ val name: String,
+ val codeParamName: String,
+)
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderModel.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderModel.kt
new file mode 100644
index 0000000000..b45e17d231
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SearchProviderModel.kt
@@ -0,0 +1,80 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+/**
+ * All data needed to identify ads of a particular provider.
+ *
+ * @property taggedCodes array of partner codes to match against the partner code parameters in the url.
+ * @property telemetryId provider name e.g. "google", "duckduckgo".
+ * @property organicCodes array of partner codes to match against the parameters in the url.
+ * Matching these codes will report the SERP as organic:<partner code>, which means the search
+ * was performed organically rather than through a SAP.
+ * @property codeParamName name of the query parameter for the partner code.
+ * @property followOnCookies array of cookie details that are used to identify follow-on searches.
+ * @property queryParamNames list of names of the query parameters for the user's search string.
+ * @property searchPageRegexp regular expression used to match the provider.
+ * @property adServerAttributes an array of strings that potentially match data-attribute keys of anchors.
+ * @property followOnParamNames array of query parameter names that are used when a follow-on search occurs.
+ * @property extraAdServersRegexps array of regular expressions that match URLs of potential ad servers.
+ * @property expectedOrganicCodes array of partner codes to match against the parameters in the url.
+ * Matching these codes will report the SERP as organic:none which means the user has done a search
+ * through the search engine's website rather than through SAP.
+ */
+data class SearchProviderModel(
+ val schema: Long,
+ val taggedCodes: List<String>,
+ val telemetryId: String,
+ val organicCodes: List<String>?,
+ val codeParamName: String,
+ val followOnCookies: List<SearchProviderCookie>?,
+ val queryParamNames: List<String>?,
+ val searchPageRegexp: Regex,
+ val adServerAttributes: List<String>?,
+ val followOnParamNames: List<String>?,
+ val extraAdServersRegexps: List<Regex>,
+ val expectedOrganicCodes: List<String>?,
+
+) {
+
+ constructor(
+ schema: Long,
+ taggedCodes: List<String> = emptyList(),
+ telemetryId: String,
+ organicCodes: List<String>? = emptyList(),
+ codeParamName: String = "",
+ followOnCookies: List<SearchProviderCookie>? = emptyList(),
+ queryParamNames: List<String> = emptyList(),
+ searchPageRegexp: String,
+ adServerAttributes: List<String>? = emptyList(),
+ followOnParamNames: List<String>? = emptyList(),
+ extraAdServersRegexps: List<String> = emptyList(),
+ expectedOrganicCodes: List<String>? = emptyList(),
+
+ ) : this(
+ schema = schema,
+ taggedCodes = taggedCodes,
+ telemetryId = telemetryId,
+ organicCodes = organicCodes,
+ codeParamName = codeParamName,
+ followOnCookies = followOnCookies,
+ queryParamNames = queryParamNames,
+ searchPageRegexp = searchPageRegexp.toRegex(),
+ adServerAttributes = adServerAttributes,
+ followOnParamNames = followOnParamNames,
+ extraAdServersRegexps = extraAdServersRegexps.map { it.toRegex() },
+ expectedOrganicCodes = expectedOrganicCodes,
+
+ )
+
+ /**
+ * Checks if any of the given URLs represent an ad from the search engine.
+ * Used to check if a clicked link was for an ad.
+ */
+ fun containsAdLinks(urlList: List<String>) = urlList.any { url -> isAd(url) }
+
+ private fun isAd(url: String) =
+ extraAdServersRegexps.any { adsRegex -> adsRegex.containsMatchIn(url) }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepository.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepository.kt
new file mode 100644
index 0000000000..436b0e22db
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepository.kt
@@ -0,0 +1,170 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+import mozilla.appservices.remotesettings.RemoteSettingsResponse
+import mozilla.components.support.base.log.logger.Logger
+import mozilla.components.support.ktx.android.org.json.asSequence
+import mozilla.components.support.ktx.android.org.json.toList
+import mozilla.components.support.remotesettings.RemoteSettingsClient
+import mozilla.components.support.remotesettings.RemoteSettingsResult
+import org.jetbrains.annotations.VisibleForTesting
+import org.json.JSONArray
+import org.json.JSONException
+import org.json.JSONObject
+import java.io.File
+
+internal const val REMOTE_PROD_ENDPOINT_URL = "https://firefox.settings.services.mozilla.com"
+internal const val REMOTE_ENDPOINT_BUCKET_NAME = "main"
+
+/**
+ * Parse SERP Telemetry json from remote config.
+ */
+class SerpTelemetryRepository(
+ rootStorageDirectory: File,
+ private val readJson: () -> JSONObject,
+ collectionName: String,
+ serverUrl: String = REMOTE_PROD_ENDPOINT_URL,
+ bucketName: String = REMOTE_ENDPOINT_BUCKET_NAME,
+) {
+ val logger = Logger("SerpTelemetryRepository")
+ private var providerList: List<SearchProviderModel> = emptyList()
+
+ @VisibleForTesting
+ internal var remoteSettingsClient = RemoteSettingsClient(
+ serverUrl = serverUrl,
+ bucketName = bucketName,
+ collectionName = collectionName,
+ storageRootDirectory = rootStorageDirectory,
+ )
+
+ /**
+ * Provides list of search providers from cache or dump and fetches from remotes server .
+ */
+ suspend fun updateProviderList(): List<SearchProviderModel> {
+ val (cacheLastModified, cacheResponse) = loadProvidersFromCache()
+ val localResponse = readJson()
+ if (cacheResponse.isEmpty() || cacheLastModified <= localResponse.getString("timestamp").toULong()) {
+ providerList = parseLocalPreinstalledData(localResponse)
+ } else if (cacheLastModified > localResponse.getString("timestamp").toULong()) {
+ providerList = cacheResponse
+ }
+ fetchRemoteResponse(cacheLastModified)
+ return providerList
+ }
+
+ @VisibleForTesting
+ internal suspend fun fetchRemoteResponse(
+ cacheLastModified: ULong?,
+ ) {
+ if (cacheLastModified == null) {
+ return
+ }
+ val remoteResponse = fetchRemoteResponse()
+ if (remoteResponse.lastModified > cacheLastModified) {
+ providerList = parseRemoteResponse(remoteResponse)
+ writeToCache(remoteResponse)
+ }
+ }
+
+ /**
+ * Writes data to local cache.
+ */
+ @VisibleForTesting
+ internal suspend fun writeToCache(records: RemoteSettingsResponse): RemoteSettingsResult {
+ return remoteSettingsClient.write(records)
+ }
+
+ /**
+ * Parses local json response.
+ */
+ @VisibleForTesting
+ internal fun parseLocalPreinstalledData(jsonObject: JSONObject): List<SearchProviderModel> {
+ return jsonObject.getJSONArray("data")
+ .asSequence()
+ .mapNotNull {
+ (it as JSONObject).toSearchProviderModel()
+ }
+ .toList()
+ }
+
+ /**
+ * Parses remote server response.
+ */
+ private fun parseRemoteResponse(response: RemoteSettingsResponse): List<SearchProviderModel> {
+ return response.records.mapNotNull {
+ it.fields.toSearchProviderModel()
+ }
+ }
+
+ /**
+ * Returns data from remote servers.
+ */
+ @VisibleForTesting
+ internal suspend fun fetchRemoteResponse(): RemoteSettingsResponse {
+ val result = remoteSettingsClient.fetch()
+ return if (result is RemoteSettingsResult.Success) {
+ result.response
+ } else {
+ RemoteSettingsResponse(emptyList(), 0u)
+ }
+ }
+
+ /**
+ * Returns search providers from local cache.
+ */
+ @VisibleForTesting
+ internal suspend fun loadProvidersFromCache(): Pair<ULong, List<SearchProviderModel>> {
+ val result = remoteSettingsClient.read()
+ return if (result is RemoteSettingsResult.Success) {
+ val response = result.response.records.mapNotNull {
+ it.fields.toSearchProviderModel()
+ }
+ val lastModified = result.response.lastModified
+ Pair(lastModified, response)
+ } else {
+ Pair(0u, emptyList())
+ }
+ }
+}
+
+@VisibleForTesting
+internal fun JSONObject.toSearchProviderModel(): SearchProviderModel? =
+ try {
+ SearchProviderModel(
+ schema = getLong("schema"),
+ taggedCodes = getJSONArray("taggedCodes").toList(),
+ telemetryId = optString("telemetryId"),
+ organicCodes = getJSONArray("organicCodes").toList(),
+ codeParamName = optString("codeParamName"),
+ followOnCookies = optJSONArray("followOnCookies")?.toListOfCookies(),
+ queryParamNames = optJSONArray("queryParamNames").toList(),
+ searchPageRegexp = optString("searchPageRegexp"),
+ adServerAttributes = optJSONArray("adServerAttributes").toList(),
+ followOnParamNames = optJSONArray("followOnParamNames")?.toList(),
+ extraAdServersRegexps = getJSONArray("extraAdServersRegexps").toList(),
+ expectedOrganicCodes = optJSONArray("expectedOrganicCodes")?.toList(),
+ )
+ } catch (e: JSONException) {
+ Logger("SerpTelemetryRepository").error("JSONException while trying to parse remote config", e)
+ null
+ }
+
+private fun JSONArray.toListOfCookies(): List<SearchProviderCookie> =
+ toList<JSONObject>().mapNotNull { jsonObject -> jsonObject.toSearchProviderCookie() }
+
+private fun JSONObject.toSearchProviderCookie(): SearchProviderCookie? =
+ try {
+ SearchProviderCookie(
+ extraCodeParamName = optString("extraCodeParamName"),
+ extraCodePrefixes = getJSONArray("extraCodePrefixes").toList(),
+ host = optString("host"),
+ name = optString("name"),
+ codeParamName = optString("codeParamName"),
+ )
+ } catch (e: JSONException) {
+ Logger("SerpTelemetryRepository").error("JSONException while trying to parse remote config", e)
+ null
+ }
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/TrackKeyInfo.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/TrackKeyInfo.kt
new file mode 100644
index 0000000000..c212c13b6f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/TrackKeyInfo.kt
@@ -0,0 +1,38 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+import java.util.Locale
+
+/**
+ * Key information about a Search Engine Result Page (SERP).
+ *
+ * @property provider The name of the search provider.
+ * @property type The search access point type (SAP). This is either "organic", "sap" or
+ * "sap-follow-on".
+ * @property code The search URL's `code` query parameter.
+ * @property channel The search URL's `channel` query parameter.
+ */
+internal data class TrackKeyInfo(
+ var provider: String,
+ var type: String,
+ var code: String?,
+ var channel: String? = null,
+) {
+ /**
+ * Returns the track key information into the following string format:
+ * `<provider>.in-content.[sap|sap-follow-on|organic].[code|none](.[channel])?`.
+ */
+ fun createTrackKey(): String {
+ return "${provider.lowercase(Locale.ROOT)}.in-content" +
+ ".${type.lowercase(Locale.ROOT)}" +
+ ".${code?.lowercase(Locale.ROOT) ?: "none"}" +
+ if (!channel?.lowercase(Locale.ROOT).isNullOrBlank()) {
+ ".${channel?.lowercase(Locale.ROOT)}"
+ } else {
+ ""
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/Utils.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/Utils.kt
new file mode 100644
index 0000000000..b541c83d5f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/Utils.kt
@@ -0,0 +1,116 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+import android.net.Uri
+import org.json.JSONObject
+
+private const val SEARCH_TYPE_SAP_FOLLOW_ON = "sap-follow-on"
+private const val SEARCH_TYPE_SAP = "sap"
+private const val SEARCH_TYPE_ORGANIC = "organic"
+private const val CHANNEL_KEY = "channel"
+private val validChannelSet = setOf("ts")
+
+/**
+ * Get a String in a specific format allowing to identify how an ads/search provider was used.
+ *
+ * @see [TrackKeyInfo.createTrackKey]
+ */
+@Suppress("NestedBlockDepth", "ComplexMethod")
+internal fun getTrackKey(
+ provider: SearchProviderModel,
+ uri: Uri,
+ cookies: List<JSONObject>,
+): String {
+ var type = SEARCH_TYPE_ORGANIC
+ val paramSet = uri.queryParameterNames
+ var code: String? = "none"
+
+ if (provider.codeParamName.isNotEmpty()) {
+ code = uri.getQueryParameter(provider.codeParamName)
+ if (code.isNullOrEmpty() &&
+ provider.telemetryId == "baidu" &&
+ uri.toString().contains("from=")
+ ) {
+ code = uri.toString().substringAfter("from=", "")
+ .substringBefore("/", "")
+ }
+ if (code != null) {
+ // The code is only included if it matches one of the specific ones.
+ if (provider.taggedCodes.contains(code)) {
+ type = SEARCH_TYPE_SAP
+ if (provider.followOnParamNames?.any { p -> paramSet.contains(p) } == true) {
+ type = SEARCH_TYPE_SAP_FOLLOW_ON
+ }
+ } else if (provider.organicCodes?.contains(code) == true) {
+ type = SEARCH_TYPE_ORGANIC
+ } else if (provider.expectedOrganicCodes?.contains(code) == true) {
+ code = "none"
+ } else {
+ code = "other"
+ }
+ } else if (provider.followOnCookies != null) {
+ // Try cookies first because Bing has followOnCookies and valid code, but no
+ // followOnParams => would track organic instead of sap-follow-on
+ getTrackKeyFromCookies(provider, uri, cookies)?.let {
+ return it.createTrackKey()
+ }
+ }
+
+ // For Bing if it didn't have a valid cookie and for all the other search engines
+ if (hasValidCode(uri.getQueryParameter(provider.codeParamName), provider)) {
+ var channel = uri.getQueryParameter(CHANNEL_KEY)
+
+ // For Bug 1751955
+ if (!validChannelSet.contains(channel)) {
+ channel = null
+ }
+ return TrackKeyInfo(provider.telemetryId, type, code, channel).createTrackKey()
+ }
+ }
+ return TrackKeyInfo(provider.telemetryId, type, code).createTrackKey()
+}
+
+private fun getTrackKeyFromCookies(
+ provider: SearchProviderModel,
+ uri: Uri,
+ cookies: List<JSONObject>,
+): TrackKeyInfo? {
+ // Especially Bing requires lots of extra work related to cookies.
+ provider.followOnCookies?.forEach { followOnCookie ->
+ val eCode = uri.getQueryParameter(followOnCookie.extraCodeParamName)
+
+ if (eCode == null || !followOnCookie.extraCodePrefixes.any { prefix ->
+ eCode.startsWith(prefix)
+ }
+ ) {
+ return@forEach
+ }
+
+ // If this cookie is present, it's probably an SAP follow-on.
+ // This might be an organic follow-on in the same session, but there
+ // is no way to tell the difference.
+ for (cookie in cookies) {
+ if (cookie.getString("name") != followOnCookie.name) {
+ continue
+ }
+ val valueList = cookie.getString("value")
+ .split("=")
+ .map { item -> item.trim() }
+
+ if (valueList.size == 2 && valueList[0] == followOnCookie.codeParamName &&
+ provider.taggedCodes.any { prefix ->
+ valueList[1] == prefix
+ }
+ ) {
+ return TrackKeyInfo(provider.telemetryId, SEARCH_TYPE_SAP_FOLLOW_ON, valueList[1])
+ }
+ }
+ }
+ return null
+}
+
+private fun hasValidCode(code: String?, provider: SearchProviderModel): Boolean =
+ code != null && provider.taggedCodes.any { prefix -> code == prefix }
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetry.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetry.kt
new file mode 100644
index 0000000000..810baeff67
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetry.kt
@@ -0,0 +1,124 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry.ads
+
+import android.net.Uri
+import androidx.annotation.VisibleForTesting
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.Engine
+import mozilla.components.feature.search.telemetry.BaseSearchTelemetry
+import mozilla.components.feature.search.telemetry.ExtensionInfo
+import mozilla.components.feature.search.telemetry.SearchProviderModel
+import mozilla.components.feature.search.telemetry.getTrackKey
+import mozilla.components.support.base.facts.Fact
+import mozilla.components.support.ktx.android.org.json.toList
+import org.json.JSONObject
+
+/**
+ * Telemetry for knowing how often users see/click ads in search and from which provider.
+ *
+ * Implemented as a browser extension based on the WebExtension API:
+ * https://developer.mozilla.org/en-US/docs/Mozilla/Add-ons/WebExtensions
+ */
+class AdsTelemetry : BaseSearchTelemetry() {
+
+ // SERP cached cookies used to check whether an ad was clicked.
+ @VisibleForTesting
+ internal var cachedCookies = listOf<JSONObject>()
+
+ override suspend fun install(
+ engine: Engine,
+ store: BrowserStore,
+ providerList: List<SearchProviderModel>,
+ ) {
+ val info = ExtensionInfo(
+ id = ADS_EXTENSION_ID,
+ resourceUrl = ADS_EXTENSION_RESOURCE_URL,
+ messageId = ADS_MESSAGE_ID,
+ )
+ installWebExtension(engine, store, info)
+ setProviderList(providerList)
+ }
+
+ override fun processMessage(message: JSONObject) {
+ // Cache the cookies list when the extension sends a message.
+ cachedCookies = message.getJSONArray(ADS_MESSAGE_COOKIES_KEY).toList()
+
+ val urls = message.getJSONArray(ADS_MESSAGE_DOCUMENT_URLS_KEY).toList<String>()
+ val uri = Uri.parse(message.getString(ADS_MESSAGE_SESSION_URL_KEY))
+ val provider = getProviderForUrl(message.getString(ADS_MESSAGE_SESSION_URL_KEY))
+
+ provider?.let {
+ if (it.containsAdLinks(urls)) {
+ emitFact(
+ SERP_SHOWN_WITH_ADDS,
+ getTrackKey(it, uri, cachedCookies),
+ )
+ }
+ }
+ }
+
+ /**
+ * To be called when the browser is navigating to a new URL, which may be a search ad.
+ *
+ * @param url The URL of the page before the search ad was clicked.
+ * This will be used to determine the originating search provider.
+ * @param urlPath A list of the URLs and load requests collected in between location changes.
+ * Clicking on a search ad generates a list of redirects from the originating search provider
+ * to the ad source. This is used to determine if there was an ad click.
+ */
+ @Suppress("ReturnCount")
+ fun checkIfAddWasClicked(url: String?, urlPath: List<String>) {
+ if (url == null) {
+ return
+ }
+ val uri = Uri.parse(url) ?: return
+ val provider = getProviderForUrl(url) ?: return
+ val paramSet = uri.queryParameterNames
+ val containsQueryParam = provider.queryParamNames?.any { paramSet.contains(it) }
+
+ if (containsQueryParam == false || !provider.containsAdLinks(urlPath)) {
+ // Do nothing if the URL does not have the search provider's query parameter or
+ // there were no ad clicks.
+ return
+ }
+
+ emitFact(
+ SERP_ADD_CLICKED,
+ getTrackKey(provider, uri, cachedCookies),
+ )
+ }
+
+ companion object {
+ /**
+ * [Fact] property indicating the user open a Search Engine Result Page
+ * of one of our search providers which contains ads.
+ */
+ const val SERP_SHOWN_WITH_ADDS = "SERP shown with adds"
+
+ /**
+ * [Fact] property indicating that an ad was clicked in a Search Engine Result Page.
+ */
+ const val SERP_ADD_CLICKED = "SERP add clicked"
+
+ @VisibleForTesting
+ internal const val ADS_EXTENSION_ID = "ads@mozac.org"
+
+ @VisibleForTesting
+ internal const val ADS_EXTENSION_RESOURCE_URL = "resource://android/assets/extensions/ads/"
+
+ @VisibleForTesting
+ internal const val ADS_MESSAGE_SESSION_URL_KEY = "url"
+
+ @VisibleForTesting
+ internal const val ADS_MESSAGE_DOCUMENT_URLS_KEY = "urls"
+
+ @VisibleForTesting
+ internal const val ADS_MESSAGE_COOKIES_KEY = "cookies"
+
+ @VisibleForTesting
+ internal const val ADS_MESSAGE_ID = "MozacBrowserAdsMessage"
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetry.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetry.kt
new file mode 100644
index 0000000000..5a888b6ebb
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetry.kt
@@ -0,0 +1,86 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry.incontent
+
+import android.net.Uri
+import androidx.annotation.VisibleForTesting
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.Engine
+import mozilla.components.feature.search.telemetry.BaseSearchTelemetry
+import mozilla.components.feature.search.telemetry.ExtensionInfo
+import mozilla.components.feature.search.telemetry.SearchProviderModel
+import mozilla.components.feature.search.telemetry.getTrackKey
+import mozilla.components.support.base.facts.Fact
+import mozilla.components.support.ktx.android.org.json.toList
+import org.json.JSONObject
+
+/**
+ * Telemetry for knowing of in-web-content searches (including follow-on searches) and the provider used.
+ *
+ * Implemented as a browser extension based on the WebExtension API:
+ * https://developer.mozilla.org/en-US/docs/Mozilla/Add-ons/WebExtensions
+ */
+class InContentTelemetry : BaseSearchTelemetry() {
+
+ override suspend fun install(
+ engine: Engine,
+ store: BrowserStore,
+ providerList: List<SearchProviderModel>,
+ ) {
+ val info = ExtensionInfo(
+ id = SEARCH_EXTENSION_ID,
+ resourceUrl = SEARCH_EXTENSION_RESOURCE_URL,
+ messageId = SEARCH_MESSAGE_ID,
+ )
+ installWebExtension(engine, store, info)
+ setProviderList(providerList)
+ }
+
+ /**
+ * Processes a message containing search-related information.
+ */
+ override fun processMessage(message: JSONObject) {
+ val cookies = message.getJSONArray(SEARCH_MESSAGE_LIST_KEY).toList<JSONObject>()
+ trackPartnerUrlTypeMetric(message.getString(SEARCH_MESSAGE_SESSION_URL_KEY), cookies)
+ }
+
+ @VisibleForTesting
+ internal fun trackPartnerUrlTypeMetric(url: String, cookies: List<JSONObject>) {
+ val provider = getProviderForUrl(url) ?: return
+ val uri = Uri.parse(url)
+ val paramSet = uri.queryParameterNames
+ val containsQueryParam = provider.queryParamNames?.any { paramSet.contains(it) }
+ if (containsQueryParam == false) {
+ return
+ }
+ emitFact(
+ IN_CONTENT_SEARCH,
+ getTrackKey(provider, uri, cookies),
+ )
+ }
+
+ companion object {
+ /**
+ * [Fact] property indicating that the user did a search, be it a new one
+ * or continuing from an existing search.
+ */
+ const val IN_CONTENT_SEARCH = "in content search"
+
+ @VisibleForTesting
+ internal const val SEARCH_EXTENSION_ID = "cookies@mozac.org"
+
+ @VisibleForTesting
+ internal const val SEARCH_EXTENSION_RESOURCE_URL = "resource://android/assets/extensions/search/"
+
+ @VisibleForTesting
+ internal const val SEARCH_MESSAGE_SESSION_URL_KEY = "url"
+
+ @VisibleForTesting
+ internal const val SEARCH_MESSAGE_LIST_KEY = "cookies"
+
+ @VisibleForTesting
+ internal const val SEARCH_MESSAGE_ID = "MozacBrowserSearchMessage"
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/AppSearchWidgetProvider.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/AppSearchWidgetProvider.kt
new file mode 100644
index 0000000000..62a4e2d83e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/AppSearchWidgetProvider.kt
@@ -0,0 +1,312 @@
+/* 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/. */
+
+package mozilla.components.feature.search.widget
+
+import android.app.PendingIntent
+import android.appwidget.AppWidgetManager
+import android.appwidget.AppWidgetManager.OPTION_APPWIDGET_MIN_WIDTH
+import android.appwidget.AppWidgetProvider
+import android.content.ComponentName
+import android.content.Context
+import android.content.Intent
+import android.os.Build
+import android.os.Bundle
+import android.view.View
+import android.widget.RemoteViews
+import androidx.annotation.Dimension
+import androidx.annotation.Dimension.Companion.DP
+import androidx.annotation.VisibleForTesting
+import androidx.appcompat.content.res.AppCompatResources
+import androidx.core.graphics.drawable.toBitmap
+import mozilla.components.feature.search.R
+import mozilla.components.feature.search.widget.BaseVoiceSearchActivity.Companion.SPEECH_PROCESSING
+import mozilla.components.support.utils.PendingIntentUtils
+
+/**
+ * An abstract [AppWidgetProvider] that implements core behaviour needed to support a Search Widget
+ * on the launcher.
+ */
+abstract class AppSearchWidgetProvider : AppWidgetProvider() {
+
+ override fun onUpdate(
+ context: Context,
+ appWidgetManager: AppWidgetManager,
+ appWidgetIds: IntArray,
+ ) {
+ val textSearchIntent = createTextSearchIntent(context)
+ val voiceSearchIntent = createVoiceSearchIntent(context)
+
+ appWidgetIds.forEach { appWidgetId ->
+ updateWidgetLayout(
+ context = context,
+ appWidgetId = appWidgetId,
+ appWidgetManager = appWidgetManager,
+ voiceSearchIntent = voiceSearchIntent,
+ textSearchIntent = textSearchIntent,
+ )
+ }
+ }
+
+ override fun onAppWidgetOptionsChanged(
+ context: Context,
+ appWidgetManager: AppWidgetManager,
+ appWidgetId: Int,
+ newOptions: Bundle?,
+ ) {
+ val textSearchIntent = createTextSearchIntent(context)
+ val voiceSearchIntent = createVoiceSearchIntent(context)
+
+ updateWidgetLayout(
+ context = context,
+ appWidgetId = appWidgetId,
+ appWidgetManager = appWidgetManager,
+ voiceSearchIntent = voiceSearchIntent,
+ textSearchIntent = textSearchIntent,
+ )
+ }
+
+ /**
+ * Builds pending intent that opens the browser and starts a new text search.
+ */
+ abstract fun createTextSearchIntent(context: Context): PendingIntent
+
+ /**
+ * If the microphone will appear on the Search Widget and the user can perform a voice search.
+ */
+ abstract fun shouldShowVoiceSearch(context: Context): Boolean
+
+ /**
+ * Activity that extends BaseVoiceSearchActivity.
+ */
+ abstract fun voiceSearchActivity(): Class<out BaseVoiceSearchActivity>
+
+ /**
+ * Config that sets the icons and the strings for search widget.
+ */
+ abstract val config: SearchWidgetConfig
+
+ /**
+ * Builds pending intent that starts a new voice search.
+ */
+ @VisibleForTesting
+ internal fun createVoiceSearchIntent(context: Context): PendingIntent? {
+ if (!shouldShowVoiceSearch(context)) {
+ return null
+ }
+
+ val voiceIntent = Intent(context, voiceSearchActivity()).apply {
+ flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK
+ putExtra(SPEECH_PROCESSING, true)
+ }
+
+ return PendingIntent.getActivity(
+ context,
+ REQUEST_CODE_VOICE,
+ voiceIntent,
+ PendingIntentUtils.defaultFlags,
+ )
+ }
+
+ private fun updateWidgetLayout(
+ context: Context,
+ appWidgetId: Int,
+ appWidgetManager: AppWidgetManager,
+ voiceSearchIntent: PendingIntent?,
+ textSearchIntent: PendingIntent,
+ ) {
+ val currentWidth =
+ appWidgetManager.getAppWidgetOptions(appWidgetId).getInt(OPTION_APPWIDGET_MIN_WIDTH)
+ val layoutSize = getLayoutSize(currentWidth)
+ // It's not enough to just hide the microphone on the "small" sized widget due to its design.
+ // The "small" widget needs a complete redesign, meaning it needs a new layout file.
+ val showMic = (voiceSearchIntent != null)
+ val layout = getLayout(layoutSize, showMic)
+ val text = getText(layoutSize, context)
+
+ val views =
+ createRemoteViews(context, layout, textSearchIntent, voiceSearchIntent, text)
+ appWidgetManager.updateAppWidget(appWidgetId, views)
+ }
+
+ private fun createRemoteViews(
+ context: Context,
+ layout: Int,
+ textSearchIntent: PendingIntent,
+ voiceSearchIntent: PendingIntent?,
+ text: String?,
+ ): RemoteViews {
+ return RemoteViews(context.packageName, layout).apply {
+ setSearchWidgetIcon(context)
+ setMicrophoneIcon(context)
+ when (layout) {
+ R.layout.mozac_search_widget_extra_small_v1,
+ R.layout.mozac_search_widget_extra_small_v2,
+ R.layout.mozac_search_widget_small_no_mic,
+ -> {
+ setOnClickPendingIntent(
+ R.id.mozac_button_search_widget_new_tab,
+ textSearchIntent,
+ )
+ }
+ R.layout.mozac_search_widget_small -> {
+ setOnClickPendingIntent(
+ R.id.mozac_button_search_widget_new_tab,
+ textSearchIntent,
+ )
+ setOnClickPendingIntent(
+ R.id.mozac_button_search_widget_voice,
+ voiceSearchIntent,
+ )
+ }
+ R.layout.mozac_search_widget_medium,
+ R.layout.mozac_search_widget_large,
+ -> {
+ setOnClickPendingIntent(
+ R.id.mozac_button_search_widget_new_tab,
+ textSearchIntent,
+ )
+ setOnClickPendingIntent(
+ R.id.mozac_button_search_widget_voice,
+ voiceSearchIntent,
+ )
+ setOnClickPendingIntent(
+ R.id.mozac_button_search_widget_new_tab_icon,
+ textSearchIntent,
+ )
+ setTextViewText(R.id.mozac_button_search_widget_new_tab, text)
+
+ // Unlike "small" widget, "medium" and "large" sizes do not have separate layouts
+ // that exclude the microphone icon, which is why we must hide it accordingly here.
+ if (voiceSearchIntent == null) {
+ setViewVisibility(R.id.mozac_button_search_widget_voice, View.GONE)
+ }
+ }
+ }
+ }
+ }
+
+ private fun RemoteViews.setMicrophoneIcon(context: Context) {
+ setImageView(
+ context,
+ R.id.mozac_button_search_widget_voice,
+ config.searchWidgetMicrophoneResource,
+ )
+ }
+
+ private fun RemoteViews.setSearchWidgetIcon(context: Context) {
+ setImageView(
+ context,
+ R.id.mozac_button_search_widget_new_tab_icon,
+ config.searchWidgetIconResource,
+ )
+ val appName = context.getString(config.appName)
+ setContentDescription(
+ R.id.mozac_button_search_widget_new_tab_icon,
+ context.getString(R.string.search_widget_content_description, appName),
+ )
+ }
+
+ private fun RemoteViews.setImageView(context: Context, viewId: Int, resourceId: Int) {
+ // gradient color available for android:fillColor only on SDK 24+
+ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
+ setImageViewResource(
+ viewId,
+ resourceId,
+ )
+ } else {
+ setImageViewBitmap(
+ viewId,
+ AppCompatResources.getDrawable(
+ context,
+ resourceId,
+ )?.toBitmap(),
+ )
+ }
+ }
+
+ // Cell sizes obtained from the actual dimensions listed in search widget specs.
+ companion object {
+ private const val DP_EXTRA_SMALL = 64
+ private const val DP_SMALL = 100
+ private const val DP_MEDIUM = 192
+ private const val DP_LARGE = 256
+ private const val REQUEST_CODE_VOICE = 1
+
+ /**
+ * It updates AppSearchWidgetProvider size and microphone icon visibility.
+ */
+ fun updateAllWidgets(context: Context, clazz: Class<out AppSearchWidgetProvider>) {
+ val widgetManager = AppWidgetManager.getInstance(context)
+ val widgetIds = widgetManager.getAppWidgetIds(
+ ComponentName(
+ context,
+ clazz,
+ ),
+ )
+ if (widgetIds.isNotEmpty()) {
+ context.sendBroadcast(
+ Intent(context, clazz).apply {
+ action = AppWidgetManager.ACTION_APPWIDGET_UPDATE
+ putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, widgetIds)
+ },
+ )
+ }
+ }
+
+ @VisibleForTesting
+ internal fun getLayoutSize(@Dimension(unit = DP) dp: Int) = when {
+ dp >= DP_LARGE -> SearchWidgetProviderSize.LARGE
+ dp >= DP_MEDIUM -> SearchWidgetProviderSize.MEDIUM
+ dp >= DP_SMALL -> SearchWidgetProviderSize.SMALL
+ dp >= DP_EXTRA_SMALL -> SearchWidgetProviderSize.EXTRA_SMALL_V2
+ else -> SearchWidgetProviderSize.EXTRA_SMALL_V1
+ }
+
+ /**
+ * Get the layout resource to use for the search widget.
+ */
+ @VisibleForTesting
+ internal fun getLayout(size: SearchWidgetProviderSize, showMic: Boolean) = when (size) {
+ SearchWidgetProviderSize.LARGE -> R.layout.mozac_search_widget_large
+ SearchWidgetProviderSize.MEDIUM -> R.layout.mozac_search_widget_medium
+ SearchWidgetProviderSize.SMALL -> {
+ if (showMic) {
+ R.layout.mozac_search_widget_small
+ } else {
+ R.layout.mozac_search_widget_small_no_mic
+ }
+ }
+ SearchWidgetProviderSize.EXTRA_SMALL_V2 -> R.layout.mozac_search_widget_extra_small_v2
+ SearchWidgetProviderSize.EXTRA_SMALL_V1 -> R.layout.mozac_search_widget_extra_small_v1
+ }
+
+ /**
+ * Get the text to place in the search widget.
+ */
+ @VisibleForTesting
+ internal fun getText(layout: SearchWidgetProviderSize, context: Context) = when (layout) {
+ SearchWidgetProviderSize.MEDIUM -> context.getString(R.string.search_widget_text_short)
+ SearchWidgetProviderSize.LARGE -> context.getString(R.string.search_widget_text_long)
+ else -> null
+ }
+ }
+}
+
+/**
+ * Client App can set from this config icons and the app name for search widget.
+ */
+data class SearchWidgetConfig(
+ val searchWidgetIconResource: Int,
+ val searchWidgetMicrophoneResource: Int,
+ val appName: Int,
+)
+
+internal enum class SearchWidgetProviderSize {
+ EXTRA_SMALL_V1,
+ EXTRA_SMALL_V2,
+ SMALL,
+ MEDIUM,
+ LARGE,
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivity.kt b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivity.kt
new file mode 100644
index 0000000000..1c70f9809d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivity.kt
@@ -0,0 +1,134 @@
+/* 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/. */
+
+package mozilla.components.feature.search.widget
+
+import android.app.Activity
+import android.content.ActivityNotFoundException
+import android.content.Intent
+import android.os.Bundle
+import android.speech.RecognizerIntent
+import androidx.activity.result.ActivityResult
+import androidx.activity.result.ActivityResultLauncher
+import androidx.activity.result.contract.ActivityResultContracts
+import androidx.annotation.VisibleForTesting
+import androidx.appcompat.app.AppCompatActivity
+import mozilla.components.support.base.log.logger.Logger
+import mozilla.components.support.utils.ext.getParcelableCompat
+import java.util.Locale
+
+/**
+ * Launches voice recognition then uses it to start a new web search.
+ */
+abstract class BaseVoiceSearchActivity : AppCompatActivity() {
+
+ /**
+ * Holds the intent that initially started this activity
+ * so that it can persist through the speech activity.
+ */
+ private var previousIntent: Intent? = null
+
+ private var activityResultLauncher: ActivityResultLauncher<Intent> = getActivityResultLauncher()
+
+ override fun onSaveInstanceState(outState: Bundle) {
+ super.onSaveInstanceState(outState)
+ outState.putParcelable(PREVIOUS_INTENT, previousIntent)
+ }
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ super.onCreate(savedInstanceState)
+ // Retrieve the previous intent from the saved state
+ previousIntent = savedInstanceState?.getParcelableCompat(PREVIOUS_INTENT, Intent::class.java)
+ if (previousIntent.isForSpeechProcessing()) {
+ // Don't reopen the speech recognizer
+ return
+ }
+
+ // The intent property is nullable, but the rest of the code below assumes it is not.
+ val intent = intent?.let { Intent(intent) } ?: Intent()
+ if (intent.isForSpeechProcessing()) {
+ previousIntent = intent
+ displaySpeechRecognizer()
+ } else {
+ finish()
+ }
+ }
+
+ /**
+ * Language locale for Voice Search.
+ */
+ abstract fun getCurrentLocale(): Locale
+
+ /**
+ * Speech recognizer popup is shown.
+ */
+ abstract fun onSpeechRecognitionStarted()
+
+ /**
+ * Start intent after voice search ,for example a browser page is open with the spokenText.
+ * @param spokenText what the user voice search
+ */
+ abstract fun onSpeechRecognitionEnded(spokenText: String)
+
+ @VisibleForTesting
+ internal fun activityResultImplementation(activityResult: ActivityResult) {
+ if (activityResult.resultCode == Activity.RESULT_OK) {
+ val spokenText =
+ activityResult.data?.getStringArrayListExtra(RecognizerIntent.EXTRA_RESULTS)
+ ?.first()
+ previousIntent?.apply {
+ spokenText?.let { onSpeechRecognitionEnded(it) }
+ }
+ }
+ finish()
+ }
+
+ private fun getActivityResultLauncher(): ActivityResultLauncher<Intent> {
+ return registerForActivityResult(
+ ActivityResultContracts.StartActivityForResult(),
+ ) {
+ activityResultImplementation(it)
+ }
+ }
+
+ /**
+ * Displays a speech recognizer popup that listens for input from the user.
+ */
+ private fun displaySpeechRecognizer() {
+ val intentSpeech = Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH).apply {
+ putExtra(
+ RecognizerIntent.EXTRA_LANGUAGE_MODEL,
+ RecognizerIntent.LANGUAGE_MODEL_FREE_FORM,
+ )
+ putExtra(
+ RecognizerIntent.EXTRA_LANGUAGE,
+ getCurrentLocale(),
+ )
+ }
+ onSpeechRecognitionStarted()
+ try {
+ activityResultLauncher.launch(intentSpeech)
+ } catch (e: ActivityNotFoundException) {
+ Logger(TAG).error("ActivityNotFoundException " + e.message.toString())
+ finish()
+ }
+ }
+
+ /**
+ * Returns true if the [SPEECH_PROCESSING] extra is present and set to true.
+ * Returns false if the intent is null.
+ */
+ private fun Intent?.isForSpeechProcessing(): Boolean =
+ this?.getBooleanExtra(SPEECH_PROCESSING, false) == true
+
+ companion object {
+ const val PREVIOUS_INTENT = "org.mozilla.components.previous_intent"
+
+ /**
+ * In [BaseVoiceSearchActivity] activity, used to store if the speech processing should start.
+ */
+ const val SPEECH_PROCESSING = "speech_processing"
+ const val TAG = "BaseVoiceSearchActivity"
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/drawable/mozac_rounded_search_widget_background.xml b/mobile/android/android-components/components/feature/search/src/main/res/drawable/mozac_rounded_search_widget_background.xml
new file mode 100644
index 0000000000..fd61fe2557
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/drawable/mozac_rounded_search_widget_background.xml
@@ -0,0 +1,9 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+
+<shape xmlns:android="http://schemas.android.com/apk/res/android">
+ <solid android:color="@color/mozac_feature_search_widget_background_color" />
+ <corners android:radius="@dimen/mozac_tab_corner_radius"/>
+</shape>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v1.xml b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v1.xml
new file mode 100644
index 0000000000..ee95021031
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v1.xml
@@ -0,0 +1,20 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ android:id="@id/mozac_button_search_widget_new_tab"
+ android:layout_width="match_parent"
+ android:layout_height="50dp"
+ android:layout_gravity="center"
+ android:background="@drawable/mozac_rounded_search_widget_background">
+
+ <ImageView
+ android:id="@+id/mozac_button_search_widget_new_tab_icon"
+ android:layout_width="50dp"
+ android:layout_height="50dp"
+ android:layout_gravity="center"
+ android:contentDescription="@string/search_widget_content_description"
+ android:scaleType="centerInside" />
+
+</FrameLayout>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v2.xml b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v2.xml
new file mode 100644
index 0000000000..468df05b75
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_extra_small_v2.xml
@@ -0,0 +1,19 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ android:id="@id/mozac_button_search_widget_new_tab"
+ android:layout_width="match_parent"
+ android:layout_height="50dp"
+ android:layout_gravity="center"
+ android:background="@drawable/mozac_rounded_search_widget_background">
+
+ <ImageView
+ android:id="@+id/mozac_button_search_widget_new_tab_icon"
+ android:layout_width="50dp"
+ android:layout_height="50dp"
+ android:layout_gravity="center"
+ android:contentDescription="@string/search_widget_content_description"
+ android:scaleType="centerInside" />
+</FrameLayout>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_large.xml b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_large.xml
new file mode 100644
index 0000000000..7801adb460
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_large.xml
@@ -0,0 +1,45 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ xmlns:tools="http://schemas.android.com/tools"
+ android:layout_width="match_parent"
+ android:layout_height="50dp"
+ android:layout_gravity="center"
+ android:background="@drawable/mozac_rounded_search_widget_background">
+
+ <ImageView
+ android:id="@+id/mozac_button_search_widget_new_tab_icon"
+ android:layout_width="50dp"
+ android:layout_height="50dp"
+ android:layout_alignParentStart="true"
+ android:contentDescription="@string/search_widget_content_description"
+ android:scaleType="centerInside" />
+
+ <TextView
+ android:id="@+id/mozac_button_search_widget_new_tab"
+ android:layout_width="match_parent"
+ android:layout_height="match_parent"
+ android:layout_alignEnd="@id/mozac_button_search_widget_voice"
+ android:layout_marginStart="9dp"
+ android:layout_toEndOf="@id/mozac_button_search_widget_new_tab_icon"
+ android:gravity="start|center_vertical"
+ android:letterSpacing="-0.025"
+ android:textAlignment="viewStart"
+ android:textColor="@color/mozac_feature_search_widget_text_color"
+ android:textSize="15sp"
+ tools:text="Search the web" />
+
+ <ImageButton
+ android:id="@+id/mozac_button_search_widget_voice"
+ android:layout_width="48dp"
+ android:layout_height="48dp"
+ android:layout_alignParentEnd="true"
+ android:background="@android:color/transparent"
+ android:layout_centerVertical="true"
+ android:padding="8dp"
+ android:layout_marginEnd="1dp"
+ android:contentDescription="@string/search_widget_voice" />
+
+</RelativeLayout>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_medium.xml b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_medium.xml
new file mode 100644
index 0000000000..5cbcdffd84
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_medium.xml
@@ -0,0 +1,45 @@
+<?xml version="1.0" encoding="utf-8"?><!-- 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/. -->
+
+<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ xmlns:tools="http://schemas.android.com/tools"
+ android:layout_width="match_parent"
+ android:layout_height="50dp"
+ android:layout_gravity="center"
+ android:background="@drawable/mozac_rounded_search_widget_background">
+
+ <ImageView
+ android:id="@+id/mozac_button_search_widget_new_tab_icon"
+ android:layout_width="50dp"
+ android:layout_height="50dp"
+ android:layout_alignParentStart="true"
+ android:contentDescription="@string/search_widget_content_description"
+ android:scaleType="centerInside" />
+
+ <TextView
+ android:id="@+id/mozac_button_search_widget_new_tab"
+ android:layout_width="match_parent"
+ android:layout_height="match_parent"
+ android:layout_alignEnd="@id/mozac_button_search_widget_voice"
+ android:layout_marginStart="9dp"
+ android:layout_toEndOf="@id/mozac_button_search_widget_new_tab_icon"
+ android:gravity="start|center_vertical"
+ android:letterSpacing="-0.025"
+ android:textAlignment="viewStart"
+ android:textColor="@color/mozac_feature_search_widget_text_color"
+ android:textSize="15sp"
+ tools:text="Search" />
+
+ <ImageButton
+ android:id="@+id/mozac_button_search_widget_voice"
+ android:layout_width="48dp"
+ android:layout_height="48dp"
+ android:layout_alignParentEnd="true"
+ android:layout_centerVertical="true"
+ android:padding="8dp"
+ android:background="@android:color/transparent"
+ android:layout_marginEnd="1dp"
+ android:contentDescription="@string/search_widget_voice" />
+
+</RelativeLayout>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small.xml b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small.xml
new file mode 100644
index 0000000000..c75217e90f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small.xml
@@ -0,0 +1,27 @@
+<?xml version="1.0" encoding="utf-8"?><!-- 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/. -->
+<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="match_parent"
+ android:layout_height="50dp"
+ android:background="@drawable/mozac_rounded_search_widget_background" >
+
+ <ImageView
+ android:id="@+id/mozac_button_search_widget_new_tab_icon"
+ android:layout_alignParentStart="true"
+ android:layout_width="50dp"
+ android:layout_height="50dp"
+ android:contentDescription="@string/search_widget_content_description"
+ android:scaleType="centerInside" />
+
+ <ImageView
+ android:id="@+id/mozac_button_search_widget_voice"
+ android:layout_alignParentEnd="true"
+ android:layout_width="50dp"
+ android:layout_height="50dp"
+ android:contentDescription="@string/search_widget_voice"
+ android:padding="10dp"
+ android:background="@android:color/transparent"
+ android:scaleType="centerInside" />
+
+</RelativeLayout>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small_no_mic.xml b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small_no_mic.xml
new file mode 100644
index 0000000000..61d2152b4b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/layout/mozac_search_widget_small_no_mic.xml
@@ -0,0 +1,19 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ android:id="@id/mozac_button_search_widget_new_tab"
+ android:layout_width="match_parent"
+ android:layout_height="50dp"
+ android:layout_gravity="center"
+ android:background="@drawable/mozac_rounded_search_widget_background"
+ android:orientation="vertical">
+
+ <ImageView
+ android:id="@+id/mozac_button_search_widget_new_tab_icon"
+ android:layout_width="40dp"
+ android:layout_height="40dp"
+ android:layout_gravity="center"
+ android:contentDescription="@string/search_widget_content_description" />
+</FrameLayout>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-am/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-am/strings.xml
new file mode 100644
index 0000000000..05a821675e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-am/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">አዲስ %1$s ትር ክፈት</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ፈልግ</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ድሩን ይፈልጉ</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">የድምጽ ፍለጋ</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ar/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ar/strings.xml
new file mode 100644
index 0000000000..d75141a022
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ar/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">افتح لسان %1$s جديد</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ابحث</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ابحث في الوِب</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">البحث الصوتي</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ast/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ast/strings.xml
new file mode 100644
index 0000000000..dab4e3690a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ast/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir nuna llingüeta de %1$s nueva</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Buscar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Buscar na web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Busca pela voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-azb/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-azb/strings.xml
new file mode 100644
index 0000000000..dad76ea236
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-azb/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">یئنی بیر %1$s تاغی آچین</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">آختاریش</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">وب‌ده آختار</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">سس‌لی آختاریش</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-be/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-be/strings.xml
new file mode 100644
index 0000000000..d01b3ed8ac
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-be/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Адкрыць новую картку %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Пошук</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Пошук у інтэрнэце</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Галасавы пошук</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-bg/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-bg/strings.xml
new file mode 100644
index 0000000000..62aa6628d4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-bg/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Отваряне на раздел с %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Търсене</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Търсене в интернет</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Гласово търсене</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-br/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-br/strings.xml
new file mode 100644
index 0000000000..efdc6aa03a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-br/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Digeriñ un ivinell %1$s nevez</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Klask</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Klask er web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Klask dre vouezh</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-bs/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-bs/strings.xml
new file mode 100644
index 0000000000..8d89c63779
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-bs/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Otvori novi %1$s tab</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Traži</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Pretraži web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Glasovna pretraga</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ca/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ca/strings.xml
new file mode 100644
index 0000000000..1821b2b9f4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ca/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Obre en una pestanya nova en %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Cerca</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Cerca al web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Cerca per veu</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-cak/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-cak/strings.xml
new file mode 100644
index 0000000000..c4630b9637
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-cak/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Tijaq jun k\'ak\'a\' %1$s ruwi\'</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Tikanöx</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Tikanöx pan ajk\'amaya\'l</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Tikanöx chi ch\'ab\'äl</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ckb/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ckb/strings.xml
new file mode 100644
index 0000000000..4f31f2232f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ckb/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">بازدەرێکی %1$sی نوێ بکەرەوە</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">گەڕان</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">بە وێبدا بگەڕێ</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">گەڕانی دەنگی</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-co/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-co/strings.xml
new file mode 100644
index 0000000000..a38dd9e634
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-co/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Apre una nova unghjetta in %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Ricercà</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Ricercà nant’à u web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ricerca vucale</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-cs/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-cs/strings.xml
new file mode 100644
index 0000000000..b889d8f9a6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-cs/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Otevřít nový panel v aplikaci %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Hledat</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Vyhledat na webu</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Hlasové vyhledávání</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-cy/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-cy/strings.xml
new file mode 100644
index 0000000000..5c8f73d4bc
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-cy/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Agor tab %1$s newydd</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Chwilio</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Chwilio’r we</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Chwilio llais</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-da/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-da/strings.xml
new file mode 100644
index 0000000000..a8af438d90
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-da/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Åbn et nyt %1$s-faneblad</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Søg</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Søg på nettet</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Stemme-søgning</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-de/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-de/strings.xml
new file mode 100644
index 0000000000..0b98201168
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-de/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Neuen %1$s-Tab öffnen</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Suche</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Das Web durchsuchen</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Sprachsuche</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-dsb/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-dsb/strings.xml
new file mode 100644
index 0000000000..c5b19b7f83
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-dsb/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Nowy rejtarik %1$s wócyniś</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Pytaś</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Web pśepytaś</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Głosowe pytanje</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-el/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-el/strings.xml
new file mode 100644
index 0000000000..d5ef9aeb67
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-el/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Άνοιγμα νέας καρτέλας %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Αναζήτηση</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Αναζήτηση στο διαδίκτυο</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Φωνητική αναζήτηση</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-en-rCA/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-en-rCA/strings.xml
new file mode 100644
index 0000000000..93a653aed7
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-en-rCA/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Open a new %1$s tab</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Search</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Search the web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Voice search</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-en-rGB/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-en-rGB/strings.xml
new file mode 100644
index 0000000000..93a653aed7
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-en-rGB/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Open a new %1$s tab</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Search</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Search the web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Voice search</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-eo/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-eo/strings.xml
new file mode 100644
index 0000000000..5ab8dc03ec
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-eo/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Malfermi novan langeton de %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Serĉi</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Serĉi en la reto</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Voĉa serĉo</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-es-rAR/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rAR/strings.xml
new file mode 100644
index 0000000000..f68109e762
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rAR/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir una nueva pestaña de %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Buscar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Buscar en la web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Búsqueda por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-es-rCL/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rCL/strings.xml
new file mode 100644
index 0000000000..f68109e762
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rCL/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir una nueva pestaña de %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Buscar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Buscar en la web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Búsqueda por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-es-rES/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rES/strings.xml
new file mode 100644
index 0000000000..60b93ee599
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rES/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir una pestaña nueva de %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Buscar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Buscar en la web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Búsqueda por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-es-rMX/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rMX/strings.xml
new file mode 100644
index 0000000000..3fb6e2091c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-es-rMX/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir una nueva pestaña en %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Buscar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Buscar en la web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Búsqueda por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-es/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-es/strings.xml
new file mode 100644
index 0000000000..60b93ee599
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-es/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir una pestaña nueva de %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Buscar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Buscar en la web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Búsqueda por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-et/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-et/strings.xml
new file mode 100644
index 0000000000..077b8a047c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-et/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Ava uus kaart %1$sis</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Otsing</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Otsi veebist</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Häälotsing</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-eu/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-eu/strings.xml
new file mode 100644
index 0000000000..9550d75f1f
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-eu/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Ireki %1$s fitxa berria</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Bilatu</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Bilatu webean</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ahots bidezko bilaketa</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-fa/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-fa/strings.xml
new file mode 100644
index 0000000000..9c2489d604
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-fa/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">گشودن یک زبانهٔ جدید %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">جست‌وجو</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">جست‌وجوی وب</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">جست‌وجوی صوتی</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ff/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ff/strings.xml
new file mode 100644
index 0000000000..c22e74b613
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ff/strings.xml
@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Uddit tabbere hesre %1$s</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Njiilaw sawto</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-fi/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-fi/strings.xml
new file mode 100644
index 0000000000..ab054f5e6e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-fi/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Avaa uusi %1$s-välilehti</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Hae</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Hae verkosta</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Äänihaku</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-fr/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-fr/strings.xml
new file mode 100644
index 0000000000..4785d484bd
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-fr/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Ouvrir un nouvel onglet %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Rechercher</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Rechercher sur le Web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Recherche vocale</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-fur/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-fur/strings.xml
new file mode 100644
index 0000000000..29ec17dd5b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-fur/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Vierç intune gnove schede in %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Cîr</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Cîr tal web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ricercje vocâl</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-fy-rNL/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-fy-rNL/strings.xml
new file mode 100644
index 0000000000..3fd2614b8a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-fy-rNL/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">In nij %1$s-ljepblêd iepenje</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Sykje</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Sykje op it web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Sprutsen sykopdracht</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-gd/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-gd/strings.xml
new file mode 100644
index 0000000000..9bf121ff5c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-gd/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Fosgail taba %1$s ùr</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Lorg</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Lorg air an lìon</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Lorg-gutha</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-gl/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-gl/strings.xml
new file mode 100644
index 0000000000..3915de9a04
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-gl/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir unha nova lapela en %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Buscar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Buscar na web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Busca por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-gn/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-gn/strings.xml
new file mode 100644
index 0000000000..7dc6496f36
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-gn/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Embojuruja tendayke pyahu %1$s-pe</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Heka</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Eheka ñandutípe</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ayvu rupi jeheka</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-hr/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-hr/strings.xml
new file mode 100644
index 0000000000..a5a05a0f6a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-hr/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Otvori novu %1$s karticu</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Traži</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Pretraži web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Glasovno pretraživanje</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-hsb/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-hsb/strings.xml
new file mode 100644
index 0000000000..56a86b500b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-hsb/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Nowy rajtark %1$s wočinić</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Pytać</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Web přepytać</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Hłosowe pytanje</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-hu/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-hu/strings.xml
new file mode 100644
index 0000000000..ec925efad6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-hu/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Új %1$s lap megnyitása</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Keresés</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Keresés a weben</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Hangalapú keresés</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-hy-rAM/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-hy-rAM/strings.xml
new file mode 100644
index 0000000000..c069279db8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-hy-rAM/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Բացել նոր %1$s ներդիր</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Որոնում</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Որոնել համացանցում</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ձայնային որոնում</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ia/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ia/strings.xml
new file mode 100644
index 0000000000..89c47c731e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ia/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Aperir un nove scheda %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Cercar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Cercar in le web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Recerca vocal</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-in/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-in/strings.xml
new file mode 100644
index 0000000000..19650ec7ec
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-in/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Buka di tab %1$s baru</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Cari</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Cari di Web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Pencarian suara</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-is/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-is/strings.xml
new file mode 100644
index 0000000000..ba5965b35c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-is/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Opna nýjan %1$s-flipa</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Leita</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Leita á vefnum</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Raddleit</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-it/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-it/strings.xml
new file mode 100644
index 0000000000..ccc30fb548
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-it/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Apri una nuova scheda in %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Cerca</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Cerca sul Web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ricerca vocale</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-iw/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-iw/strings.xml
new file mode 100644
index 0000000000..cb0f9346ac
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-iw/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">פתיחת לשונית %1$s חדשה</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">חיפוש</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">חיפוש ברשת</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">חיפוש קולי</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ja/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ja/strings.xml
new file mode 100644
index 0000000000..bc9d0bf5b9
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ja/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">%1$s の新しいタブで開く</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">検索</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ウェブを検索</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">音声検索</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ka/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ka/strings.xml
new file mode 100644
index 0000000000..e815df0a74
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ka/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">ახალი %1$s-ჩანართის გახსნა</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">მოიძიეთ</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">მოიძიეთ ინტერნეტში</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">ხმოვანი ძიება</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-kaa/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-kaa/strings.xml
new file mode 100644
index 0000000000..e968e33430
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-kaa/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Jańa%1$sbetin ashıw</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Izlew</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Internetten izlew</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Dawıslı izlew</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-kab/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-kab/strings.xml
new file mode 100644
index 0000000000..db660eeff2
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-kab/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Ldi-t iccer amaynut %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Nadi</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Nadi di web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Anadi aɣectan</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-kk/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-kk/strings.xml
new file mode 100644
index 0000000000..bd9f756463
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-kk/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Жаңа %1$s бетін ашу</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Іздеу</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Интернетте іздеу</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Дауыстық іздеу</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-kmr/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-kmr/strings.xml
new file mode 100644
index 0000000000..6c7ebf5a20
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-kmr/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Hilpekînek nû ya %1$s `ê veke</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Lêgerîn</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Di webê de bigere</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Lêgerîna dengî</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ko/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ko/strings.xml
new file mode 100644
index 0000000000..65c8238a99
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ko/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">새 %1$s 탭 열기</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">검색</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">웹 검색</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">음성 검색</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-lo/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-lo/strings.xml
new file mode 100644
index 0000000000..119d048839
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-lo/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">ເປີດແຖບ %1$s ໃໝ່</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ຄົ້ນຫາ</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ຄົ້ນຫາເວັບໄຊທ</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">ຄົ້ນຫາດ້ວຍສຽງ</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-lt/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-lt/strings.xml
new file mode 100644
index 0000000000..6387d888d2
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-lt/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Atverti naują „%1$s“ kortelę</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Ieškoti</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Ieškokite internete</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Paieška balsu</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-my/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-my/strings.xml
new file mode 100644
index 0000000000..4b497f553d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-my/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">တပ်ဘ်%1$s အသစ်ဖွင့်မည်</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ရှာဖွေမည်</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ဝက်ဘ်တွင်ရှာမည်</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">အသံဖြင့်ရှာမည်</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-nb-rNO/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-nb-rNO/strings.xml
new file mode 100644
index 0000000000..2093f48242
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-nb-rNO/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Åpne en ny %1$s-fane</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Søk</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Søk på nettet</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Stemmesøk</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-night/colors.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-night/colors.xml
new file mode 100644
index 0000000000..50b0875a99
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-night/colors.xml
@@ -0,0 +1,10 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+<resources>
+ <color name="mozac_feature_search_widget_background_color">@color/photonDarkGrey60</color>
+ <color name="mozac_feature_search_widget_color">@color/photonLightGrey05</color>
+ <color name="mozac_feature_search_widget_text_color">@color/photonLightGrey05</color>
+
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-nl/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-nl/strings.xml
new file mode 100644
index 0000000000..dc238c1191
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-nl/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Een nieuw %1$s-tabblad openen</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Zoeken</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Zoeken op het web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Gesproken zoekopdracht</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-nn-rNO/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-nn-rNO/strings.xml
new file mode 100644
index 0000000000..dfad13db60
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-nn-rNO/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Opne ei ny %1$s-fane</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Søk</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Søk på nettet</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Stemmesøk</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-oc/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-oc/strings.xml
new file mode 100644
index 0000000000..9d63b2eb12
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-oc/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Dobrir dins un onglet %1$s novèl</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Recèrca</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Recercar sul web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Recèrca a la votz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-pa-rIN/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-pa-rIN/strings.xml
new file mode 100644
index 0000000000..4a35511cdc
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-pa-rIN/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">ਨਵੀਂ %1$s ਟੈਬ ਖੋਲ੍ਹੋ</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ਖੋਜ</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ਵੈੱਬ ‘ਤੇ ਖੋਜੋ</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">ਆਵਾਜ਼ ਰਾਹੀਂ ਖੋਜੋ</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-pa-rPK/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-pa-rPK/strings.xml
new file mode 100644
index 0000000000..961ba1dcc4
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-pa-rPK/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">%1$s نال نویں ٹیب کھولھو</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">کھوج</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ویب دی کھوج</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">آواز دی کھوج</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-pl/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-pl/strings.xml
new file mode 100644
index 0000000000..eb4e6c3b1c
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-pl/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Otwórz nową kartę w przeglądarce %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Szukaj</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Szukaj w Internecie</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Wyszukiwanie głosowe</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-pt-rBR/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-pt-rBR/strings.xml
new file mode 100644
index 0000000000..8f34cd8284
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-pt-rBR/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir nova aba no %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Pesquisar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Pesquisar na web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Pesquisa por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-pt-rPT/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-pt-rPT/strings.xml
new file mode 100644
index 0000000000..b8d8f409b6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-pt-rPT/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Abrir um novo separador do %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Pesquisar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Pesquisar na Internet</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Pesquisa por voz</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-rm/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-rm/strings.xml
new file mode 100644
index 0000000000..c4af7423be
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-rm/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Avrir in nov tab da %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Tschertgar</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Tschertgar en il web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Tschertga vocala</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ru/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ru/strings.xml
new file mode 100644
index 0000000000..057ba662d6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ru/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Открыть новую вкладку %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Поиск</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Поиск в Интернете</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Голосовой поиск</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-sat/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-sat/strings.xml
new file mode 100644
index 0000000000..c7d64349f0
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-sat/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">ᱱᱟᱶᱟ ᱴᱮᱵᱽ %1$s ᱡᱷᱤᱡᱽ ᱢᱮ</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ᱥᱮᱸᱫᱽᱨᱟ</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ᱣᱮᱵᱽ ᱨᱮ ᱥᱮᱸᱫᱽᱨᱟᱭ ᱢᱮ</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">ᱨᱚᱲ ᱥᱮᱸᱫᱽᱨᱟ</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-sc/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-sc/strings.xml
new file mode 100644
index 0000000000..6c458d92a3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-sc/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Aberi in un’ischeda de %1$s noa</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Chirca</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Chirca in sa rete</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Chirca cun sa boghe</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-si/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-si/strings.xml
new file mode 100644
index 0000000000..7a6ddb27e5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-si/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">නව %1$s පටිත්තක් අරින්න</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">සොයන්න</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">වියමනහි සොයන්න</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">හඬ සෙවුම</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-sk/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-sk/strings.xml
new file mode 100644
index 0000000000..8cc11bc103
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-sk/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Otvoriť novú kartu %1$su</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Hľadať</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Vyhľadávanie na webe</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Hlasové vyhľadávanie</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-skr/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-skr/strings.xml
new file mode 100644
index 0000000000..f2dbedb987
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-skr/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">نویں %1$s ٹیب کھولو</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ڳولو</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ویب ڳولو</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">آواز نال ڳولݨ</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-sl/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-sl/strings.xml
new file mode 100644
index 0000000000..96d3701731
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-sl/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Odpri nov zavihek v %1$su</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Išči</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Iskanje po spletu</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Glasovno iskanje</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-sq/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-sq/strings.xml
new file mode 100644
index 0000000000..cfe062bbc3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-sq/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Hapni një skedë të re %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Kërko</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Kërkoni në web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Kërkim zanor</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-sr/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-sr/strings.xml
new file mode 100644
index 0000000000..2cc2c10161
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-sr/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Отвори нови %1$s језичак</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Претражи</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Претражи веб</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Гласовна претрага</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-su/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-su/strings.xml
new file mode 100644
index 0000000000..f180cfabf5
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-su/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Buka dina tab %1$s anyar</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Paluruh</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Paluruh raramat</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Paluruh sora</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-sv-rSE/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-sv-rSE/strings.xml
new file mode 100644
index 0000000000..9c3f5f1428
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-sv-rSE/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Öppna en ny %1$s-flik</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Sök</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Sök på webben</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Röstsökning</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ta/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ta/strings.xml
new file mode 100644
index 0000000000..d04a02b330
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ta/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">%1$s ஐ புதிய கீற்றில் திற</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">தேடு</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">இணையத்தில் தேடு</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">குரல் தேடல்</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-te/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-te/strings.xml
new file mode 100644
index 0000000000..dae1fcac80
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-te/strings.xml
@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">వెతకండి</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">జాలంలో వెతకండి</string>
+ </resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-tg/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-tg/strings.xml
new file mode 100644
index 0000000000..b25dee92ec
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-tg/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Кушодани варақаи нави %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Ҷустуҷӯ</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Ҷустуҷӯ дар Интернет</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ҷустуҷӯи овозӣ</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-th/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-th/strings.xml
new file mode 100644
index 0000000000..224c92e420
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-th/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">เปิดแท็บ %1$s ใหม่</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ค้นหา</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">ค้นหาเว็บ</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">ค้นหาด้วยเสียง</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-tl/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-tl/strings.xml
new file mode 100644
index 0000000000..28f70ae9be
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-tl/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Buksan ang bagong %1$s tab</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Hanapin</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Hanapin sa web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Paghanap gamit ang boses</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-tr/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-tr/strings.xml
new file mode 100644
index 0000000000..92c352702a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-tr/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Yeni %1$s sekmesi aç</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Ara</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Web’de ara</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Sesle ara</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-trs/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-trs/strings.xml
new file mode 100644
index 0000000000..c42a2fe23b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-trs/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Nā\'nïn riña rakïj ñanj nakàa %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Nānà\'huì\'</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Nānà\'huì\' riña web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Nānà\'huì\' ngà nanèt</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-tt/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-tt/strings.xml
new file mode 100644
index 0000000000..e2b8d1f5b6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-tt/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Яңа %1$s табын ачу</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Эзләү</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Интернетта эзләү</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Тавышлы эзләү</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-ug/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-ug/strings.xml
new file mode 100644
index 0000000000..27df6dae14
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-ug/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">يېڭى%1$sبەتكۈچىنى ئېچىش</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">ئىزدەش</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">توردىن ئىزدەش</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">ئاۋازلىق ئىزدەش</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-uk/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-uk/strings.xml
new file mode 100644
index 0000000000..b3060b0b0a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-uk/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Відкрити нову вкладку %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Пошук</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Пошук в Інтернеті</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Голосовий пошук</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-uz/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-uz/strings.xml
new file mode 100644
index 0000000000..9ecebb4caa
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-uz/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Yangi %1$s ta varaqni ochish</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Qidirish</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Internetdan qidirish</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Ovozli qidiruv</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-vi/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-vi/strings.xml
new file mode 100644
index 0000000000..361cbdc149
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-vi/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Mở thẻ %1$s mới</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Tìm kiếm</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Tìm kiếm trên mạng</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Tìm kiếm bằng giọng nói</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-yo/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-yo/strings.xml
new file mode 100644
index 0000000000..e530cea4a6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-yo/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Ṣí táàbù tuntun %1$s</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Ṣàwarí</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Wá lórí wẹ́ẹ̀bù</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Wíwá pẹ̀lú ohùn</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-zh-rCN/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-zh-rCN/strings.xml
new file mode 100644
index 0000000000..0fd0474f78
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-zh-rCN/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">新建 %1$s 标签页</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">搜索</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">网上搜索</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">语音搜索</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values-zh-rTW/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values-zh-rTW/strings.xml
new file mode 100644
index 0000000000..a4baf4e3d8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values-zh-rTW/strings.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">開啟新 %1$s 分頁</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">搜尋</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">搜尋 Web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">語音搜尋</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values/colors.xml b/mobile/android/android-components/components/feature/search/src/main/res/values/colors.xml
new file mode 100644
index 0000000000..07c5961bbe
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values/colors.xml
@@ -0,0 +1,9 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+<resources>
+ <color name="mozac_feature_search_widget_background_color">@color/photonLightGrey10</color>
+ <color name="mozac_feature_search_widget_text_color">@color/photonDarkGrey90</color>
+ <color name="mozac_feature_search_widget_color">@color/photonDarkGrey90</color>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values/dimens.xml b/mobile/android/android-components/components/feature/search/src/main/res/values/dimens.xml
new file mode 100644
index 0000000000..52c1f7545a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values/dimens.xml
@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+<resources>
+ <dimen name="mozac_tab_corner_radius">8dp</dimen>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/main/res/values/strings.xml b/mobile/android/android-components/components/feature/search/src/main/res/values/strings.xml
new file mode 100644
index 0000000000..6f1f2b42e7
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/main/res/values/strings.xml
@@ -0,0 +1,16 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- 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/. -->
+
+<resources>
+ <!-- Search Widget -->
+ <!-- Content description for searching with a widget. The first parameter is the name of the application.-->
+ <string name="search_widget_content_description">Open a new %1$s tab</string>
+ <!-- Text preview for smaller sized widgets -->
+ <string name="search_widget_text_short">Search</string>
+ <!-- Text preview for larger sized widgets -->
+ <string name="search_widget_text_long">Search the web</string>
+ <!-- Content description (not visible, for screen readers etc.): Voice search -->
+ <string name="search_widget_voice">Voice search</string>
+</resources>
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/BrowserStoreSeachAdapterTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/BrowserStoreSeachAdapterTest.kt
new file mode 100644
index 0000000000..37c2f83468
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/BrowserStoreSeachAdapterTest.kt
@@ -0,0 +1,97 @@
+/* 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/. */
+
+package mozilla.components.feature.search
+
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.SessionState
+import mozilla.components.browser.state.state.createCustomTab
+import mozilla.components.browser.state.state.createTab
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.search.SearchRequest
+import mozilla.components.support.test.any
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.whenever
+import org.junit.Assert.assertFalse
+import org.junit.Assert.assertTrue
+import org.junit.Before
+import org.junit.Test
+import org.mockito.Mockito.never
+import org.mockito.Mockito.verify
+
+private const val SELECTED_TAB_ID = "1"
+private const val CUSTOM_TAB_ID = "2"
+
+class BrowserStoreSearchAdapterTest {
+
+ private lateinit var browserStore: BrowserStore
+ private val state = BrowserState(
+ tabs = listOf(createTab(id = SELECTED_TAB_ID, url = "https://mozilla.org", private = true)),
+ customTabs = listOf(createCustomTab(id = CUSTOM_TAB_ID, url = "https://firefox.com", source = SessionState.Source.Internal.CustomTab)),
+ selectedTabId = SELECTED_TAB_ID,
+ )
+
+ @Before
+ fun setup() {
+ browserStore = mock()
+ whenever(browserStore.state).thenReturn(state)
+ }
+
+ @Test
+ fun `adapter does nothing with null tab`() {
+ whenever(browserStore.state).thenReturn(BrowserState())
+ val searchAdapter = BrowserStoreSearchAdapter(browserStore)
+
+ searchAdapter.sendSearch(isPrivate = false, text = "normal search")
+ searchAdapter.sendSearch(isPrivate = true, text = "private search")
+
+ verify(browserStore, never()).dispatch(any())
+ assertFalse(searchAdapter.isPrivateSession())
+ }
+
+ @Test
+ fun `sendSearch with selected tab`() {
+ val searchAdapter = BrowserStoreSearchAdapter(browserStore)
+ searchAdapter.sendSearch(isPrivate = false, text = "normal search")
+ verify(browserStore).dispatch(
+ ContentAction.UpdateSearchRequestAction(
+ SELECTED_TAB_ID,
+ SearchRequest(isPrivate = false, query = "normal search"),
+ ),
+ )
+
+ searchAdapter.sendSearch(isPrivate = true, text = "private search")
+ verify(browserStore).dispatch(
+ ContentAction.UpdateSearchRequestAction(
+ SELECTED_TAB_ID,
+ SearchRequest(isPrivate = true, query = "private search"),
+ ),
+ )
+
+ assertTrue(searchAdapter.isPrivateSession())
+ }
+
+ @Test
+ fun `sendSearch with custom tab`() {
+ val searchAdapter = BrowserStoreSearchAdapter(browserStore, CUSTOM_TAB_ID)
+ searchAdapter.sendSearch(isPrivate = false, text = "normal search")
+ verify(browserStore).dispatch(
+ ContentAction.UpdateSearchRequestAction(
+ CUSTOM_TAB_ID,
+ SearchRequest(isPrivate = false, query = "normal search"),
+ ),
+ )
+
+ searchAdapter.sendSearch(isPrivate = true, text = "private search")
+ verify(browserStore).dispatch(
+ ContentAction.UpdateSearchRequestAction(
+ CUSTOM_TAB_ID,
+ SearchRequest(isPrivate = true, query = "private search"),
+ ),
+ )
+
+ assertFalse(searchAdapter.isPrivateSession())
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchFeatureTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchFeatureTest.kt
new file mode 100644
index 0000000000..17d6d9bfe3
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchFeatureTest.kt
@@ -0,0 +1,134 @@
+/* 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/. */
+
+package mozilla.components.feature.search
+
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.action.TabListAction
+import mozilla.components.browser.state.selector.selectedTab
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.createTab
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.search.SearchRequest
+import mozilla.components.support.test.any
+import mozilla.components.support.test.eq
+import mozilla.components.support.test.ext.joinBlocking
+import mozilla.components.support.test.libstate.ext.waitUntilIdle
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.rule.MainCoroutineRule
+import org.junit.After
+import org.junit.Assert.assertNull
+import org.junit.Before
+import org.junit.Rule
+import org.junit.Test
+import org.mockito.Mockito.times
+import org.mockito.Mockito.verify
+
+private const val SELECTED_TAB_ID = "1"
+
+class SearchFeatureTest {
+
+ @get:Rule
+ val coroutinesTestRule = MainCoroutineRule()
+
+ private lateinit var performSearch: (SearchRequest, String) -> Unit
+ private lateinit var store: BrowserStore
+ private lateinit var searchFeature: SearchFeature
+
+ @Before
+ fun before() {
+ store = BrowserStore(
+ mockBrowserState(),
+ )
+ performSearch = mock()
+ searchFeature = SearchFeature(store, null, performSearch).apply {
+ start()
+ }
+ }
+
+ private fun mockBrowserState(): BrowserState {
+ return BrowserState(
+ tabs = listOf(
+ createTab("https://www.duckduckgo.com", id = "0"),
+ createTab("https://www.mozilla.org", id = SELECTED_TAB_ID),
+ createTab("https://www.wikipedia.org", id = "2"),
+ ),
+ selectedTabId = SELECTED_TAB_ID,
+ )
+ }
+
+ @After
+ fun after() {
+ searchFeature.stop()
+ }
+
+ @Test
+ fun `GIVEN a tab is selected WHEN a search request is sent THEN a search should be performed`() {
+ verify(performSearch, times(0)).invoke(any(), eq(SELECTED_TAB_ID))
+
+ val normalSearchRequest = SearchRequest(isPrivate = false, query = "query")
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, normalSearchRequest)).joinBlocking()
+
+ verify(performSearch, times(1)).invoke(any(), eq(SELECTED_TAB_ID))
+ verify(performSearch, times(1)).invoke(normalSearchRequest, SELECTED_TAB_ID)
+
+ val privateSearchRequest = SearchRequest(isPrivate = true, query = "query")
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, privateSearchRequest)).joinBlocking()
+
+ verify(performSearch, times(2)).invoke(any(), eq(SELECTED_TAB_ID))
+ verify(performSearch, times(1)).invoke(privateSearchRequest, SELECTED_TAB_ID)
+ }
+
+ @Test
+ fun `GIVEN no tab is selected WHEN a search request is sent THEN no search should be performed`() {
+ store.dispatch(TabListAction.RemoveTabAction(tabId = SELECTED_TAB_ID, selectParentIfExists = false))
+
+ verify(performSearch, times(0)).invoke(any(), eq(SELECTED_TAB_ID))
+
+ val normalSearchRequest = SearchRequest(isPrivate = false, query = "query")
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, normalSearchRequest)).joinBlocking()
+
+ verify(performSearch, times(0)).invoke(any(), eq(SELECTED_TAB_ID))
+ verify(performSearch, times(0)).invoke(normalSearchRequest, SELECTED_TAB_ID)
+
+ val privateSearchRequest = SearchRequest(isPrivate = true, query = "query")
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, privateSearchRequest)).joinBlocking()
+
+ verify(performSearch, times(0)).invoke(any(), eq(SELECTED_TAB_ID))
+ verify(performSearch, times(0)).invoke(privateSearchRequest, SELECTED_TAB_ID)
+ }
+
+ @Test
+ fun `WHEN a search request has been handled THEN that request should have been consumed`() {
+ val normalSearchRequest = SearchRequest(isPrivate = false, query = "query")
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, normalSearchRequest)).joinBlocking()
+ store.waitUntilIdle()
+
+ assertNull(store.state.selectedTab!!.content.searchRequest)
+
+ val privateSearchRequest = SearchRequest(isPrivate = true, query = "query")
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, privateSearchRequest)).joinBlocking()
+ store.waitUntilIdle()
+
+ assertNull(store.state.selectedTab!!.content.searchRequest)
+ }
+
+ @Test
+ fun `WHEN the same search is requested two times THEN both search requests are preformed and consumed`() {
+ val searchRequest = SearchRequest(isPrivate = false, query = "query")
+ verify(performSearch, times(0)).invoke(searchRequest, SELECTED_TAB_ID)
+
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, searchRequest)).joinBlocking()
+ store.waitUntilIdle()
+
+ verify(performSearch, times(1)).invoke(searchRequest, SELECTED_TAB_ID)
+ assertNull(store.state.selectedTab!!.content.searchRequest)
+
+ store.dispatch(ContentAction.UpdateSearchRequestAction(SELECTED_TAB_ID, searchRequest)).joinBlocking()
+ store.waitUntilIdle()
+
+ verify(performSearch, times(2)).invoke(searchRequest, SELECTED_TAB_ID)
+ assertNull(store.state.selectedTab!!.content.searchRequest)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchUseCasesTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchUseCasesTest.kt
new file mode 100644
index 0000000000..0168491b0b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/SearchUseCasesTest.kt
@@ -0,0 +1,663 @@
+/* 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/. */
+
+package mozilla.components.feature.search
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import mozilla.components.browser.state.action.BrowserAction
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.action.EngineAction
+import mozilla.components.browser.state.action.TabListAction
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.SearchState
+import mozilla.components.browser.state.state.SessionState
+import mozilla.components.browser.state.state.availableSearchEngines
+import mozilla.components.browser.state.state.createTab
+import mozilla.components.browser.state.state.searchEngines
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.EngineSession.LoadUrlFlags
+import mozilla.components.feature.search.ext.createSearchEngine
+import mozilla.components.feature.session.SessionUseCases
+import mozilla.components.feature.tabs.TabsUseCases
+import mozilla.components.support.test.ext.joinBlocking
+import mozilla.components.support.test.libstate.ext.waitUntilIdle
+import mozilla.components.support.test.middleware.CaptureActionsMiddleware
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.whenever
+import org.junit.After
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNull
+import org.junit.Before
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.mockito.Mockito.doReturn
+import org.mockito.Mockito.verify
+
+@RunWith(AndroidJUnit4::class)
+class SearchUseCasesTest {
+
+ private lateinit var searchEngine: SearchEngine
+ private lateinit var store: BrowserStore
+ private lateinit var useCases: SearchUseCases
+ private lateinit var tabsUseCases: TabsUseCases
+ private lateinit var sessionUseCases: SessionUseCases
+ private lateinit var loadUrlUseCase: SessionUseCases.DefaultLoadUrlUseCase
+
+ private val middleware = CaptureActionsMiddleware<BrowserState, BrowserAction>()
+
+ private val searchTerms = "mozilla android"
+ private val searchUrl = "https://example.org/?q=mozilla%20android"
+ private val searchEngineName = "Test"
+
+ @Before
+ fun setup() {
+ searchEngine = createSearchEngine(
+ name = searchEngineName,
+ url = "https://example.org/?q={searchTerms}",
+ icon = mock(),
+ )
+
+ tabsUseCases = mock()
+ sessionUseCases = mock()
+ loadUrlUseCase = mock()
+ doReturn(loadUrlUseCase).`when`(sessionUseCases).loadUrl
+
+ store = BrowserStore(
+ initialState = BrowserState(
+ search = SearchState(
+ regionSearchEngines = listOf(searchEngine),
+ ),
+ ),
+ middleware = listOf(middleware),
+ )
+
+ useCases = SearchUseCases(
+ store,
+ tabsUseCases,
+ sessionUseCases,
+ )
+ }
+
+ @After
+ fun tearDown() {
+ middleware.reset()
+ }
+
+ @Test
+ fun `GIVEN existing Session and Tab WHEN default search invoked THEN expected actions are dispatched`() {
+ val id = "mozilla"
+ store.dispatch(
+ TabListAction.AddTabAction(
+ tab = createTab(url = "https://www.mozilla.org", id = id),
+ select = true,
+ ),
+ ).joinBlocking()
+
+ useCases.defaultSearch(
+ searchTerms = searchTerms,
+ searchEngine = searchEngine,
+ )
+ store.waitUntilIdle()
+
+ val isSearchAction = middleware.findFirstAction(ContentAction.UpdateIsSearchAction::class)
+ assertEquals(id, isSearchAction.sessionId)
+ assertEquals(true, isSearchAction.isSearch)
+ assertEquals(searchEngineName, isSearchAction.searchEngineName)
+
+ middleware.assertLastAction(EngineAction.LoadUrlAction::class) { action ->
+ assertEquals(id, action.tabId)
+ assertEquals(searchUrl, action.url)
+ }
+ }
+
+ @Test
+ fun `GIVEN existing Session, no existing Tab WHEN default search invoked THEN add tab is called`() {
+ val newTabUseCase: TabsUseCases.AddNewTabUseCase = mock()
+ whenever(tabsUseCases.addTab).thenReturn(newTabUseCase)
+ val newTabId = "9876"
+ whenever(
+ newTabUseCase(
+ url = searchUrl,
+ isSearch = true,
+ searchEngineName = searchEngineName,
+ ),
+ ).thenReturn(newTabId)
+
+ useCases.defaultSearch(
+ searchTerms = searchTerms,
+ sessionId = "mozilla",
+ searchEngine = searchEngine,
+ )
+ store.waitUntilIdle()
+
+ verify(newTabUseCase).invoke(
+ url = searchUrl,
+ isSearch = true,
+ searchEngineName = searchEngineName,
+ )
+
+ middleware.assertLastAction(ContentAction.UpdateSearchTermsAction::class) { action ->
+ assertEquals(newTabId, action.sessionId)
+ assertEquals(searchTerms, action.searchTerms)
+ }
+ }
+
+ @Test
+ fun defaultSearchOnNewSession() {
+ val searchTerms = "mozilla android"
+
+ val newTabUseCase: TabsUseCases.AddNewTabUseCase = mock()
+ whenever(tabsUseCases.addTab).thenReturn(newTabUseCase)
+ whenever(newTabUseCase(searchUrl, isSearch = true)).thenReturn("2342")
+
+ useCases.newTabSearch(searchTerms, SessionState.Source.Internal.NewTab)
+ store.waitUntilIdle()
+
+ verify(newTabUseCase).invoke(
+ searchUrl,
+ parentId = null,
+ selectTab = true,
+ source = SessionState.Source.Internal.NewTab,
+ isSearch = true,
+ )
+
+ val searchTermsAction = middleware.findFirstAction(ContentAction.UpdateSearchTermsAction::class)
+ assertEquals("2342", searchTermsAction.sessionId)
+ assertEquals(searchTerms, searchTermsAction.searchTerms)
+ }
+
+ @Test
+ fun `GIVEN additional headers and a load url flag WHEN NewTabSearchUseCase creates a new tab THEN addTab is called`() {
+ val source = SessionState.Source.Internal.UserEntered
+ val flags = LoadUrlFlags.select(LoadUrlFlags.ALLOW_JAVASCRIPT_URL)
+ val additionalHeaders = mapOf("X-Extra-Header" to "true")
+ val sessionId = "2342"
+
+ val newTabUseCase: TabsUseCases.AddNewTabUseCase = mock()
+ whenever(tabsUseCases.addTab).thenReturn(newTabUseCase)
+ whenever(
+ newTabUseCase(
+ url = searchUrl,
+ isSearch = true,
+ flags = flags,
+ source = source,
+ additionalHeaders = additionalHeaders,
+ ),
+ ).thenReturn(sessionId)
+
+ useCases.newTabSearch(
+ searchTerms = searchTerms,
+ source = source,
+ flags = flags,
+ additionalHeaders = additionalHeaders,
+ )
+ store.waitUntilIdle()
+
+ verify(newTabUseCase).invoke(
+ url = searchUrl,
+ flags = flags,
+ source = source,
+ isSearch = true,
+ additionalHeaders = additionalHeaders,
+ )
+
+ val searchTermsAction =
+ middleware.findFirstAction(ContentAction.UpdateSearchTermsAction::class)
+ assertEquals(sessionId, searchTermsAction.sessionId)
+ assertEquals(searchTerms, searchTermsAction.searchTerms)
+ }
+
+ @Test
+ fun `DefaultSearchUseCase creates new tab if no session is selected`() {
+ val newTabUseCase: TabsUseCases.AddNewTabUseCase = mock()
+ whenever(tabsUseCases.addTab).thenReturn(newTabUseCase)
+ whenever(newTabUseCase(searchUrl, isSearch = true)).thenReturn("2342")
+
+ useCases.defaultSearch(searchTerms)
+ store.waitUntilIdle()
+
+ verify(newTabUseCase).invoke(
+ searchUrl,
+ parentId = null,
+ selectTab = true,
+ source = SessionState.Source.Internal.NewTab,
+ isSearch = true,
+ )
+
+ val searchTermsAction = middleware.findFirstAction(ContentAction.UpdateSearchTermsAction::class)
+ assertEquals("2342", searchTermsAction.sessionId)
+ assertEquals(searchTerms, searchTermsAction.searchTerms)
+ }
+
+ @Test
+ fun `GIVEN additional headers and a load url flag WHEN DefaultSearchUseCase creates new tab THEN addTab is called`() {
+ val flags = LoadUrlFlags.select(LoadUrlFlags.ALLOW_JAVASCRIPT_URL)
+ val additionalHeaders = mapOf("X-Extra-Header" to "true")
+ val sessionId = "2342"
+
+ val newTabUseCase: TabsUseCases.AddNewTabUseCase = mock()
+ whenever(tabsUseCases.addTab).thenReturn(newTabUseCase)
+ whenever(
+ newTabUseCase(
+ url = searchUrl,
+ flags = flags,
+ isSearch = true,
+ searchEngineName = searchEngineName,
+ additionalHeaders = additionalHeaders,
+ ),
+ ).thenReturn(sessionId)
+
+ useCases.defaultSearch(
+ searchTerms = searchTerms,
+ searchEngine = searchEngine,
+ flags = flags,
+ additionalHeaders = additionalHeaders,
+ )
+ store.waitUntilIdle()
+
+ verify(newTabUseCase).invoke(
+ url = searchUrl,
+ flags = flags,
+ isSearch = true,
+ searchEngineName = searchEngineName,
+ additionalHeaders = additionalHeaders,
+ )
+
+ val searchTermsAction = middleware.findFirstAction(ContentAction.UpdateSearchTermsAction::class)
+ assertEquals(sessionId, searchTermsAction.sessionId)
+ assertEquals(searchTerms, searchTermsAction.searchTerms)
+ }
+
+ @Test
+ fun newPrivateTabSearch() {
+ val newTabUseCase: TabsUseCases.AddNewTabUseCase = mock()
+ whenever(tabsUseCases.addTab).thenReturn(newTabUseCase)
+ whenever(
+ newTabUseCase(
+ searchUrl,
+ source = SessionState.Source.Internal.None,
+ private = true,
+ isSearch = true,
+ ),
+ ).thenReturn("1177")
+
+ useCases.newPrivateTabSearch.invoke(searchTerms)
+ store.waitUntilIdle()
+
+ verify(newTabUseCase).invoke(
+ searchUrl,
+ parentId = null,
+ selectTab = true,
+ private = true,
+ source = SessionState.Source.Internal.None,
+ isSearch = true,
+ )
+
+ val searchTermsAction = middleware.findFirstAction(ContentAction.UpdateSearchTermsAction::class)
+ assertEquals("1177", searchTermsAction.sessionId)
+ assertEquals(searchTerms, searchTermsAction.searchTerms)
+ }
+
+ @Test
+ fun newPrivateTabSearchWithParentSession() {
+ val newTabUseCase: TabsUseCases.AddNewTabUseCase = mock()
+ whenever(tabsUseCases.addTab).thenReturn(newTabUseCase)
+ whenever(
+ newTabUseCase(
+ searchUrl,
+ source = SessionState.Source.Internal.None,
+ parentId = "test-parent",
+ private = true,
+ isSearch = true,
+ ),
+ ).thenReturn("1177")
+
+ useCases.newPrivateTabSearch.invoke(searchTerms, parentSessionId = "test-parent")
+
+ store.waitUntilIdle()
+
+ verify(newTabUseCase).invoke(
+ searchUrl,
+ parentId = "test-parent",
+ selectTab = true,
+ private = true,
+ source = SessionState.Source.Internal.None,
+ isSearch = true,
+ )
+
+ val searchTermsAction = middleware.findFirstAction(ContentAction.UpdateSearchTermsAction::class)
+ assertEquals("1177", searchTermsAction.sessionId)
+ assertEquals(searchTerms, searchTermsAction.searchTerms)
+ }
+
+ @Test
+ fun `Selecting search engine`() {
+ val store = BrowserStore(getBrowserState())
+
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ useCases.selectSearchEngine.invoke(
+ store.findSearchEngineById("engine-d"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals("engine-d", store.state.search.userSelectedSearchEngineId)
+ assertNull(store.state.search.userSelectedSearchEngineName)
+
+ useCases.selectSearchEngine.invoke(
+ store.findSearchEngineById("engine-b"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals("engine-b", store.state.search.userSelectedSearchEngineId)
+ assertEquals("Engine B", store.state.search.userSelectedSearchEngineName)
+
+ useCases.selectSearchEngine.invoke(
+ store.findSearchEngineById("engine-f"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals("engine-f", store.state.search.userSelectedSearchEngineId)
+ assertNull(store.state.search.userSelectedSearchEngineName)
+ }
+
+ @Test
+ fun `addSearchEngine - add bundled engine`() {
+ val store = BrowserStore(getBrowserState())
+
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(7, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ useCases.addSearchEngine.invoke(
+ store.findSearchEngineById("engine-i"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals(8, store.state.search.searchEngines.size)
+ assertEquals(2, store.state.search.availableSearchEngines.size)
+
+ assertEquals(4, store.state.search.regionSearchEngines.size)
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+
+ assertEquals("engine-i", store.state.search.regionSearchEngines[3].id)
+ assertEquals("Engine I", store.state.search.regionSearchEngines[3].name)
+ }
+
+ @Test
+ fun `addSearchEngine - add additional bundled engine`() {
+ val store = BrowserStore(getBrowserState())
+
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(7, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ useCases.addSearchEngine.invoke(
+ store.findSearchEngineById("engine-h"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals(8, store.state.search.searchEngines.size)
+ assertEquals(2, store.state.search.availableSearchEngines.size)
+
+ assertEquals(1, store.state.search.additionalAvailableSearchEngines.size)
+ assertEquals(2, store.state.search.additionalSearchEngines.size)
+
+ assertEquals("engine-h", store.state.search.additionalSearchEngines[1].id)
+ assertEquals("Engine H", store.state.search.additionalSearchEngines[1].name)
+ }
+
+ @Test
+ fun `addSearchEngine - add custom engine`() {
+ val store = BrowserStore(getBrowserState())
+
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(7, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ useCases.addSearchEngine.invoke(
+ createSearchEngine(
+ name = "Engine X",
+ url = "https://www.example.org/?q={searchTerms}",
+ icon = mock(),
+ ),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals(8, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ assertEquals(3, store.state.search.customSearchEngines.size)
+ assertEquals("Engine X", store.state.search.customSearchEngines[2].name)
+ assertEquals(
+ "https://www.example.org/?q={searchTerms}",
+ store.state.search.customSearchEngines[2].resultUrls[0],
+ )
+ }
+
+ @Test
+ fun `removeSearchEngine - remove bundled engine`() {
+ val store = BrowserStore(getBrowserState())
+
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(7, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ useCases.removeSearchEngine.invoke(
+ store.findSearchEngineById("engine-b"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals(6, store.state.search.searchEngines.size)
+ assertEquals(4, store.state.search.availableSearchEngines.size)
+
+ assertEquals(2, store.state.search.regionSearchEngines.size)
+ assertEquals(2, store.state.search.hiddenSearchEngines.size)
+
+ assertEquals("engine-b", store.state.search.hiddenSearchEngines[1].id)
+ assertEquals("Engine B", store.state.search.hiddenSearchEngines[1].name)
+ }
+
+ @Test
+ fun `removeSearchEngine - remove additional bundled engine`() {
+ val store = BrowserStore(getBrowserState())
+
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(7, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ useCases.removeSearchEngine.invoke(
+ store.findSearchEngineById("engine-f"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals(6, store.state.search.searchEngines.size)
+ assertEquals(4, store.state.search.availableSearchEngines.size)
+
+ assertEquals(0, store.state.search.additionalSearchEngines.size)
+ assertEquals(3, store.state.search.additionalAvailableSearchEngines.size)
+
+ assertEquals("engine-f", store.state.search.additionalAvailableSearchEngines[2].id)
+ assertEquals("Engine F", store.state.search.additionalAvailableSearchEngines[2].name)
+ }
+
+ @Test
+ fun `removeSearchEngine - remove custom engine`() {
+ val store = BrowserStore(getBrowserState())
+
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(7, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ useCases.removeSearchEngine.invoke(
+ store.findSearchEngineById("engine-d"),
+ )
+
+ store.waitUntilIdle()
+
+ assertEquals(6, store.state.search.searchEngines.size)
+ assertEquals(3, store.state.search.availableSearchEngines.size)
+
+ assertEquals(1, store.state.search.customSearchEngines.size)
+ }
+
+ @Test
+ fun `GIVEN disable search engine use case is invoked WHEN engine gets unselected THEN ID is stored in search state`() {
+ val store = BrowserStore(getBrowserState())
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(0, store.state.search.disabledSearchEngineIds.size)
+
+ useCases.updateDisabledSearchEngineIds.invoke(
+ searchEngineId = "engine-d",
+ isEnabled = false,
+ )
+ store.waitUntilIdle()
+
+ assertEquals(1, store.state.search.disabledSearchEngineIds.size)
+ }
+
+ @Test
+ fun `GIVEN disable search engine use case is invoked WHEN engine gets selected THEN ID is removed from search state`() {
+ val store = BrowserStore(getBrowserState(disabledSearchEngineIds = listOf("engine-d")))
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(1, store.state.search.disabledSearchEngineIds.size)
+
+ useCases.updateDisabledSearchEngineIds.invoke(
+ searchEngineId = "engine-d",
+ isEnabled = true,
+ )
+ store.waitUntilIdle()
+
+ assertEquals(0, store.state.search.disabledSearchEngineIds.size)
+ }
+
+ @Test
+ fun `WHEN restore search engines use case is invoked GIVEN there are hidden engines THEN hidden engines are added back to the bundled engine list`() {
+ val regionSearchEngines = listOf(
+ SearchEngine("bundled-engine-a", "Regional Engine A", mock(), type = SearchEngine.Type.BUNDLED),
+ SearchEngine("bundled-engine-b", "Regional Engine B", mock(), type = SearchEngine.Type.BUNDLED),
+ )
+
+ val hiddenEngine = SearchEngine(
+ "bundled-engine-c",
+ "Regional Engine C",
+ mock(),
+ type = SearchEngine.Type.BUNDLED,
+ )
+
+ val store = BrowserStore(getBrowserState(hiddenSearchEngine = listOf(hiddenEngine), regionSearchEngines = regionSearchEngines))
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(2, store.state.search.regionSearchEngines.size)
+ assertEquals(1, store.state.search.hiddenSearchEngines.size)
+
+ assertEquals("bundled-engine-a", store.state.search.regionSearchEngines[0].id)
+ assertEquals("bundled-engine-b", store.state.search.regionSearchEngines[1].id)
+ assertEquals("bundled-engine-c", store.state.search.hiddenSearchEngines[0].id)
+
+ useCases.restoreHiddenSearchEngines.invoke()
+ store.waitUntilIdle()
+
+ assertEquals(3, store.state.search.regionSearchEngines.size)
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+
+ assertEquals("bundled-engine-a", store.state.search.regionSearchEngines[0].id)
+ assertEquals("bundled-engine-b", store.state.search.regionSearchEngines[1].id)
+ assertEquals("bundled-engine-c", store.state.search.regionSearchEngines[2].id)
+ }
+
+ @Test
+ fun `WHEN restore search engines use case is invoked GIVEN there are no hidden engines THEN do nothing`() {
+ val regionSearchEngines = listOf(
+ SearchEngine("bundled-engine-a", "Regional Engine A", mock(), type = SearchEngine.Type.BUNDLED),
+ SearchEngine("bundled-engine-b", "Regional Engine B", mock(), type = SearchEngine.Type.BUNDLED),
+ SearchEngine("bundled-engine-c", "Regional Engine C", mock(), type = SearchEngine.Type.BUNDLED),
+ )
+ val store = BrowserStore(getBrowserState(hiddenSearchEngine = emptyList(), regionSearchEngines = regionSearchEngines))
+ val useCases = SearchUseCases(store, mock(), mock())
+
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+ assertEquals(3, store.state.search.regionSearchEngines.size)
+
+ assertEquals("bundled-engine-a", store.state.search.regionSearchEngines[0].id)
+ assertEquals("bundled-engine-b", store.state.search.regionSearchEngines[1].id)
+ assertEquals("bundled-engine-c", store.state.search.regionSearchEngines[2].id)
+
+ useCases.restoreHiddenSearchEngines.invoke()
+ store.waitUntilIdle()
+
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+ assertEquals(3, store.state.search.regionSearchEngines.size)
+
+ assertEquals("bundled-engine-a", store.state.search.regionSearchEngines[0].id)
+ assertEquals("bundled-engine-b", store.state.search.regionSearchEngines[1].id)
+ assertEquals("bundled-engine-c", store.state.search.regionSearchEngines[2].id)
+ }
+}
+
+private fun getBrowserState(
+ disabledSearchEngineIds: List<String> = emptyList(),
+ regionSearchEngines: List<SearchEngine> = listOf(
+ SearchEngine("engine-a", "Engine A", mock(), type = SearchEngine.Type.BUNDLED),
+ SearchEngine("engine-b", "Engine B", mock(), type = SearchEngine.Type.BUNDLED),
+ SearchEngine("engine-c", "Engine C", mock(), type = SearchEngine.Type.BUNDLED),
+ ),
+ hiddenSearchEngine: List<SearchEngine> = listOf(
+ SearchEngine(
+ "engine-i",
+ "Engine I",
+ mock(),
+ type = SearchEngine.Type.BUNDLED,
+ ),
+ ),
+) = BrowserState(
+ search = SearchState(
+ region = RegionState("US", "US"),
+ regionSearchEngines = regionSearchEngines,
+ customSearchEngines = listOf(
+ SearchEngine("engine-d", "Engine D", mock(), type = SearchEngine.Type.CUSTOM),
+ SearchEngine("engine-e", "Engine E", mock(), type = SearchEngine.Type.CUSTOM),
+ ),
+ applicationSearchEngines = listOf(
+ SearchEngine("engine-j", "Engine J", mock(), type = SearchEngine.Type.APPLICATION),
+ ),
+ additionalSearchEngines = listOf(
+ SearchEngine("engine-f", "Engine F", mock(), type = SearchEngine.Type.BUNDLED_ADDITIONAL),
+ ),
+ additionalAvailableSearchEngines = listOf(
+ SearchEngine("engine-g", "Engine G", mock(), type = SearchEngine.Type.BUNDLED_ADDITIONAL),
+ SearchEngine("engine-h", "Engine H", mock(), type = SearchEngine.Type.BUNDLED_ADDITIONAL),
+ ),
+ hiddenSearchEngines = hiddenSearchEngine,
+ disabledSearchEngineIds = disabledSearchEngineIds,
+ regionDefaultSearchEngineId = "engine-b",
+ userSelectedSearchEngineId = null,
+ userSelectedSearchEngineName = null,
+ ),
+)
+
+private fun BrowserStore.findSearchEngineById(id: String): SearchEngine {
+ val searchEngine = (state.search.searchEngines + state.search.availableSearchEngines).find {
+ it.id == id
+ }
+ return requireNotNull(searchEngine)
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/BrowserStoreKtTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/BrowserStoreKtTest.kt
new file mode 100644
index 0000000000..c29082e0e1
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/BrowserStoreKtTest.kt
@@ -0,0 +1,117 @@
+/* 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/. */
+
+package mozilla.components.feature.search.ext
+
+import mozilla.components.browser.state.action.SearchAction
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.SearchState
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.support.test.mock
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertNull
+import org.junit.Assert.assertTrue
+import org.junit.Test
+import java.util.concurrent.CountDownLatch
+import java.util.concurrent.TimeUnit
+
+class BrowserStoreKtTest {
+ @Test
+ fun `waitForDefaultSearchEngine - with state already loaded`() {
+ val store = BrowserStore(
+ BrowserState(
+ search = SearchState(
+ regionSearchEngines = listOf(
+ SearchEngine(
+ id = "google",
+ name = "Google",
+ icon = mock(),
+ type = SearchEngine.Type.BUNDLED,
+ ),
+ ),
+ userSelectedSearchEngineId = "google",
+ complete = true,
+ ),
+ ),
+ )
+
+ val latch = CountDownLatch(1)
+
+ store.waitForSelectedOrDefaultSearchEngine { searchEngine ->
+ assertNotNull(searchEngine)
+ assertEquals("google", searchEngine!!.id)
+ latch.countDown()
+ }
+
+ assertTrue(latch.await(10, TimeUnit.SECONDS))
+ }
+
+ @Test
+ fun `waitForDefaultSearchEngine - with state dispatched later`() {
+ val store = BrowserStore()
+
+ val latch = CountDownLatch(1)
+
+ store.waitForSelectedOrDefaultSearchEngine { searchEngine ->
+ assertNotNull(searchEngine)
+ assertEquals("google", searchEngine!!.id)
+ latch.countDown()
+ }
+
+ store.dispatch(
+ SearchAction.SetSearchEnginesAction(
+ regionSearchEngines = listOf(
+ SearchEngine(
+ id = "google",
+ name = "Google",
+ icon = mock(),
+ type = SearchEngine.Type.BUNDLED,
+ ),
+ ),
+ userSelectedSearchEngineId = null,
+ userSelectedSearchEngineName = null,
+ regionDefaultSearchEngineId = "google",
+ customSearchEngines = emptyList(),
+ hiddenSearchEngines = emptyList(),
+ disabledSearchEngineIds = emptyList(),
+ additionalAvailableSearchEngines = emptyList(),
+ additionalSearchEngines = emptyList(),
+ regionSearchEnginesOrder = listOf("google"),
+ ),
+ )
+
+ assertTrue(latch.await(10, TimeUnit.SECONDS))
+ }
+
+ @Test
+ fun `waitForDefaultSearchEngine - no default was loaded`() {
+ val store = BrowserStore()
+
+ val latch = CountDownLatch(1)
+
+ store.waitForSelectedOrDefaultSearchEngine { searchEngine ->
+ assertNull(searchEngine)
+ latch.countDown()
+ }
+
+ store.dispatch(
+ SearchAction.SetSearchEnginesAction(
+ regionSearchEngines = listOf(),
+ userSelectedSearchEngineId = null,
+ userSelectedSearchEngineName = null,
+ regionDefaultSearchEngineId = "default",
+ customSearchEngines = emptyList(),
+ hiddenSearchEngines = emptyList(),
+ disabledSearchEngineIds = emptyList(),
+ additionalAvailableSearchEngines = emptyList(),
+ additionalSearchEngines = emptyList(),
+ regionSearchEnginesOrder = listOf("google"),
+ ),
+ )
+
+ assertTrue(latch.await(10, TimeUnit.SECONDS))
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/SearchEngineKtTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/SearchEngineKtTest.kt
new file mode 100644
index 0000000000..6c117132ff
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/ext/SearchEngineKtTest.kt
@@ -0,0 +1,196 @@
+/* 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/. */
+
+package mozilla.components.feature.search.ext
+
+import android.graphics.Bitmap
+import android.net.Uri
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.state.SearchState
+import mozilla.components.support.test.mock
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertNull
+import org.junit.Test
+import org.junit.runner.RunWith
+import java.util.UUID
+
+@RunWith(AndroidJUnit4::class)
+class SearchEngineKtTest {
+
+ @Test
+ fun `WHEN search engine is created THEN the correct properties are set`() {
+ val name = "name"
+ val url = "https://www.example.com/search?q={searchTerms}"
+ val icon: Bitmap = mock()
+ val suggestUrl = "https://www.example.com/search"
+ val isGeneral = true
+ val searchEngine = createSearchEngine(
+ name = name,
+ url = url,
+ icon = icon,
+ suggestUrl = suggestUrl,
+ isGeneral = isGeneral,
+ )
+
+ assertNotNull(searchEngine.id)
+ assertEquals(name, searchEngine.name)
+ assertEquals(icon, searchEngine.icon)
+ assertEquals(SearchEngine.Type.CUSTOM, searchEngine.type)
+ assertEquals(listOf(url), searchEngine.resultUrls)
+ assertEquals(suggestUrl, searchEngine.suggestUrl)
+ assertEquals(isGeneral, searchEngine.isGeneral)
+ }
+
+ @Test
+ fun `Create search URL for startpage`() {
+ val searchEngine = SearchEngine(
+ id = UUID.randomUUID().toString(),
+ name = "Escosia",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf(
+ "https://www.startpage.com/sp/search?q={searchTerms}",
+ ),
+ )
+
+ assertEquals(
+ "https://www.startpage.com/sp/search?q=Hello%20World",
+ searchEngine.buildSearchUrl("Hello World"),
+ )
+ }
+
+ @Test
+ fun `Create search URL for ecosia`() {
+ val searchEngine = createSearchEngine(
+ name = "Ecosia",
+ icon = mock(),
+ url = "https://www.ecosia.org/search?q={searchTerms}",
+ )
+
+ assertEquals(
+ "https://www.ecosia.org/search?q=Hello%20World",
+ searchEngine.buildSearchUrl("Hello World"),
+ )
+ }
+
+ @Test
+ fun `GIVEN ecosia search engine and a set of urls THEN search terms are determined when present`() {
+ val searchEngine = createSearchEngine(
+ name = "Ecosia",
+ icon = mock(),
+ url = "https://www.ecosia.org/search?q={searchTerms}",
+ )
+
+ assertNull(searchEngine.parseSearchTerms(Uri.parse("https://www.ecosia.org/search?q=")))
+ assertNull(searchEngine.parseSearchTerms(Uri.parse("https://www.ecosia.org/search?attr=moz-test")))
+
+ assertEquals(
+ "second test search",
+ searchEngine.parseSearchTerms(Uri.parse("https://www.ecosia.org/search?q=second%20test%20search")),
+ )
+
+ assertEquals(
+ "Another test",
+ searchEngine.parseSearchTerms(Uri.parse("https://www.ecosia.org/search?r=134s7&attr=moz-test&q=Another%20test&d=136697676793")),
+ )
+ }
+
+ @Test
+ fun `GIVEN empty search state THEN search terms are never determined`() {
+ val searchState = SearchState()
+ assertNull(searchState.parseSearchTerms("https://google.com/search/?q=the%20sandbaggers"))
+ }
+
+ @Test
+ fun `GIVEN a search state and a set of urls THEN search terms are determined when present`() {
+ val google = createSearchEngine(
+ name = "Google",
+ icon = mock(),
+ url = "https://google.com/search/?q={searchTerms}",
+ )
+ val ecosia = createSearchEngine(
+ name = "Ecosia",
+ icon = mock(),
+ url = "https://www.ecosia.org/search?q={searchTerms}",
+ )
+ val baidu = createSearchEngine(
+ name = "Baidu",
+ icon = mock(),
+ url = "https://www.baidu.com/s?wd={searchTerms}",
+ )
+ val searchState = SearchState(
+ regionSearchEngines = listOf(google, baidu),
+ additionalSearchEngines = listOf(ecosia),
+ customSearchEngines = listOf(baidu, ecosia),
+ )
+
+ assertNull(searchState.parseSearchTerms("https://www.ecosia.org/search?q="))
+ assertNull(searchState.parseSearchTerms("http://help.baidu.com/"))
+ assertEquals(
+ "神舟十二号载人飞行任务标识发布",
+ searchState.parseSearchTerms("https://www.baidu.com/s?cl=3&tn=baidutop10&fr=top1000&wd=%E7%A5%9E%E8%88%9F%E5%8D%81%E4%BA%8C%E5%8F%B7%E8%BD%BD%E4%BA%BA%E9%A3%9E%E8%A1%8C%E4%BB%BB%E5%8A%A1%E6%A0%87%E8%AF%86%E5%8F%91%E5%B8%83&rsv_idx=2&rsv_dl=fyb_n_homepage&hisfilter=1"),
+ )
+ assertEquals(
+ "the sandbaggers",
+ searchState.parseSearchTerms("https://google.com/search/?q=the%20sandbaggers"),
+ )
+ assertEquals(
+ "фаерфокс",
+ searchState.parseSearchTerms("https://google.com/search/?q=%D1%84%D0%B0%D0%B5%D1%80%D1%84%D0%BE%D0%BA%D1%81"),
+ )
+ assertEquals(
+ "Another test",
+ searchState.parseSearchTerms("https://www.ecosia.org/search?r=134s7&attr=moz-test&q=Another%20test&d=136697676793"),
+ )
+ }
+
+ @Test
+ fun `GIVEN search engine parameter can not be found THEN search terms are never determined`() {
+ val invalidEngine = SearchEngine(
+ id = UUID.randomUUID().toString(),
+ name = "invalid",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://mozilla.org/search/?q={invalid}"),
+ )
+
+ val searchState = SearchState(
+ regionSearchEngines = listOf(invalidEngine),
+ )
+
+ assertNull(searchState.parseSearchTerms("https://mozilla.org/search/?q=test"))
+ }
+
+ @Test
+ fun `GIVEN a search state and a set of input encoding THEN search terms are encoded by input encoding parameter`() {
+ val searchEngine = createSearchEngine(
+ name = "Yahoo! Auctions",
+ icon = mock(),
+ url = "https://auctions.yahoo.co.jp/search/search&p={searchTerms}",
+ inputEncoding = "EUC-JP",
+ )
+
+ assertEquals(
+ "https://auctions.yahoo.co.jp/search/search&p=%A5%D5%A5%A1%A5%A4%A5%E4%A1%BC%A5%D5%A5%A9%A5%C3%A5%AF%A5%B9",
+ searchEngine.buildSearchUrl("ファイヤーフォックス"),
+ )
+ }
+
+ @Test
+ fun `GIVEN invalid input encoding THEN encoding of search terms are determined as UTF-8`() {
+ val searchEngine = createSearchEngine(
+ name = "name",
+ icon = mock(),
+ url = "https://www.example.com/search?q={searchTerms}",
+ inputEncoding = "INVALID-ENOCODING",
+ )
+
+ assertEquals(
+ "https://www.example.com/search?q=%E7%81%AB%E7%8B%90",
+ searchEngine.buildSearchUrl("火狐"),
+ )
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddlewareTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddlewareTest.kt
new file mode 100644
index 0000000000..b144facd8e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/AdsTelemetryMiddlewareTest.kt
@@ -0,0 +1,126 @@
+/* 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/. */
+
+package mozilla.components.feature.search.middleware
+
+import mozilla.components.browser.state.action.ContentAction
+import mozilla.components.browser.state.action.TabListAction
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.ContentState
+import mozilla.components.browser.state.state.LoadRequestState
+import mozilla.components.browser.state.state.TabSessionState
+import mozilla.components.browser.state.state.createTab
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry
+import mozilla.components.support.test.ext.joinBlocking
+import mozilla.components.support.test.mock
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertNull
+import org.junit.Before
+import org.junit.Test
+import org.mockito.Mockito.verify
+
+class AdsTelemetryMiddlewareTest {
+ val sessionId = "session"
+ lateinit var adsMiddleware: AdsTelemetryMiddleware
+ lateinit var browserState: BrowserState
+
+ @Before
+ fun setup() {
+ adsMiddleware = AdsTelemetryMiddleware(mock())
+ browserState = BrowserState(
+ tabs = listOf(TabSessionState(content = ContentState("https://mozilla.org"), id = sessionId)),
+ )
+ }
+
+ @Test
+ fun `GIVEN redirectChain empty WHEN a new URL loads THEN the redirectChain starts from the current tab url`() {
+ val store = BrowserStore(
+ initialState = browserState,
+ middleware = listOf(adsMiddleware),
+ )
+
+ store.dispatch(
+ ContentAction.UpdateLoadRequestAction(
+ sessionId,
+ LoadRequestState(
+ url = "https://mozilla.org/firefox",
+ triggeredByRedirect = false,
+ triggeredByUser = false,
+ ),
+ ),
+ ).joinBlocking()
+
+ assertEquals(1, adsMiddleware.redirectChain.size)
+ assertEquals("https://mozilla.org", adsMiddleware.redirectChain[sessionId]!!.root)
+ }
+
+ @Test
+ fun `GIVEN redirectChain is not empty WHEN a new URL loads THEN that URL is added to the chain`() {
+ adsMiddleware.redirectChain[sessionId] = RedirectChain("https://mozilla.org")
+ val store = BrowserStore(
+ initialState = browserState,
+ middleware = listOf(adsMiddleware),
+ )
+
+ store.dispatch(
+ ContentAction.UpdateLoadRequestAction(
+ sessionId,
+ LoadRequestState(
+ url = "https://mozilla.org/firefox",
+ triggeredByRedirect = false,
+ triggeredByUser = false,
+ ),
+ ),
+ ).joinBlocking()
+
+ assertEquals(1, adsMiddleware.redirectChain.size)
+ assertEquals("https://mozilla.org", adsMiddleware.redirectChain[sessionId]!!.root)
+ assertEquals(1, adsMiddleware.redirectChain.size)
+ assertEquals("https://mozilla.org/firefox", adsMiddleware.redirectChain[sessionId]!!.chain[0])
+ }
+
+ @Test
+ fun `WHEN the session URL is updated THEN check if an ad was clicked`() {
+ val adsTelemetry: AdsTelemetry = mock()
+ val adsMiddleware = AdsTelemetryMiddleware(adsTelemetry)
+ adsMiddleware.redirectChain[sessionId] = RedirectChain("https://mozilla.org")
+ adsMiddleware.redirectChain[sessionId]!!.chain.add("https://mozilla.org/firefox")
+ val store = BrowserStore(
+ initialState = browserState,
+ middleware = listOf(adsMiddleware),
+ )
+
+ store
+ .dispatch(ContentAction.UpdateUrlAction(sessionId, "https://mozilla.org/firefox"))
+ .joinBlocking()
+
+ verify(adsTelemetry).checkIfAddWasClicked(
+ "https://mozilla.org",
+ listOf("https://mozilla.org/firefox"),
+ )
+ }
+
+ @Test
+ fun `GIVEN a location update WHEN ads telemetry is recorded THEN redirect chain is reset`() {
+ val tab = createTab(id = "1", url = "http://mozilla.org")
+ val store = BrowserStore(
+ initialState = browserState,
+ middleware = listOf(adsMiddleware),
+ )
+ store.dispatch(TabListAction.AddTabAction(tab)).joinBlocking()
+ store.dispatch(
+ ContentAction.UpdateLoadRequestAction(
+ tab.id,
+ LoadRequestState("https://mozilla.org", true, true),
+ ),
+ ).joinBlocking()
+
+ assertNotNull(adsMiddleware.redirectChain[tab.id])
+
+ store.dispatch(ContentAction.UpdateUrlAction(tab.id, "https://mozilla.org")).joinBlocking()
+ assertNull(adsMiddleware.redirectChain[tab.id])
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/SearchMiddlewareTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/SearchMiddlewareTest.kt
new file mode 100644
index 0000000000..a93a6e1700
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/middleware/SearchMiddlewareTest.kt
@@ -0,0 +1,1921 @@
+/* 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/. */
+
+package mozilla.components.feature.search.middleware
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import kotlinx.coroutines.test.TestDispatcher
+import mozilla.components.browser.state.action.SearchAction
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.browser.state.state.availableSearchEngines
+import mozilla.components.browser.state.state.searchEngines
+import mozilla.components.browser.state.state.selectedOrDefaultSearchEngine
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.feature.search.ext.createSearchEngine
+import mozilla.components.feature.search.storage.CustomSearchEngineStorage
+import mozilla.components.feature.search.storage.SearchMetadataStorage
+import mozilla.components.support.test.ext.joinBlocking
+import mozilla.components.support.test.fakes.android.FakeSharedPreferences
+import mozilla.components.support.test.libstate.ext.waitUntilIdle
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.robolectric.testContext
+import mozilla.components.support.test.rule.MainCoroutineRule
+import mozilla.components.support.test.rule.runTestOnMain
+import org.junit.After
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertFalse
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertNull
+import org.junit.Assert.assertTrue
+import org.junit.Before
+import org.junit.Rule
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.mockito.Mockito.doReturn
+import org.mockito.Mockito.verify
+import org.mockito.Mockito.verifyNoMoreInteractions
+import java.util.Locale
+import java.util.UUID
+
+@RunWith(AndroidJUnit4::class)
+class SearchMiddlewareTest {
+ @get:Rule
+ val coroutinesTestRule = MainCoroutineRule()
+ private val dispatcher = coroutinesTestRule.testDispatcher
+
+ private lateinit var originalLocale: Locale
+
+ @Before
+ fun setUp() {
+ originalLocale = Locale.getDefault()
+ }
+
+ @After
+ fun tearDown() {
+ if (Locale.getDefault() != originalLocale) {
+ Locale.setDefault(originalLocale)
+ }
+ }
+
+ @Test
+ fun `Loads search engines for locale (US)`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertNotNull(store.state.search.regionSearchEngines.find { it.name == "Google" })
+ assertNull(store.state.search.regionSearchEngines.find { it.name == "Yandex" })
+ }
+
+ @Test
+ fun `WHEN distribution doesn't exist THEN Loads default search engines`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ "test",
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertNotNull(store.state.search.regionSearchEngines.find { it.name == "Google" })
+ assertNull(store.state.search.regionSearchEngines.find { it.name == "Yandex" })
+ }
+
+ fun `Loads search engines for locale (An)`() {
+ Locale.setDefault(Locale("an", "AN"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("AN", "AN"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(5, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipedia", store.state.search.regionSearchEngines[4].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (CA)`() {
+ Locale.setDefault(Locale("CA", "CA"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("CA", "CA"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(5, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Viquipèdia (ca)", store.state.search.regionSearchEngines[4].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (CY)`() {
+ Locale.setDefault(Locale("cy", "CY"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("CY", "CY"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.co.uk", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Wicipedia (cy)", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (fy-NL)`() {
+ Locale.setDefault(Locale("fy", "NL"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("FY", "NL"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(5, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipedia (fy)", store.state.search.regionSearchEngines[4].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (en-AU)`() {
+ Locale.setDefault(Locale("en", "AU"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("EN", "AU"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.com.au", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipedia", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (en-GB)`() {
+ Locale.setDefault(Locale("en", "GB"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("EN", "GB"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(7, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.co.uk", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Qwant", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Wikipedia", store.state.search.regionSearchEngines[5].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[6].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (en-IE)`() {
+ Locale.setDefault(Locale("en", "IE"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("EN", "IE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(7, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.co.uk", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Qwant", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Wikipedia", store.state.search.regionSearchEngines[5].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[6].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (fr-BE)`() {
+ Locale.setDefault(Locale("fr", "BE"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("FR", "BE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("Qwant", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipédia (fr)", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (fr-CA)`() {
+ Locale.setDefault(Locale("fr", "CA"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("FR", "CA"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.ca", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipédia (fr)", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (fr-FR)`() {
+ Locale.setDefault(Locale("fr", "FR"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("FR", "FR"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(7, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("Qwant", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipédia (fr)", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Amazon.fr", store.state.search.regionSearchEngines[5].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[6].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (de-AT)`() {
+ Locale.setDefault(Locale("de", "AT"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("DE", "AT"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(8, store.state.search.regionSearchEngines.size)
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.de", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Ecosia", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Qwant", store.state.search.regionSearchEngines[5].name)
+ assertEquals("Wikipedia (de)", store.state.search.regionSearchEngines[6].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[7].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (DE)`() {
+ Locale.setDefault(Locale("de", "DE"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("DE", "DE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(8, store.state.search.regionSearchEngines.size)
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.de", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Ecosia", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Qwant", store.state.search.regionSearchEngines[5].name)
+ assertEquals("Wikipedia (de)", store.state.search.regionSearchEngines[6].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[7].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (DSB)`() {
+ Locale.setDefault(Locale("dsb", "DE"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("DSB", "DE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.de", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipedija (dsb)", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (HSB)`() {
+ Locale.setDefault(Locale("hsb", "DE"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("HSB", "DE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.de", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipedija (hsb)", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (ES)`() {
+ Locale.setDefault(Locale("es", "ES"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("ES", "ES"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("Wikipedia (es)", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Amazon.es", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (IT)`() {
+ Locale.setDefault(Locale("it", "IT"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("it", "IT"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("Wikipedia (it)", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Amazon.it", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for Locale (lij)`() {
+ Locale.setDefault(Locale("lij", "ZE"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("lij", "ZE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.it", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipedia (lij)", store.state.search.regionSearchEngines[4].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (SE)`() {
+ Locale.setDefault(Locale("sv", "SE"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("sv", "SE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(7, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Prisjakt", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("Wikipedia (sv)", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Amazon.se", store.state.search.regionSearchEngines[5].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[6].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (PL)`() {
+ Locale.setDefault(Locale("pl", "PL"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("pl", "PL"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(5, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[2].name)
+ assertEquals("Wikipedia (pl)", store.state.search.regionSearchEngines[3].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[4].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (RU)`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("RU", "RU"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertNull(store.state.search.regionSearchEngines.find { it.name == "Yandex" })
+ assertNotNull(store.state.search.regionSearchEngines.find { it.name == "Google" })
+ assertNotNull(store.state.search.regionSearchEngines.find { it.name == "DuckDuckGo" })
+ }
+
+ @Test
+ fun `Loads additional search engines`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ additionalBundledSearchEngineIds = listOf("reddit", "youtube"),
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(2, store.state.search.additionalAvailableSearchEngines.size)
+
+ val first = store.state.search.additionalAvailableSearchEngines[0]
+ assertEquals("Reddit", first.name)
+ assertEquals("reddit", first.id)
+
+ val second = store.state.search.additionalAvailableSearchEngines[1]
+ assertEquals("YouTube", second.name)
+ assertEquals("youtube", second.id)
+
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+ }
+
+ @Test
+ fun `Loads additional search engine and honors user choice`() = runTestOnMain {
+ val metadataStorage = SearchMetadataStorage(testContext, preferences = lazy { FakeSharedPreferences() })
+ metadataStorage.setAdditionalSearchEngines(listOf("reddit"))
+
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ additionalBundledSearchEngineIds = listOf("reddit", "youtube"),
+ metadataStorage = metadataStorage,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isNotEmpty())
+
+ assertEquals(1, store.state.search.additionalAvailableSearchEngines.size)
+ assertEquals(1, store.state.search.additionalSearchEngines.size)
+
+ val additional = store.state.search.additionalSearchEngines[0]
+ assertEquals("Reddit", additional.name)
+ assertEquals("reddit", additional.id)
+
+ val available = store.state.search.additionalAvailableSearchEngines[0]
+ assertEquals("YouTube", available.name)
+ assertEquals("youtube", available.id)
+
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNotNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+ }
+
+ @Test
+ fun `Loads custom search engines`() = runTestOnMain {
+ val searchEngine = SearchEngine(
+ id = "test-search",
+ name = "Test Engine",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf(),
+ suggestUrl = null,
+ )
+
+ val storage = CustomSearchEngineStorage(testContext, dispatcher)
+ storage.saveSearchEngine(searchEngine)
+
+ val store = BrowserStore(
+ middleware = listOf(
+ SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = storage,
+ ),
+ ),
+ )
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.customSearchEngines.isNotEmpty())
+ assertNull(store.state.search.userSelectedSearchEngineId)
+ }
+
+ @Test
+ fun `Loads default search engine ID`() = runTestOnMain {
+ val storage = SearchMetadataStorage(testContext)
+ storage.setUserSelectedSearchEngine("test-id", null)
+
+ val middleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ metadataStorage = storage,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(middleware),
+ )
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals("test-id", store.state.search.userSelectedSearchEngineId)
+ }
+
+ @Test
+ fun `Update default search engine`() {
+ val storage = SearchMetadataStorage(testContext)
+ val id = "test-id-${UUID.randomUUID()}"
+
+ run {
+ val store = BrowserStore(
+ middleware = listOf(
+ SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ metadataStorage = storage,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ ),
+ ),
+ )
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertNull(store.state.search.userSelectedSearchEngineId)
+
+ store.dispatch(SearchAction.SelectSearchEngineAction(id, null)).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals(id, store.state.search.userSelectedSearchEngineId)
+ }
+
+ run {
+ val store = BrowserStore(
+ middleware = listOf(
+ SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ metadataStorage = storage,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ ),
+ ),
+ )
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals(id, store.state.search.userSelectedSearchEngineId)
+ }
+ }
+
+ @Test
+ fun `Updates and persists additional search engines`() {
+ val storage = SearchMetadataStorage(testContext, preferences = lazy { FakeSharedPreferences() })
+ val middleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ metadataStorage = storage,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ additionalBundledSearchEngineIds = listOf(
+ "reddit",
+ "youtube",
+ ),
+ )
+
+ // First run: Add additional search engine
+ run {
+ val store = BrowserStore(middleware = listOf(middleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(2, store.state.search.additionalAvailableSearchEngines.size)
+
+ val first = store.state.search.additionalAvailableSearchEngines[0]
+ assertEquals("Reddit", first.name)
+ assertEquals("reddit", first.id)
+
+ val second = store.state.search.additionalAvailableSearchEngines[1]
+ assertEquals("YouTube", second.name)
+ assertEquals("youtube", second.id)
+
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ store.dispatch(
+ SearchAction.AddAdditionalSearchEngineAction("youtube"),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isNotEmpty())
+
+ assertEquals(1, store.state.search.additionalSearchEngines.size)
+ assertEquals(1, store.state.search.additionalAvailableSearchEngines.size)
+
+ assertEquals("Reddit", store.state.search.additionalAvailableSearchEngines[0].name)
+ assertEquals("reddit", store.state.search.additionalAvailableSearchEngines[0].id)
+
+ assertEquals("YouTube", store.state.search.additionalSearchEngines[0].name)
+ assertEquals("youtube", store.state.search.additionalSearchEngines[0].id)
+
+ assertNotNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ assertNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+ }
+
+ // Second run: Restores additional search engine and removes it
+ run {
+ val store = BrowserStore(middleware = listOf(middleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isNotEmpty())
+
+ assertEquals(1, store.state.search.additionalSearchEngines.size)
+ assertEquals(1, store.state.search.additionalAvailableSearchEngines.size)
+
+ assertEquals("Reddit", store.state.search.additionalAvailableSearchEngines[0].name)
+ assertEquals("reddit", store.state.search.additionalAvailableSearchEngines[0].id)
+
+ assertEquals("YouTube", store.state.search.additionalSearchEngines[0].name)
+ assertEquals("youtube", store.state.search.additionalSearchEngines[0].id)
+
+ assertNotNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ assertNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ store.dispatch(
+ SearchAction.RemoveAdditionalSearchEngineAction(
+ "youtube",
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(2, store.state.search.additionalAvailableSearchEngines.size)
+
+ val first = store.state.search.additionalAvailableSearchEngines[0]
+ assertEquals("Reddit", first.name)
+ assertEquals("reddit", first.id)
+
+ val second = store.state.search.additionalAvailableSearchEngines[1]
+ assertEquals("YouTube", second.name)
+ assertEquals("youtube", second.id)
+
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+ }
+
+ // Third run: Restores without additional search engine
+ run {
+ val store = BrowserStore(middleware = listOf(middleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(2, store.state.search.additionalAvailableSearchEngines.size)
+
+ val first = store.state.search.additionalAvailableSearchEngines[0]
+ assertEquals("Reddit", first.name)
+ assertEquals("reddit", first.id)
+
+ val second = store.state.search.additionalAvailableSearchEngines[1]
+ assertEquals("YouTube", second.name)
+ assertEquals("youtube", second.id)
+
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNull(store.state.search.searchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "youtube" })
+ assertNotNull(store.state.search.availableSearchEngines.find { searchEngine -> searchEngine.id == "reddit" })
+ }
+ }
+
+ @Test
+ fun `Custom search engines - Create, Update, Delete`() {
+ runTestOnMain {
+ val storage: SearchMiddleware.CustomStorage = mock()
+ doReturn(emptyList<SearchEngine>()).`when`(storage).loadSearchEngineList()
+
+ val store = BrowserStore(
+ middleware = listOf(
+ SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = storage,
+ ),
+ ),
+ )
+
+ store.dispatch(
+ SearchAction.SetRegionAction(RegionState.Default),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.customSearchEngines.isEmpty())
+ verify(storage).loadSearchEngineList()
+ verifyNoMoreInteractions(storage)
+
+ // Add a custom search engine
+
+ val engine1 = SearchEngine("test-id-1", "test engine one", mock(), type = SearchEngine.Type.CUSTOM)
+ store.dispatch(SearchAction.UpdateCustomSearchEngineAction(engine1)).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.customSearchEngines.isNotEmpty())
+ assertEquals(1, store.state.search.customSearchEngines.size)
+ verify(storage).saveSearchEngine(engine1)
+ verifyNoMoreInteractions(storage)
+
+ // Add another custom search engine
+
+ val engine2 = SearchEngine("test-id-2", "test engine two", mock(), type = SearchEngine.Type.CUSTOM)
+ store.dispatch(SearchAction.UpdateCustomSearchEngineAction(engine2)).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.customSearchEngines.isNotEmpty())
+ assertEquals(2, store.state.search.customSearchEngines.size)
+ verify(storage).saveSearchEngine(engine2)
+ verifyNoMoreInteractions(storage)
+
+ assertEquals("test engine one", store.state.search.customSearchEngines[0].name)
+ assertEquals("test engine two", store.state.search.customSearchEngines[1].name)
+
+ // Update first engine
+
+ val updated = engine1.copy(
+ name = "updated engine",
+ )
+ store.dispatch(SearchAction.UpdateCustomSearchEngineAction(updated)).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.customSearchEngines.isNotEmpty())
+ assertEquals(2, store.state.search.customSearchEngines.size)
+ verify(storage).saveSearchEngine(updated)
+ verifyNoMoreInteractions(storage)
+
+ assertEquals("updated engine", store.state.search.customSearchEngines[0].name)
+ assertEquals("test engine two", store.state.search.customSearchEngines[1].name)
+
+ // Remove second engine
+
+ store.dispatch(SearchAction.RemoveCustomSearchEngineAction(engine2.id)).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.customSearchEngines.isNotEmpty())
+ assertEquals(1, store.state.search.customSearchEngines.size)
+ verify(storage).removeSearchEngine(engine2.id)
+ verifyNoMoreInteractions(storage)
+
+ assertEquals("updated engine", store.state.search.customSearchEngines[0].name)
+ }
+ }
+
+ @Test
+ fun `GIVEN disabled engines list contains elements WHEN metadata storage is created THEN the engines are disabled`() = runTestOnMain {
+ val additionalBundledSearchEngineIds = setOf("reddit", "youtube")
+ val metadataStorage = SearchMetadataStorage(
+ testContext,
+ additionalBundledSearchEngineIds,
+ lazy { FakeSharedPreferences() },
+ )
+ val disabledSearchEngineIds = metadataStorage.getDisabledSearchEngineIds()
+ assertTrue(disabledSearchEngineIds.contains("reddit"))
+ assertTrue(disabledSearchEngineIds.contains("youtube"))
+ }
+
+ @Test
+ fun `WHEN update disabled engine action is sent THEN search state and storage get updated`() = runTestOnMain {
+ val metadataStorage = SearchMetadataStorage(testContext, preferences = lazy { FakeSharedPreferences() })
+ metadataStorage.setAdditionalSearchEngines(listOf("reddit"))
+
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ additionalBundledSearchEngineIds = listOf("reddit", "youtube"),
+ metadataStorage = metadataStorage,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertFalse(metadataStorage.getDisabledSearchEngineIds().contains("bing"))
+ assertFalse(store.state.search.disabledSearchEngineIds.contains("bing"))
+
+ store.dispatch(
+ SearchAction.UpdateDisabledSearchEngineIdsAction(
+ "bing",
+ false,
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(metadataStorage.getDisabledSearchEngineIds().contains("bing"))
+ assertTrue(store.state.search.disabledSearchEngineIds.contains("bing"))
+
+ store.dispatch(
+ SearchAction.UpdateDisabledSearchEngineIdsAction(
+ "bing",
+ true,
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertFalse(metadataStorage.getDisabledSearchEngineIds().contains("bing"))
+ assertFalse(store.state.search.disabledSearchEngineIds.contains("bing"))
+ }
+
+ @Test
+ fun `WHEN restore hidden search engines action THEN hidden engines are added back to bundled engines list`() = runTestOnMain {
+ val metadataStorage = SearchMetadataStorage(testContext, preferences = lazy { FakeSharedPreferences() })
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ metadataStorage = metadataStorage,
+ )
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+ wait(store, dispatcher)
+
+ val google = store.state.search.regionSearchEngines.find { searchEngine -> searchEngine.name == "Google" }
+ assertNotNull(google!!)
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+ assertEquals(0, metadataStorage.getHiddenSearchEngines().size)
+
+ store.dispatch(SearchAction.HideSearchEngineAction(google.id)).joinBlocking()
+ wait(store, dispatcher)
+
+ assertNull(store.state.search.regionSearchEngines.find { it.id == google.id })
+
+ assertEquals(1, store.state.search.hiddenSearchEngines.size)
+ assertEquals(1, metadataStorage.getHiddenSearchEngines().size)
+ assertNotNull(store.state.search.hiddenSearchEngines.find { it.id == google.id })
+ assertNotNull(metadataStorage.getHiddenSearchEngines().find { it == google.id })
+
+ store.dispatch(SearchAction.RestoreHiddenSearchEnginesAction).joinBlocking()
+ wait(store, dispatcher)
+
+ assertNotNull(store.state.search.regionSearchEngines.find { it.id == google.id })
+
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+ assertEquals(0, metadataStorage.getHiddenSearchEngines().size)
+ assertNull(store.state.search.hiddenSearchEngines.find { it.id == google.id })
+ assertNull(metadataStorage.getHiddenSearchEngines().find { it == google.id })
+ }
+
+ @Test
+ fun `Hiding and showing search engines`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ metadataStorage = SearchMetadataStorage(testContext),
+ )
+
+ val google = BrowserStore(middleware = listOf(searchMiddleware)).let { store ->
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ store.state.search.regionSearchEngines.find { searchEngine -> searchEngine.name == "Google" }
+ }
+ assertNotNull(google!!)
+
+ run {
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertNotNull(store.state.search.regionSearchEngines.find { it.id == google.id })
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+
+ store.dispatch(
+ SearchAction.HideSearchEngineAction(google.id),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertNull(store.state.search.regionSearchEngines.find { it.id == google.id })
+ assertEquals(1, store.state.search.hiddenSearchEngines.size)
+ assertNotNull(store.state.search.hiddenSearchEngines.find { it.id == google.id })
+ }
+
+ run {
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertNull(store.state.search.regionSearchEngines.find { it.id == google.id })
+ assertEquals(1, store.state.search.hiddenSearchEngines.size)
+ assertNotNull(store.state.search.hiddenSearchEngines.find { it.id == google.id })
+
+ store.dispatch(
+ SearchAction.ShowSearchEngineAction(google.id),
+ ).joinBlocking()
+
+ assertNotNull(store.state.search.regionSearchEngines.find { it.id == google.id })
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+ }
+
+ run {
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertNotNull(store.state.search.regionSearchEngines.find { it.id == google.id })
+ assertEquals(0, store.state.search.hiddenSearchEngines.size)
+ }
+ }
+
+ @Test
+ fun `Keeps user choice based on search engine name even if search engine id changes`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ metadataStorage = SearchMetadataStorage(testContext),
+ )
+
+ run {
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ val google = store.state.search.searchEngines.find { it.name == "Google" }
+ assertNotNull(google!!)
+ assertEquals("google-b-1-m", google.id)
+
+ store.dispatch(
+ SearchAction.SelectSearchEngineAction(
+ searchEngineId = "google-b-1-m",
+ searchEngineName = "Google",
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals("google-b-1-m", store.state.search.userSelectedSearchEngineId)
+ assertEquals("Google", store.state.search.userSelectedSearchEngineName)
+
+ val searchEngine = store.state.search.selectedOrDefaultSearchEngine
+ assertNotNull(searchEngine!!)
+ assertEquals("google-b-1-m", searchEngine.id)
+ assertEquals("Google", searchEngine.name)
+ }
+
+ run {
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("DE", "DE"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals("google-b-1-m", store.state.search.userSelectedSearchEngineId)
+ assertEquals("Google", store.state.search.userSelectedSearchEngineName)
+
+ val searchEngine = store.state.search.selectedOrDefaultSearchEngine
+ assertNotNull(searchEngine!!)
+ assertEquals("google-b-m", searchEngine.id)
+ assertEquals("Google", searchEngine.name)
+ }
+ }
+
+ @Test
+ fun `Adding and restoring custom search engine`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ metadataStorage = SearchMetadataStorage(testContext),
+ )
+
+ run {
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals(0, store.state.search.customSearchEngines.size)
+
+ store.dispatch(
+ SearchAction.UpdateCustomSearchEngineAction(
+ SearchEngine(
+ id = UUID.randomUUID().toString(),
+ name = "Example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf(
+ "https://example.org/?q=%s",
+ ),
+ ),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals(1, store.state.search.customSearchEngines.size)
+ }
+
+ run {
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals(1, store.state.search.customSearchEngines.size)
+ }
+ }
+
+ @Test
+ fun `Migration - custom search engine and default search engine`() {
+ val customStorage = CustomSearchEngineStorage(testContext, dispatcher)
+ val metadataStorage = SearchMetadataStorage(testContext)
+
+ run {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = customStorage,
+ metadataStorage = metadataStorage,
+ migration = object : SearchMiddleware.Migration {
+ override fun getValuesToMigrate() = SearchMiddleware.Migration.MigrationValues(
+ customSearchEngines = listOf(
+ createSearchEngine(
+ name = "Example",
+ url = "https://example.org/?q={searchTerms}",
+ icon = mock(),
+ ),
+ ),
+ defaultSearchEngineName = "Example",
+ )
+ },
+ )
+
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals(1, store.state.search.customSearchEngines.size)
+
+ val selectedSearchEngine = store.state.search.selectedOrDefaultSearchEngine
+ assertNotNull(selectedSearchEngine!!)
+
+ assertEquals("Example", selectedSearchEngine.name)
+ assertEquals("https://example.org/?q={searchTerms}", selectedSearchEngine.resultUrls[0])
+ }
+
+ run {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = customStorage,
+ metadataStorage = metadataStorage,
+ )
+
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertEquals(1, store.state.search.customSearchEngines.size)
+
+ val selectedSearchEngine = store.state.search.selectedOrDefaultSearchEngine
+ assertNotNull(selectedSearchEngine!!)
+
+ assertEquals("Example", selectedSearchEngine.name)
+ assertEquals("https://example.org/?q={searchTerms}", selectedSearchEngine.resultUrls[0])
+ }
+ }
+
+ @Test
+ fun `Migration - default search engine`() {
+ val customStorage = CustomSearchEngineStorage(testContext, dispatcher)
+ val metadataStorage = SearchMetadataStorage(testContext)
+
+ run {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = customStorage,
+ metadataStorage = metadataStorage,
+ migration = object : SearchMiddleware.Migration {
+ override fun getValuesToMigrate() = SearchMiddleware.Migration.MigrationValues(
+ customSearchEngines = listOf(),
+ defaultSearchEngineName = "Amazon.com",
+ )
+ },
+ )
+
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ val selectedSearchEngine = store.state.search.selectedOrDefaultSearchEngine
+ assertNotNull(selectedSearchEngine!!)
+
+ assertEquals("Amazon.com", selectedSearchEngine.name)
+ assertTrue(selectedSearchEngine.resultUrls[0].startsWith("https://www.amazon.com/"))
+ }
+
+ run {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = customStorage,
+ metadataStorage = metadataStorage,
+ )
+
+ val store = BrowserStore(middleware = listOf(searchMiddleware))
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ val selectedSearchEngine = store.state.search.selectedOrDefaultSearchEngine
+ assertNotNull(selectedSearchEngine!!)
+
+ assertEquals("Amazon.com", selectedSearchEngine.name)
+ assertTrue(selectedSearchEngine.resultUrls[0].startsWith("https://www.amazon.com/"))
+ }
+ }
+
+ @Test
+ fun `Reorders list of region search engines after adding previously removed search engines`() {
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("US", "US"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ // ///////////////////////////////////////////////////////////////////////////////////////////
+ // Verify initial state
+ // ///////////////////////////////////////////////////////////////////////////////////////////
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.com", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Wikipedia", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+
+ store.dispatch(
+ SearchAction.HideSearchEngineAction(
+ "google-b-1-m",
+ ),
+ ).joinBlocking()
+
+ store.dispatch(
+ SearchAction.HideSearchEngineAction(
+ "ddg",
+ ),
+ ).joinBlocking()
+
+ // ///////////////////////////////////////////////////////////////////////////////////////////
+ // Verify after hiding search engines
+ // ///////////////////////////////////////////////////////////////////////////////////////////
+
+ assertEquals(4, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Bing", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Amazon.com", store.state.search.regionSearchEngines[1].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[2].name)
+ assertEquals("Wikipedia", store.state.search.regionSearchEngines[3].name)
+
+ assertEquals("Bing", store.state.search.selectedOrDefaultSearchEngine!!.name)
+
+ println(store.state.search.regionSearchEngines)
+
+ store.dispatch(
+ SearchAction.ShowSearchEngineAction("google-b-1-m"),
+ ).joinBlocking()
+
+ store.dispatch(
+ SearchAction.ShowSearchEngineAction("ddg"),
+ ).joinBlocking()
+
+ // ///////////////////////////////////////////////////////////////////////////////////////////
+ // Verify state after adding search engines back
+ // ///////////////////////////////////////////////////////////////////////////////////////////
+
+ assertEquals(6, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.com", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("eBay", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Wikipedia", store.state.search.regionSearchEngines[5].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+
+ @Test
+ fun `Loads search engines for locale (JA)`() {
+ Locale.setDefault(Locale("ja", "JA"))
+ val searchMiddleware = SearchMiddleware(
+ testContext,
+ ioDispatcher = dispatcher,
+ customStorage = CustomSearchEngineStorage(testContext, dispatcher),
+ )
+
+ val store = BrowserStore(
+ middleware = listOf(searchMiddleware),
+ )
+
+ assertTrue(store.state.search.regionSearchEngines.isEmpty())
+
+ store.dispatch(
+ SearchAction.SetRegionAction(
+ RegionState("JA", "JA"),
+ ),
+ ).joinBlocking()
+
+ wait(store, dispatcher)
+
+ assertTrue(store.state.search.regionSearchEngines.isNotEmpty())
+ assertTrue(store.state.search.additionalAvailableSearchEngines.isEmpty())
+ assertTrue(store.state.search.additionalSearchEngines.isEmpty())
+
+ assertEquals(8, store.state.search.regionSearchEngines.size)
+
+ assertEquals("Google", store.state.search.regionSearchEngines[0].name)
+ assertEquals("Bing", store.state.search.regionSearchEngines[1].name)
+ assertEquals("Amazon.co.jp", store.state.search.regionSearchEngines[2].name)
+ assertEquals("DuckDuckGo", store.state.search.regionSearchEngines[3].name)
+ assertEquals("楽天市場", store.state.search.regionSearchEngines[4].name)
+ assertEquals("Wikipedia (ja)", store.state.search.regionSearchEngines[5].name)
+ assertEquals("Yahoo! JAPAN", store.state.search.regionSearchEngines[6].name)
+ assertEquals("Yahoo!オークション", store.state.search.regionSearchEngines[7].name)
+
+ assertEquals("Google", store.state.search.selectedOrDefaultSearchEngine!!.name)
+ }
+}
+
+private fun wait(store: BrowserStore, dispatcher: TestDispatcher) {
+ // First we wait for the InitAction that may still need to be processed.
+ store.waitUntilIdle()
+
+ // Now we wait for the Middleware that may need to asynchronously process an action the test dispatched
+ dispatcher.scheduler.advanceUntilIdle()
+
+ // Since the Middleware may have dispatched an action, we now wait for the store again.
+ store.waitUntilIdle()
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionManagerTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionManagerTest.kt
new file mode 100644
index 0000000000..1103680e3e
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionManagerTest.kt
@@ -0,0 +1,146 @@
+/* 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/. */
+
+package mozilla.components.feature.search.region
+
+import kotlinx.coroutines.test.runTest
+import mozilla.components.service.location.LocationService
+import mozilla.components.support.test.fakes.FakeClock
+import mozilla.components.support.test.fakes.android.FakeContext
+import mozilla.components.support.test.fakes.android.FakeSharedPreferences
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertNull
+import org.junit.Test
+
+class RegionManagerTest {
+ @Test
+ fun `Initial state`() {
+ val regionManager = RegionManager(
+ context = FakeContext(),
+ locationService = FakeLocationService(),
+ currentTime = FakeClock()::time,
+ preferences = lazy { FakeSharedPreferences() },
+ )
+
+ assertNull(regionManager.region())
+ }
+
+ @Test
+ fun `First update`() = runTest {
+ val locationService = FakeLocationService(
+ region = LocationService.Region("DE", "Germany"),
+ )
+
+ val regionManager = RegionManager(
+ context = FakeContext(),
+ locationService = locationService,
+ currentTime = FakeClock()::time,
+ preferences = lazy { FakeSharedPreferences() },
+ )
+
+ val updatedRegion = regionManager.update()
+ assertNotNull(updatedRegion!!)
+ assertEquals("DE", updatedRegion.current)
+ assertEquals("DE", updatedRegion.home)
+ }
+
+ @Test
+ fun `Updating to new home region`() = runTest {
+ val clock = FakeClock()
+
+ val locationService = FakeLocationService(
+ region = LocationService.Region("DE", "Germany"),
+ )
+
+ val regionManager = RegionManager(
+ context = FakeContext(),
+ locationService = locationService,
+ currentTime = clock::time,
+ preferences = lazy { FakeSharedPreferences() },
+ )
+
+ regionManager.update()
+
+ locationService.region = LocationService.Region("FR", "France")
+
+ // Should not be updated since the "home" region didn't change
+ assertNull(regionManager.update())
+ assertEquals("DE", regionManager.region()?.home)
+ assertEquals("FR", regionManager.region()?.current)
+
+ // Let's jump one week into the future!
+ clock.advanceBy(60L * 60L * 24L * 7L * 1000L)
+
+ // Still not updated because we switch after two weeks
+ assertNull(regionManager.update())
+ assertEquals("DE", regionManager.region()?.home)
+ assertEquals("FR", regionManager.region()?.current)
+
+ // Let's move the clock 8 more days into the future
+ clock.advanceBy(60L * 60L * 24L * 8L * 1000L)
+
+ val updatedRegion = (regionManager.update())
+ assertNotNull(updatedRegion!!)
+ assertEquals("FR", updatedRegion.home)
+ assertEquals("FR", updatedRegion.current)
+ assertEquals("FR", regionManager.region()?.home)
+ assertEquals("FR", regionManager.region()?.current)
+ }
+
+ @Test
+ fun `Switching back to home region after staying in different region shortly`() = runTest {
+ val clock = FakeClock()
+
+ val locationService = FakeLocationService(
+ region = LocationService.Region("DE", "Germany"),
+ )
+
+ val regionManager = RegionManager(
+ context = FakeContext(),
+ locationService = locationService,
+ currentTime = clock::time,
+ preferences = lazy { FakeSharedPreferences() },
+ )
+
+ regionManager.update()
+
+ // Let's jump one week into the future!
+ clock.advanceBy(60L * 60L * 24L * 7L * 1000L)
+
+ locationService.region = LocationService.Region("FR", "France")
+
+ // Should not be updated since the "home" region didn't change
+ assertNull(regionManager.update())
+ assertEquals("DE", regionManager.region()?.home)
+ assertEquals("FR", regionManager.region()?.current)
+
+ // Next day, we are back in the home region
+ clock.advanceBy(60L * 60L * 24L * 1000L)
+
+ locationService.region = LocationService.Region("DE", "Germany")
+ assertNull(regionManager.update())
+ assertEquals("DE", regionManager.region()?.home)
+ assertEquals("DE", regionManager.region()?.current)
+
+ // Another week forward, we are back in France
+ clock.advanceBy(60L * 60L * 24L * 7L * 1000L)
+
+ locationService.region = LocationService.Region("FR", "France")
+
+ // The "home" region should not have changed since we haven't been in the other region the
+ // whole time.
+ assertNull(regionManager.update())
+ assertEquals("DE", regionManager.region()?.home)
+ assertEquals("FR", regionManager.region()?.current)
+ }
+}
+
+class FakeLocationService(
+ var region: LocationService.Region? = null,
+ private val hasRegionCached: Boolean = false,
+) : LocationService {
+ override suspend fun fetchRegion(readFromCache: Boolean): LocationService.Region? = region
+ override fun hasRegionCached(): Boolean = hasRegionCached
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionMiddlewareTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionMiddlewareTest.kt
new file mode 100644
index 0000000000..b05dc5e7c9
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/region/RegionMiddlewareTest.kt
@@ -0,0 +1,153 @@
+/* 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/. */
+
+package mozilla.components.feature.search.region
+
+import mozilla.components.browser.state.action.InitAction
+import mozilla.components.browser.state.action.SearchAction.RefreshSearchEnginesAction
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.service.location.LocationService
+import mozilla.components.support.test.ext.joinBlocking
+import mozilla.components.support.test.fakes.FakeClock
+import mozilla.components.support.test.fakes.android.FakeContext
+import mozilla.components.support.test.fakes.android.FakeSharedPreferences
+import mozilla.components.support.test.libstate.ext.waitUntilIdle
+import mozilla.components.support.test.rule.MainCoroutineRule
+import mozilla.components.support.test.rule.runTestOnMain
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNotEquals
+import org.junit.Before
+import org.junit.Rule
+import org.junit.Test
+
+class RegionMiddlewareTest {
+
+ @get:Rule
+ val coroutinesTestRule = MainCoroutineRule()
+ private val dispatcher = coroutinesTestRule.testDispatcher
+
+ private lateinit var locationService: FakeLocationService
+ private lateinit var clock: FakeClock
+ private lateinit var regionManager: RegionManager
+
+ @Before
+ fun setUp() {
+ clock = FakeClock()
+ locationService = FakeLocationService()
+ regionManager = RegionManager(
+ context = FakeContext(),
+ locationService = locationService,
+ currentTime = clock::time,
+ preferences = lazy { FakeSharedPreferences() },
+ )
+ }
+
+ @Test
+ fun `Updates region on init`() {
+ val middleware = RegionMiddleware(FakeContext(), locationService, dispatcher)
+ middleware.regionManager = regionManager
+
+ locationService.region = LocationService.Region("FR", "France")
+
+ val store = BrowserStore(
+ middleware = listOf(middleware),
+ )
+
+ store.waitUntilIdle()
+ middleware.updateJob?.joinBlocking()
+ store.waitUntilIdle()
+
+ assertNotEquals(RegionState.Default, store.state.search.region)
+ assertEquals("FR", store.state.search.region!!.home)
+ assertEquals("FR", store.state.search.region!!.current)
+ }
+
+ @Test
+ fun `Uses default region if could never get updated`() {
+ val middleware = RegionMiddleware(FakeContext(), locationService, dispatcher)
+ middleware.regionManager = regionManager
+
+ val store = BrowserStore(
+ middleware = listOf(middleware),
+ )
+
+ store.dispatch(InitAction).joinBlocking()
+
+ dispatcher.scheduler.advanceUntilIdle()
+ store.waitUntilIdle()
+
+ assertEquals(RegionState.Default, store.state.search.region)
+ assertEquals("XX", store.state.search.region!!.home)
+ assertEquals("XX", store.state.search.region!!.current)
+ }
+
+ @Test
+ fun `Dispatches cached home region and update later`() = runTestOnMain {
+ val middleware = RegionMiddleware(FakeContext(), locationService, dispatcher)
+ middleware.regionManager = regionManager
+
+ locationService.region = LocationService.Region("FR", "France")
+ regionManager.update()
+
+ val store = BrowserStore(
+ middleware = listOf(middleware),
+ )
+
+ store.dispatch(InitAction).joinBlocking()
+ middleware.updateJob?.joinBlocking()
+ store.waitUntilIdle()
+
+ assertEquals("FR", store.state.search.region!!.home)
+ assertEquals("FR", store.state.search.region!!.current)
+
+ locationService.region = LocationService.Region("DE", "Germany")
+ regionManager.update()
+
+ store.dispatch(InitAction).joinBlocking()
+ middleware.updateJob?.joinBlocking()
+ store.waitUntilIdle()
+
+ assertEquals("FR", store.state.search.region!!.home)
+ assertEquals("DE", store.state.search.region!!.current)
+
+ clock.advanceBy(1000L * 60L * 60L * 24L * 21L)
+
+ store.dispatch(InitAction).joinBlocking()
+ middleware.updateJob?.joinBlocking()
+ store.waitUntilIdle()
+
+ assertEquals("DE", store.state.search.region!!.home)
+ assertEquals("DE", store.state.search.region!!.current)
+ }
+
+ @Test
+ fun `GIVEN a locale is already selected WHEN the locale changes THEN update region on RefreshSearchEngines`() = runTestOnMain {
+ val middleware = RegionMiddleware(FakeContext(), locationService, dispatcher)
+ middleware.regionManager = regionManager
+
+ locationService.region = LocationService.Region("FR", "France")
+
+ val store = BrowserStore(
+ middleware = listOf(middleware),
+ )
+
+ store.dispatch(InitAction).joinBlocking()
+ middleware.updateJob?.joinBlocking()
+ store.waitUntilIdle()
+
+ assertEquals("FR", store.state.search.region!!.home)
+ assertEquals("FR", store.state.search.region!!.current)
+
+ locationService.region = LocationService.Region("DE", "Germany")
+ regionManager.update()
+
+ store.dispatch(RefreshSearchEnginesAction).joinBlocking()
+ middleware.updateJob?.joinBlocking()
+ store.waitUntilIdle()
+
+ assertEquals("FR", store.state.search.region!!.home)
+ assertEquals("DE", store.state.search.region!!.current)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorageTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorageTest.kt
new file mode 100644
index 0000000000..bfd6691432
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/BundledSearchEnginesStorageTest.kt
@@ -0,0 +1,196 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import kotlinx.coroutines.test.runTest
+import mozilla.components.browser.state.search.RegionState
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.support.test.robolectric.testContext
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertFalse
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertTrue
+import org.junit.Test
+import org.junit.runner.RunWith
+import java.util.Locale
+
+@RunWith(AndroidJUnit4::class)
+class BundledSearchEnginesStorageTest {
+ @Test
+ fun `Load search engines for en-US from assets`() = runTest {
+ val storage = BundledSearchEnginesStorage(testContext)
+
+ val engines = storage.load(RegionState("US", "US"), Locale("en", "US"))
+ val searchEngines = engines.list
+
+ assertEquals(6, searchEngines.size)
+ }
+
+ @Test
+ fun `Load search engines for all known locales without region`() = runTest {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val locales = Locale.getAvailableLocales()
+ assertTrue(locales.isNotEmpty())
+
+ for (locale in locales) {
+ val engines = storage.load(RegionState.Default, locale)
+ assertTrue(engines.list.isNotEmpty())
+ assertFalse(engines.defaultSearchEngineId.isEmpty())
+ }
+ }
+
+ @Test
+ fun `Load search engines for de-DE with global US region override`() = runTest {
+ // Without region
+ run {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState.Default, Locale("de", "DE"))
+ val searchEngines = engines.list
+
+ assertEquals(8, searchEngines.size)
+ assertContainsSearchEngine("google-b-m", searchEngines)
+ assertContainsNotSearchEngine("google-2018", searchEngines)
+ }
+ // With region
+ run {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState("US", "US"), Locale("de", "DE"))
+ val searchEngines = engines.list
+
+ assertEquals(8, searchEngines.size)
+ assertContainsSearchEngine("google-b-1-m", searchEngines)
+ assertContainsNotSearchEngine("google", searchEngines)
+ }
+ }
+
+ @Test
+ fun `Load search engines for en-US with local RU region override`() = runTest {
+ // Without region
+ run {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState.Default, Locale("en", "US"))
+ val searchEngines = engines.list
+
+ println("searchEngines = $searchEngines")
+ assertEquals(6, searchEngines.size)
+ assertContainsNotSearchEngine("yandex-en", searchEngines)
+ }
+ // With region
+ run {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState("RU", "RU"), Locale("en", "US"))
+ val searchEngines = engines.list
+
+ println("searchEngines = $searchEngines")
+ assertEquals(5, searchEngines.size)
+ assertContainsSearchEngine("google-com-nocodes", searchEngines)
+ assertContainsNotSearchEngine("yandex-en", searchEngines)
+ }
+ }
+
+ @Test
+ fun `Load search engines for zh-CN_CN locale with searchDefault override`() = runTest {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState("CN", "CN"), Locale("zh", "CN"))
+ val searchEngines = engines.list
+
+ // visibleDefaultEngines: ["google-b-m", "bing", "baidu", "ddg", "wikipedia-zh-CN"]
+ // searchOrder (default): ["Google", "Bing"]
+
+ assertEquals(
+ listOf("google-b-m", "bing", "baidu", "ddg", "wikipedia-zh-CN"),
+ searchEngines.map { it.id },
+ )
+
+ // searchDefault: "百度"
+ val default = searchEngines.find { it.id == engines.defaultSearchEngineId }
+ assertNotNull(default)
+ assertEquals("baidu", default!!.id)
+ }
+
+ @Test
+ fun `Load search engines for ru_RU locale with engines not in searchOrder`() = runTest {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState("RU", "RU"), Locale("ru", "RU"))
+ val searchEngines = engines.list
+
+ assertEquals(
+ listOf("google-com-nocodes", "ddg", "wikipedia-ru"),
+ searchEngines.map { it.id },
+ )
+
+ // searchDefault: "Google"
+ val default = searchEngines.find { it.id == engines.defaultSearchEngineId }
+ assertNotNull(default)
+ assertEquals("google-com-nocodes", default!!.id)
+ }
+
+ @Test
+ fun `Load search engines for trs locale with non-google initial engines and no default`() = runTest {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState.Default, Locale("trs", ""))
+ val searchEngines = engines.list
+
+ // visibleDefaultEngines: ["google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-es"]
+ // searchOrder (default): ["Google", "Bing"]
+
+ assertEquals(
+ listOf("google-b-m", "bing", "amazondotcom", "ddg", "wikipedia-es"),
+ searchEngines.map { it.id },
+ )
+
+ // searchDefault (default): "Google"
+ val default = searchEngines.find { it.id == engines.defaultSearchEngineId }
+ assertNotNull(default)
+ assertEquals("google-b-m", default!!.id)
+ }
+
+ @Test
+ fun `Load search engines for locale not in configuration`() = runTest {
+ val storage = BundledSearchEnginesStorage(testContext)
+ val engines = storage.load(RegionState.Default, Locale("xx", "XX"))
+ val searchEngines = engines.list
+
+ assertEquals(5, searchEngines.size)
+ }
+
+ private fun assertContainsSearchEngine(identifier: String, searchEngines: List<SearchEngine>) {
+ searchEngines.forEach {
+ if (identifier == it.id) {
+ return
+ }
+ }
+ throw AssertionError("Search engine $identifier not in list")
+ }
+
+ private fun assertContainsNotSearchEngine(identifier: String, searchEngines: List<SearchEngine>) {
+ searchEngines.forEach {
+ if (identifier == it.id) {
+ throw AssertionError("Search engine $identifier in list")
+ }
+ }
+ }
+
+ @Test
+ fun `Verify values of Google search engine`() = runTest {
+ val storage = BundledSearchEnginesStorage(testContext)
+
+ val engines = storage.load(RegionState("US", "US"), Locale("en", "US"))
+ val searchEngines = engines.list
+
+ assertEquals(6, searchEngines.size)
+
+ val google = searchEngines.find { it.name == "Google" }
+ assertNotNull(google!!)
+
+ assertEquals("google-b-1-m", google.id)
+ assertEquals("Google", google.name)
+ assertEquals(SearchEngine.Type.BUNDLED, google.type)
+ assertNotNull(google.icon)
+ assertEquals("https://www.google.com/complete/search?client=firefox&q={searchTerms}", google.suggestUrl)
+ assertTrue(google.resultUrls.isNotEmpty())
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/CustomSearchEngineStorageTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/CustomSearchEngineStorageTest.kt
new file mode 100644
index 0000000000..2da25c0b5d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/CustomSearchEngineStorageTest.kt
@@ -0,0 +1,105 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import kotlinx.coroutines.test.runTest
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.robolectric.testContext
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertTrue
+import org.junit.Ignore
+import org.junit.Test
+import org.junit.runner.RunWith
+
+@RunWith(AndroidJUnit4::class)
+class CustomSearchEngineStorageTest {
+ @Test
+ fun `saveSearchEngine successfully saves`() = runTest {
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search"),
+ )
+
+ val storage = CustomSearchEngineStorage(testContext, coroutineContext)
+ assertTrue(storage.saveSearchEngine(searchEngine))
+ assertTrue(storage.getSearchFile(searchEngine.id).baseFile.exists())
+ }
+
+ @Test
+ fun `loadSearchEngine successfully loads after saving`() = runTest {
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search"),
+ )
+
+ val storage = CustomSearchEngineStorage(testContext, coroutineContext)
+ assertTrue(storage.saveSearchEngine(searchEngine))
+
+ val storedSearchEngine = storage.loadSearchEngine(searchEngine.id)
+ assertEquals(searchEngine.id, storedSearchEngine.id)
+ assertEquals(searchEngine.name, storedSearchEngine.name)
+ assertEquals(searchEngine.type, storedSearchEngine.type)
+ assertEquals(searchEngine.resultUrls, storedSearchEngine.resultUrls)
+ }
+
+ @Test
+ @Ignore("https://github.com/mozilla-mobile/android-components/issues/8124")
+ fun `loadSearchEngineList successfully loads after saving`() = runTest {
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search"),
+ )
+ val searchEngineTwo = SearchEngine(
+ id = "id2",
+ name = "searchTwo",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.searchtwo.com/search"),
+ )
+
+ val storage = CustomSearchEngineStorage(testContext, coroutineContext)
+ assertTrue(storage.saveSearchEngine(searchEngine))
+ assertTrue(storage.saveSearchEngine(searchEngineTwo))
+
+ val storedSearchEngines = storage.loadSearchEngineList()
+ assertEquals(2, storedSearchEngines.size)
+ assertEquals(searchEngine.id, storedSearchEngines[0].id)
+ assertEquals(searchEngine.name, storedSearchEngines[0].name)
+ assertEquals(searchEngine.type, storedSearchEngines[0].type)
+ assertEquals(searchEngine.resultUrls, storedSearchEngines[0].resultUrls)
+ assertEquals(searchEngineTwo.id, storedSearchEngines[1].id)
+ assertEquals(searchEngineTwo.name, storedSearchEngines[1].name)
+ assertEquals(searchEngineTwo.type, storedSearchEngines[1].type)
+ assertEquals(searchEngineTwo.resultUrls, storedSearchEngines[1].resultUrls)
+ }
+
+ @Test
+ fun `removeSearchEngine successfully deletes`() = runTest {
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search"),
+ )
+
+ val storage = CustomSearchEngineStorage(testContext, coroutineContext)
+ assertTrue(storage.saveSearchEngine(searchEngine))
+
+ storage.removeSearchEngine(searchEngine.id)
+ assertTrue(storage.loadSearchEngineList().isEmpty())
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/ParseSearchPluginsTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/ParseSearchPluginsTest.kt
new file mode 100644
index 0000000000..662fbd76b6
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/ParseSearchPluginsTest.kt
@@ -0,0 +1,106 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.text.TextUtils
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.feature.search.middleware.SearchExtraParams
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertFalse
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertTrue
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.robolectric.ParameterizedRobolectricTestRunner
+import java.io.File
+import java.io.FileInputStream
+
+@RunWith(ParameterizedRobolectricTestRunner::class)
+class ParseSearchPluginsTest(private val searchEngineIdentifier: String) {
+
+ @Test
+ @Throws(Exception::class)
+ fun parserNoSearchExtraParams() {
+ val stream = FileInputStream(File(basePath, searchEngineIdentifier))
+ val searchEngine = SearchEngineReader(type = SearchEngine.Type.BUNDLED)
+ .loadStream(searchEngineIdentifier, stream)
+
+ assertEquals(searchEngineIdentifier, searchEngine.id)
+
+ assertNotNull(searchEngine.name)
+ assertFalse(TextUtils.isEmpty(searchEngine.name))
+
+ assertNotNull(searchEngine.icon)
+
+ val searchUrl = searchEngine.resultUrls
+ assertTrue(searchUrl.isNotEmpty())
+
+ stream.close()
+ }
+
+ @Test
+ @Throws(Exception::class)
+ fun parserWithSearchExtraParams() {
+ val stream = FileInputStream(File(basePath, searchEngineIdentifier))
+ val searchEngineName = "test"
+ val featureEnablerName = "t"
+ val featureEnablerParam = "enabled"
+ val channelIdName = "p"
+ val channelIdParam = "12345"
+ val searchExtraParams = SearchExtraParams(
+ searchEngineName = searchEngineName,
+ featureEnablerName = featureEnablerName,
+ featureEnablerParam = featureEnablerParam,
+ channelIdName = channelIdName,
+ channelIdParam = channelIdParam,
+ )
+ val searchEngine =
+ SearchEngineReader(
+ type = SearchEngine.Type.BUNDLED,
+ searchExtraParams = searchExtraParams,
+ ).loadStream(
+ identifier = searchEngineIdentifier,
+ stream = stream,
+ )
+
+ assertEquals(searchEngineIdentifier, searchEngine.id)
+
+ assertNotNull(searchEngine.name)
+ assertFalse(TextUtils.isEmpty(searchEngine.name))
+
+ assertNotNull(searchEngine.icon)
+
+ val searchUrl = searchEngine.resultUrls
+ assertTrue(searchUrl.isNotEmpty())
+
+ if (searchEngine.name.startsWith(searchEngineName)) {
+ for (url in searchUrl) {
+ assertTrue(url.contains("=$featureEnablerParam"))
+ assertTrue(url.endsWith("=$channelIdParam"))
+ }
+ } else {
+ for (url in searchUrl) {
+ assertFalse(url.endsWith("=$channelIdParam"))
+ assertFalse(url.contains("=$featureEnablerParam"))
+ }
+ }
+
+ stream.close()
+ }
+
+ companion object {
+ @JvmStatic
+ @ParameterizedRobolectricTestRunner.Parameters(name = "{0}")
+ fun searchPlugins(): Collection<Array<Any>> = basePath.list().orEmpty()
+ .mapNotNull { it as Any }
+ .map { arrayOf(it) }
+ .apply { if (isEmpty()) { throw IllegalStateException("No search plugins found.") } }
+
+ private val basePath: File
+ get() = ParseSearchPluginsTest::class.java.classLoader!!
+ .getResource("searchplugins").file
+ .let { File(it) }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineReaderTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineReaderTest.kt
new file mode 100644
index 0000000000..e1bdea0901
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineReaderTest.kt
@@ -0,0 +1,84 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.util.AtomicFile
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.robolectric.testContext
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertTrue
+import org.junit.Test
+import org.junit.runner.RunWith
+import java.io.File
+import java.io.IOException
+
+@RunWith(AndroidJUnit4::class)
+class SearchEngineReaderTest {
+ @Test
+ fun `SearchEngineReader can read from a file`() {
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ inputEncoding = "ISO-8859-1",
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search"),
+ )
+ val readSearchEngine = saveAndLoadSearchEngine(searchEngine)
+
+ assertEquals(searchEngine.id, readSearchEngine.id)
+ assertEquals(searchEngine.name, readSearchEngine.name)
+ assertEquals(searchEngine.inputEncoding, readSearchEngine.inputEncoding)
+ assertEquals(searchEngine.type, readSearchEngine.type)
+ assertEquals(searchEngine.resultUrls, readSearchEngine.resultUrls)
+ assertTrue(readSearchEngine.isGeneral)
+ }
+
+ @Test(expected = IOException::class)
+ fun `Parsing not existing file will throw exception`() {
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search"),
+ )
+ val reader = SearchEngineReader(type = SearchEngine.Type.CUSTOM)
+ val invalidFile = AtomicFile(File("", ""))
+ reader.loadFile(searchEngine.id, invalidFile)
+ }
+
+ @Test
+ fun `WHEN SearchEngineReader is loading bundled search engines from a file THEN the correct SearchEngine properties are parsed`() {
+ for (id in GENERAL_SEARCH_ENGINE_IDS + setOf("mozilla", "wikipedia")) {
+ val searchEngine = SearchEngine(
+ id = id,
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.BUNDLED,
+ resultUrls = listOf("https://www.example.com/search"),
+ )
+ val readSearchEngine = saveAndLoadSearchEngine(searchEngine)
+
+ assertEquals(searchEngine.id, readSearchEngine.id)
+ assertEquals(searchEngine.name, readSearchEngine.name)
+ assertEquals(searchEngine.type, readSearchEngine.type)
+ assertEquals(searchEngine.resultUrls, readSearchEngine.resultUrls)
+ assertEquals(id in GENERAL_SEARCH_ENGINE_IDS, readSearchEngine.isGeneral)
+ }
+ }
+ private fun saveAndLoadSearchEngine(searchEngine: SearchEngine): SearchEngine {
+ val storage = CustomSearchEngineStorage(testContext)
+ val writer = SearchEngineWriter()
+ val reader = SearchEngineReader(type = searchEngine.type)
+ val file = storage.getSearchFile(searchEngine.id)
+
+ writer.saveSearchEngineXML(searchEngine, file)
+
+ return reader.loadFile(searchEngine.id, file)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineWriterTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineWriterTest.kt
new file mode 100644
index 0000000000..487231f6f9
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/storage/SearchEngineWriterTest.kt
@@ -0,0 +1,212 @@
+/* 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/. */
+
+package mozilla.components.feature.search.storage
+
+import android.util.AtomicFile
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import mozilla.components.browser.state.search.SearchEngine
+import mozilla.components.support.test.any
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.robolectric.testContext
+import mozilla.components.support.test.whenever
+import org.junit.Assert.assertFalse
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertTrue
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.mockito.Mockito.spy
+import org.w3c.dom.DOMException
+import org.w3c.dom.DOMException.INVALID_CHARACTER_ERR
+import org.w3c.dom.Document
+import java.io.File
+import java.io.StringWriter
+import javax.xml.parsers.DocumentBuilderFactory
+import javax.xml.transform.OutputKeys
+import javax.xml.transform.TransformerConfigurationException
+import javax.xml.transform.TransformerException
+import javax.xml.transform.TransformerFactory
+import javax.xml.transform.dom.DOMSource
+import javax.xml.transform.stream.StreamResult
+
+@RunWith(AndroidJUnit4::class)
+class SearchEngineWriterTest {
+ @Test
+ fun `buildSearchEngineXML builds search engine xml correctly`() {
+ val writer = SearchEngineWriter()
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ inputEncoding = "UTF-8",
+ resultUrls = listOf("https://www.example.com/search?q={searchTerms}'"),
+ )
+
+ val document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument()
+ writer.buildSearchEngineXML(searchEngine, document)
+ val searchXML = document.xmlToString()
+ assertTrue(searchXML!!.contains(searchEngine.name))
+ assertTrue(searchXML.contains(IMAGE_URI_PREFIX))
+ assertTrue(searchXML.contains(URL_TYPE_SEARCH_HTML))
+ assertTrue(searchXML.contains("https://www.example.com/search?q={searchTerms}"))
+ assertTrue(searchXML.contains("UTF-8"))
+ assertFalse(searchXML.contains(URL_TYPE_SUGGEST_JSON))
+ }
+
+ @Test
+ fun `buildSearchEngineXML builds multiple resultUrls correctly`() {
+ val writer = SearchEngineWriter()
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf(
+ "https://www.example.com/search?q=%s",
+ "https://www.example.com/search1?q=%s",
+ "https://www.example.com/search2?q=%s",
+ ),
+ )
+
+ val document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument()
+ writer.buildSearchEngineXML(searchEngine, document)
+ val searchXML = document.xmlToString()
+ assertTrue(searchXML!!.contains(searchEngine.name))
+ assertTrue(searchXML.contains(IMAGE_URI_PREFIX))
+ assertTrue(searchXML.contains(URL_TYPE_SEARCH_HTML))
+ searchEngine.resultUrls.forEach {
+ val url = it.replace("%s", "{searchTerms}")
+ searchXML.contains(url)
+ }
+ assertFalse(searchXML.contains(URL_TYPE_SUGGEST_JSON))
+ }
+
+ @Test
+ fun `buildSearchEngineXML builds suggestUrl correctly`() {
+ val writer = SearchEngineWriter()
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ suggestUrl = "https://www.example.com/search-suggestion?q=%s",
+ )
+
+ val document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument()
+ writer.buildSearchEngineXML(searchEngine, document)
+ val searchXML = document.xmlToString()
+ assertTrue(searchXML!!.contains(searchEngine.name))
+ assertTrue(searchXML.contains(IMAGE_URI_PREFIX))
+ assertFalse(searchXML.contains(URL_TYPE_SEARCH_HTML))
+ assertTrue(searchXML.contains(URL_TYPE_SUGGEST_JSON))
+ assertTrue(searchXML.contains("https://www.example.com/search-suggestion?q={searchTerms}"))
+ }
+
+ @Test
+ fun `buildSearchEngineXML successfully for search engines with XML escaped characters`() {
+ val writer = SearchEngineWriter()
+ val invalidSearchEngineNameAmp = SearchEngine(
+ id = "id1",
+ name = "&&&example&&&",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ suggestUrl = "https://www.example.com/search-suggestion?q=%s",
+ )
+ val invalidResultUrlLessSign = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search?<q=%s"),
+ )
+ val invalidResultUrlGreaterSign = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search?>q=%s"),
+ )
+ val invalidSuggestionUrlApo = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ suggestUrl = "https://www.example.com/search-'suggestion'?q=%s",
+ )
+
+ assertNotNull(
+ writer.buildSearchEngineXML(
+ invalidSearchEngineNameAmp,
+ DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument(),
+ ),
+ )
+ assertNotNull(
+ writer.buildSearchEngineXML(
+ invalidResultUrlLessSign,
+ DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument(),
+ ),
+ )
+ assertNotNull(
+ writer.buildSearchEngineXML(
+ invalidResultUrlGreaterSign,
+ DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument(),
+ ),
+ )
+ assertNotNull(
+ writer.buildSearchEngineXML(
+ invalidSuggestionUrlApo,
+ DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument(),
+ ),
+ )
+ }
+
+ @Test
+ fun `saveSearchEngineXML returns false when failed to write to a bad file data`() {
+ val writer = spy(SearchEngineWriter())
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search?q=%s'"),
+ )
+
+ val badFile = AtomicFile(File("", ""))
+ assertFalse(writer.saveSearchEngineXML(searchEngine, badFile))
+ }
+
+ @Test
+ fun `saveSearchEngineXML returns false when there's DOMException while generating XML doc`() {
+ val storage = CustomSearchEngineStorage(testContext)
+ val writer = spy(SearchEngineWriter())
+ val searchEngine = SearchEngine(
+ id = "id1",
+ name = "example",
+ icon = mock(),
+ type = SearchEngine.Type.CUSTOM,
+ resultUrls = listOf("https://www.example.com/search?q=%s'"),
+ )
+
+ val file = storage.getSearchFile(searchEngine.id)
+ val mockDoc: Document = mock()
+ whenever(mockDoc.createElement(any())).thenThrow(DOMException(INVALID_CHARACTER_ERR, ""))
+ assertFalse(writer.saveSearchEngineXML(searchEngine, file, mockDoc))
+ }
+}
+
+private fun Document.xmlToString(): String? {
+ val writer = StringWriter()
+ try {
+ val tf = TransformerFactory.newInstance().newTransformer()
+ tf.setOutputProperty(OutputKeys.ENCODING, "UTF-8")
+ tf.transform(DOMSource(this), StreamResult(writer))
+ } catch (e: TransformerConfigurationException) {
+ return null
+ } catch (e: TransformerException) {
+ return null
+ }
+
+ return writer.toString()
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/ParserTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/ParserTest.kt
new file mode 100644
index 0000000000..6d69abd67b
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/ParserTest.kt
@@ -0,0 +1,149 @@
+/* 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/. */
+
+package mozilla.components.feature.search.suggestions
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import org.junit.Assert.assertEquals
+import org.junit.Test
+import org.junit.runner.RunWith
+
+@RunWith(AndroidJUnit4::class)
+class ParserTest {
+
+ @Test
+ fun `can parse a response from Google`() {
+ val json = "[\"firefox\",[\"firefox\",\"firefox for mac\",\"firefox quantum\",\"firefox update\",\"firefox esr\",\"firefox focus\",\"firefox addons\",\"firefox extensions\",\"firefox nightly\",\"firefox clear cache\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox for mac", "firefox quantum", "firefox update", "firefox esr", "firefox focus", "firefox addons", "firefox extensions", "firefox nightly", "firefox clear cache")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Amazon`() {
+ val json = "[\"firefox\",[\"firefox for fire tv\",\"firefox movie\",\"firefox app\",\"firefox books\",\"firefox glider\",\"firefox stick\",\"firefox for firestick\",\"firefox\",\"firefox books series\",\"firefox browser\"],[{\"nodes\":[{\"name\":\"Apps & Games\",\"alias\":\"mobile-apps\"}]},{},{},{},{},{},{},{},{},{}],[],\"344I6KZL0KU9N\"]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox for fire tv", "firefox movie", "firefox app", "firefox books", "firefox glider", "firefox stick", "firefox for firestick", "firefox", "firefox books series", "firefox browser")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Azerdict`() {
+ val json = "{\"query\":\"code\",\"suggestions\":[\"code\",\"codec\",\"codex\",\"codeine\",\"code of laws\"]}"
+
+ val results = azerdictResponseParser(json)
+ val expectedResults = listOf("code", "codec", "codex", "codeine", "code of laws")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Baidu`() {
+ val json = "[\"firefox\",[\"firefox rocket\",\"firefox手机浏览器\",\"firefox是什么意思\",\"firefox focus\",\"firefox风哥\",\"firefox官网\",\"firefox吧\",\"firefox os\",\"firefox国际版\",\"firefox android\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox rocket", "firefox手机浏览器", "firefox是什么意思", "firefox focus", "firefox风哥", "firefox官网", "firefox吧", "firefox os", "firefox国际版", "firefox android")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Bing`() {
+ val json = "[\"firefox\",[\"firefox\",\"firefox download\",\"firefox for windows 10\",\"firefox browser\",\"firefox quantum\",\"firefox esr\",\"firefox 64-bit\",\"firefox mozilla\",\"firefox nightly\",\"firefox update\",\"firefox install\",\"firefox focus\",\"firefox beta\",\"firefox developer edition\",\"firefox portable\",\"firefox add-ons\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox download", "firefox for windows 10", "firefox browser", "firefox quantum", "firefox esr", "firefox 64-bit", "firefox mozilla", "firefox nightly", "firefox update", "firefox install", "firefox focus", "firefox beta", "firefox developer edition", "firefox portable", "firefox add-ons")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Coccoc`() {
+ val json = "[\"firefox\",[\"firefox\",\"firefox tiếng việt\",\"firefox download\",\"firefox quantum\",\"firefox 51\",\"firefox 42\",\"firefox english\",\"firefox portable\",\"firefox 49\",\"firefox viet nam\"],[\"\",\"\",\"\",\"\",\"\",\"\",\"\",\"\",\"\",\"\"],[\"\",\"\",\"\",\"\",\"\",\"\",\"\",\"\",\"\",\"\"],{\"google:suggesttype\":[\"QUERY\",\"QUERY\",\"QUERY\",\"QUERY\",\"QUERY\",\"QUERY\",\"QUERY\",\"QUERY\",\"QUERY\",\"QUERY\"]}]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox tiếng việt", "firefox download", "firefox quantum", "firefox 51", "firefox 42", "firefox english", "firefox portable", "firefox 49", "firefox viet nam")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Daum`() {
+ val json = "{\"q\":\"firefox\",\"rq\":\"firefox\",\"items\":[\"firefox\",\"firefox download\",\"firefox focus\",\"mozilla firefox\",\"firefox adobe flash\",\"firefox 삭제\",\"firefox 한글판\",\"firefox 즐겨찾기 가져오기\",\"firefox quantum\",\"firefox 57\",\"mozila firefox\",\"mozilla firefox download\",\"mozilla firefox 삭제\",\"Hacking Firefox\",\"Programming Firefox\"],\"r_items\":[]}"
+
+ val results = daumResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox download", "firefox focus", "mozilla firefox", "firefox adobe flash", "firefox 삭제", "firefox 한글판", "firefox 즐겨찾기 가져오기", "firefox quantum", "firefox 57", "mozila firefox", "mozilla firefox download", "mozilla firefox 삭제", "Hacking Firefox", "Programming Firefox")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Duck Duck Go`() {
+ val json = "[\"firefox\",[\"firefox\",\"firefox browser\",\"firefox.com\",\"firefox update\",\"firefox for mac\",\"firefox quantum\",\"firefox extensions\",\"firefox esr\",\"firefox clear cache\",\"firefox themes\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox browser", "firefox.com", "firefox update", "firefox for mac", "firefox quantum", "firefox extensions", "firefox esr", "firefox clear cache", "firefox themes")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Naver`() {
+ val json = "[\"firefox\",[\"firefox\",\"Mozilla Firefox\",\"firefox add-on to detect vulnerable websites\",\"firefox ak\",\"firefox as gaeilge\",\"firefox developer edition\",\"firefox down\",\"firefox for dummies\",\"firefox for mac\",\"firefox for mobile\"],[],[\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=Mozilla+Firefox\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+add-on+to+detect+vulnerable+websites\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+ak\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+as+gaeilge\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+developer+edition\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+down\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+for+dummies\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+for+mac\",\"http://search.naver.com/search.naver?where=nexearch&sm=osp_sug&ie=utf8&query=firefox+for+mobile\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "Mozilla Firefox", "firefox add-on to detect vulnerable websites", "firefox ak", "firefox as gaeilge", "firefox developer edition", "firefox down", "firefox for dummies", "firefox for mac", "firefox for mobile")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Prisjakt`() {
+ val json = "[\"firefox\",[\"Firefox\",\"Firefox\",\"Mountain Equipment Firefox Pants (Herr)\",\"Mountain Equipment Firefox (Herr)\",\"Firefox (DE)\"],[\"L\\u00e4gsta pris: 149:-\",\"L\\u00e4gsta pris: 205:-\",\"L\\u00e4gsta pris: 1 559:-\",\"L\\u00e4gsta pris: 2 773:-\",\"L\\u00e4gsta pris: 198:-\"],[\"https:\\/\\/www.prisjakt.nu\\/produkt.php?p=886794\",\"https:\\/\\/www.prisjakt.nu\\/produkt.php?p=53272\",\"https:\\/\\/www.prisjakt.nu\\/produkt.php?p=3548472\",\"https:\\/\\/www.prisjakt.nu\\/produkt.php?p=1822581\",\"https:\\/\\/www.prisjakt.nu\\/produkt.php?p=3408551\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("Firefox", "Firefox", "Mountain Equipment Firefox Pants (Herr)", "Mountain Equipment Firefox (Herr)", "Firefox (DE)")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Qwant`() {
+ val json = "{\"status\":\"success\",\"data\":{\"items\":[{\"value\":\"firefox (video game)\",\"suggestType\":3},{\"value\":\"firefox addons\",\"suggestType\":12},{\"value\":\"firefox\",\"suggestType\":2},{\"value\":\"firefox quantum\",\"suggestType\":12},{\"value\":\"firefox focus\",\"suggestType\":12}],\"special\":[],\"availableQwick\":[]}}"
+
+ val results = qwantResponseParser(json)
+ val expectedResults = listOf("firefox (video game)", "firefox addons", "firefox", "firefox quantum", "firefox focus")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Seznam`() {
+ val json = "[\"firefox\", [\"firefox\", \"firefox ak keep it up\", \"firefox ak city to city\", \"firefox ak all those people\", \"firefox ak who can act\", \"firefox ak color the trees\", \"firefox ak habibi\", \"firefox ak zodiac\", \"firefox ak the draft\", \"mozilla firefox\"], [], []]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox ak keep it up", "firefox ak city to city", "firefox ak all those people", "firefox ak who can act", "firefox ak color the trees", "firefox ak habibi", "firefox ak zodiac", "firefox ak the draft", "mozilla firefox")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Wikipedia`() {
+ val json = "[\"code\",[\"Code\",\"Code Geass\",\"Codeine\",\"Codename: Kids Next Door\",\"Code page\",\"Codex Sinaiticus\",\"Code talker\",\"Code Black (TV series)\",\"Codependency\",\"Codex Vaticanus\"],[\"In communications and information processing, code is a system of rules to convert information\\u2014such as a letter, word, sound, image, or gesture\\u2014into another form or representation, sometimes shortened or secret, for communication through a communication channel or storage in a storage medium.\",\"Code Geass: Lelouch of the Rebellion (\\u30b3\\u30fc\\u30c9\\u30ae\\u30a2\\u30b9 \\u53cd\\u9006\\u306e\\u30eb\\u30eb\\u30fc\\u30b7\\u30e5, K\\u014ddo Giasu: Hangyaku no Rur\\u016bshu), often referred to as simply Code Geass, is a Japanese anime series created by Sunrise, directed by Gor\\u014d Taniguchi, and written by Ichir\\u014d \\u014ckouchi, with original character designs by manga authors Clamp.\",\"Codeine is an opiate used to treat pain, as a cough medicine, and for diarrhea. It is typically used to treat mild to moderate degrees of pain.\",\"Codename: Kids Next Door, commonly abbreviated to Kids Next Door or KND, is an American animated television series created by Tom Warburton for Cartoon Network, and the 13th of the network's Cartoon Cartoons.\",\"In computing, a code page is a table of values that describes the character set used for encoding a particular set of characters, usually combined with a number of control characters.\",\"Codex Sinaiticus (Greek: \\u03a3\\u03b9\\u03bd\\u03b1\\u03ca\\u03c4\\u03b9\\u03ba\\u03cc\\u03c2 \\u039a\\u03ce\\u03b4\\u03b9\\u03ba\\u03b1\\u03c2, Hebrew: \\u05e7\\u05d5\\u05d3\\u05e7\\u05e1 \\u05e1\\u05d9\\u05e0\\u05d0\\u05d9\\u05d8\\u05d9\\u05e7\\u05d5\\u05e1\\u200e; Shelfmarks and references: London, Brit.\",\"Code talkers are people in the 20th century who used obscure languages as a means of secret communication during wartime.\",\"Code Black is an American medical drama television series created by Michael Seitzman which premiered on CBS on September 30, 2015. It takes place in an overcrowded and understaffed emergency room in Los Angeles, California, and is based on a documentary by Ryan McGarry.\",\"Codependency is a controversial concept for a dysfunctional helping relationship where one person supports or enables another person's addiction, poor mental health, immaturity, irresponsibility, or under-achievement.\",\"The Codex Vaticanus (The Vatican, Bibl. Vat., Vat. gr. 1209; no. B or 03 Gregory-Aland, \\u03b4 1 von Soden) is regarded as the oldest extant manuscript of the Greek Bible (Old and New Testament), one of the four great uncial codices.\"],[\"https://en.wikipedia.org/wiki/Code\",\"https://en.wikipedia.org/wiki/Code_Geass\",\"https://en.wikipedia.org/wiki/Codeine\",\"https://en.wikipedia.org/wiki/Codename:_Kids_Next_Door\",\"https://en.wikipedia.org/wiki/Code_page\",\"https://en.wikipedia.org/wiki/Codex_Sinaiticus\",\"https://en.wikipedia.org/wiki/Code_talker\",\"https://en.wikipedia.org/wiki/Code_Black_(TV_series)\",\"https://en.wikipedia.org/wiki/Codependency\",\"https://en.wikipedia.org/wiki/Codex_Vaticanus\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("Code", "Code Geass", "Codeine", "Codename: Kids Next Door", "Code page", "Codex Sinaiticus", "Code talker", "Code Black (TV series)", "Codependency", "Codex Vaticanus")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Yahoo`() {
+ val json = "[\"firefox\",[\"firefox\",\"firefox browser\",\"firefox.com\",\"firefox update\"],[],[]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox browser", "firefox.com", "firefox update")
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `can parse a response from Yandex`() {
+ val json = "[\"firefox\",[\"firefox\",\"firefox download\",\"firefox browser\",\"firefox update\",\"firefox.com\"]]"
+
+ val results = defaultResponseParser(json)
+ val expectedResults = listOf("firefox", "firefox download", "firefox browser", "firefox update", "firefox.com")
+ assertEquals(expectedResults, results)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/SearchSuggestionClientTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/SearchSuggestionClientTest.kt
new file mode 100644
index 0000000000..3ffd14698a
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/suggestions/SearchSuggestionClientTest.kt
@@ -0,0 +1,110 @@
+/* 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/. */
+
+package mozilla.components.feature.search.suggestions
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import kotlinx.coroutines.test.runTest
+import mozilla.components.browser.state.state.BrowserState
+import mozilla.components.browser.state.state.SearchState
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.feature.search.ext.createSearchEngine
+import mozilla.components.support.test.mock
+import mozilla.components.support.test.robolectric.testContext
+import org.junit.Assert.assertEquals
+import org.junit.Test
+import org.junit.runner.RunWith
+import java.io.IOException
+
+@RunWith(AndroidJUnit4::class)
+class SearchSuggestionClientTest {
+ companion object {
+ val GOOGLE_MOCK_RESPONSE: SearchSuggestionFetcher = { "[\"firefox\",[\"firefox\",\"firefox for mac\",\"firefox quantum\",\"firefox update\",\"firefox esr\",\"firefox focus\",\"firefox addons\",\"firefox extensions\",\"firefox nightly\",\"firefox clear cache\"]]" }
+ val QWANT_MOCK_RESPONSE: SearchSuggestionFetcher = { "{\"status\":\"success\",\"data\":{\"items\":[{\"value\":\"firefox (video game)\",\"suggestType\":3},{\"value\":\"firefox addons\",\"suggestType\":12},{\"value\":\"firefox\",\"suggestType\":2},{\"value\":\"firefox quantum\",\"suggestType\":12},{\"value\":\"firefox focus\",\"suggestType\":12}],\"special\":[],\"availableQwick\":[]}}" }
+ val SERVER_ERROR_RESPONSE: SearchSuggestionFetcher = { "Server error. Try again later" }
+ }
+
+ private val searchEngine = createSearchEngine(
+ name = "Test",
+ url = "https://localhost?q={searchTerms}",
+ suggestUrl = "https://localhost/suggestions?q={searchTerms}",
+ icon = mock(),
+ )
+
+ @Test
+ fun `Get a list of results based on the Google search engine`() = runTest {
+ val client = SearchSuggestionClient(searchEngine, GOOGLE_MOCK_RESPONSE)
+ val expectedResults = listOf("firefox", "firefox for mac", "firefox quantum", "firefox update", "firefox esr", "firefox focus", "firefox addons", "firefox extensions", "firefox nightly", "firefox clear cache")
+
+ val results = client.getSuggestions("firefox")
+
+ assertEquals(expectedResults, results)
+ }
+
+ @Test
+ fun `Get a list of results based on a non google search engine`() = runTest {
+ val qwant = createSearchEngine(
+ name = "Qwant",
+ url = "https://localhost?q={searchTerms}",
+ suggestUrl = "https://localhost/suggestions?q={searchTerms}",
+ icon = mock(),
+ )
+ val client = SearchSuggestionClient(qwant, QWANT_MOCK_RESPONSE)
+ val expectedResults = listOf("firefox (video game)", "firefox addons", "firefox", "firefox quantum", "firefox focus")
+
+ val results = client.getSuggestions("firefox")
+
+ assertEquals(expectedResults, results)
+ }
+
+ @Test(expected = SearchSuggestionClient.ResponseParserException::class)
+ fun `Check that a bad response will throw a parser exception`() = runTest {
+ val client = SearchSuggestionClient(searchEngine, SERVER_ERROR_RESPONSE)
+
+ client.getSuggestions("firefox")
+ }
+
+ @Test(expected = SearchSuggestionClient.FetchException::class)
+ fun `Check that an exception in the suggestionFetcher will re-throw an IOException`() = runTest {
+ val client = SearchSuggestionClient(searchEngine) { throw IOException() }
+
+ client.getSuggestions("firefox")
+ }
+
+ @Test
+ fun `Check that a search engine without a suggestURI will return an empty suggestion list`() = runTest {
+ val searchEngine = createSearchEngine(
+ name = "Test",
+ url = "https://localhost?q={searchTerms}",
+ icon = mock(),
+ )
+ val client = SearchSuggestionClient(searchEngine) { "no-op" }
+
+ val results = client.getSuggestions("firefox")
+
+ assertEquals(emptyList<String>(), results)
+ }
+
+ @Test
+ fun `Default search engine is used if search engine manager provided`() = runTest {
+ val store = BrowserStore(
+ BrowserState(
+ search = SearchState(
+ regionSearchEngines = listOf(searchEngine),
+ ),
+ ),
+ )
+
+ val client = SearchSuggestionClient(
+ testContext,
+ store,
+ GOOGLE_MOCK_RESPONSE,
+ )
+ val expectedResults = listOf("firefox", "firefox for mac", "firefox quantum", "firefox update", "firefox esr", "firefox focus", "firefox addons", "firefox extensions", "firefox nightly", "firefox clear cache")
+
+ val results = client.getSuggestions("firefox")
+
+ assertEquals(expectedResults, results)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetryTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetryTest.kt
new file mode 100644
index 0000000000..90024bbb56
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/BaseSearchTelemetryTest.kt
@@ -0,0 +1,187 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import kotlinx.coroutines.runBlocking
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.Engine
+import mozilla.components.support.test.any
+import mozilla.components.support.test.eq
+import mozilla.components.support.test.mock
+import org.json.JSONObject
+import org.junit.Assert.assertEquals
+import org.junit.Before
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.mockito.Mock
+import org.mockito.Mockito.doAnswer
+import org.mockito.Mockito.doReturn
+import org.mockito.Mockito.spy
+import org.mockito.Mockito.verify
+import org.mockito.Mockito.`when`
+import java.io.File
+
+@RunWith(AndroidJUnit4::class)
+class BaseSearchTelemetryTest {
+
+ private lateinit var baseTelemetry: BaseSearchTelemetry
+ private lateinit var handler: BaseSearchTelemetry.SearchTelemetryMessageHandler
+
+ @Mock
+ private lateinit var mockRepo: SerpTelemetryRepository
+
+ private val mockReadJson: () -> JSONObject = mock()
+ private val mockRootStorageDirectory: File = mock()
+
+ private fun createMockProviderList(): List<SearchProviderModel> = listOf(
+ SearchProviderModel(
+ schema = 1698656464939,
+ taggedCodes = listOf("monline_7_dg"),
+ telemetryId = "baidu",
+ organicCodes = emptyList(),
+ codeParamName = "tn",
+ queryParamNames = listOf("wd"),
+ searchPageRegexp = "^https://(?:m|www)\\\\.baidu\\\\.com/(?:s|baidu)",
+ followOnParamNames = listOf("oq"),
+ extraAdServersRegexps = listOf("^https?://www\\\\.baidu\\\\.com/baidu\\\\.php?"),
+ expectedOrganicCodes = emptyList(),
+ ),
+ )
+
+ private val rawJson = """
+ {
+ "data": [
+ {
+ "schema": 1698656464939,
+ "taggedCodes": [
+ "monline_7_dg"
+ ],
+ "telemetryId": "baidu",
+ "organicCodes": [],
+ "codeParamName": "tn",
+ "queryParamNames": [
+ "wd"
+ ],
+ "searchPageRegexp": "^https://(?:m|www)\\.baidu\\.com/(?:s|baidu)",
+ "followOnParamNames": [
+ "oq"
+ ],
+ "extraAdServersRegexps": [
+ "^https?://www\\.baidu\\.com/baidu\\.php?"
+ ],
+ "id": "19c434a3-d173-4871-9743-290ac92a3f6a",
+ "last_modified": 1698666532326
+ }],
+ "timestamp": 16
+}
+ """.trimIndent()
+
+ @Before
+ fun setup() {
+ baseTelemetry = spy(
+ object : BaseSearchTelemetry() {
+ override suspend fun install(
+ engine: Engine,
+ store: BrowserStore,
+ providerList: List<SearchProviderModel>,
+ ) {
+ // mock, do nothing
+ }
+
+ override fun processMessage(message: JSONObject) {
+ // mock, do nothing
+ }
+ },
+ )
+ handler = baseTelemetry.SearchTelemetryMessageHandler()
+ mockRepo = spy(SerpTelemetryRepository(mockRootStorageDirectory, mockReadJson, "test"))
+ }
+
+ @Test
+ fun `GIVEN an engine WHEN installWebExtension is called THEN the provided extension is installed in engine`() {
+ val engine: Engine = mock()
+ val store: BrowserStore = mock()
+ val id = "id"
+ val resourceUrl = "resourceUrl"
+ val messageId = "messageId"
+ val extensionInfo = ExtensionInfo(id, resourceUrl, messageId)
+
+ baseTelemetry.installWebExtension(engine, store, extensionInfo)
+
+ verify(engine).installBuiltInWebExtension(
+ id = eq(id),
+ url = eq(resourceUrl),
+ onSuccess = any(),
+ onError = any(),
+ )
+ }
+
+ @Test
+ fun `GIVEN a search provider does not exist for the url WHEN getProviderForUrl is called THEN return null`() {
+ val url = "https://www.mozilla.com/search?q=firefox"
+ baseTelemetry.providerList = createMockProviderList()
+
+ assertEquals(null, baseTelemetry.getProviderForUrl(url))
+ }
+
+ @Test(expected = IllegalStateException::class)
+ fun `GIVEN an extension message WHEN that cannot be processed THEN throw IllegalStateException`() {
+ val message = "message"
+
+ handler.onMessage(message, mock())
+ }
+
+ @Test
+ fun `GIVEN an extension message WHEN received THEN pass it to processMessage`() {
+ val message = JSONObject()
+
+ handler.onMessage(message, mock())
+
+ verify(baseTelemetry).processMessage(message)
+ }
+
+ @Test
+ fun `GIVEN empty cacheResponse WHEN initializeProviderList is called THEN update providerList`(): Unit =
+ runBlocking {
+ val localResponse = JSONObject(rawJson)
+ val cacheResponse: Pair<ULong, List<SearchProviderModel>> = Pair(0u, emptyList())
+
+ `when`(mockRepo.loadProvidersFromCache()).thenReturn(cacheResponse)
+ doAnswer {
+ localResponse
+ }.`when`(mockReadJson)()
+
+ `when`(mockRepo.parseLocalPreinstalledData(localResponse)).thenReturn(createMockProviderList())
+ doReturn(Unit).`when`(mockRepo).fetchRemoteResponse(any())
+
+ baseTelemetry.setProviderList(mockRepo.updateProviderList())
+
+ assertEquals(baseTelemetry.providerList.toString(), createMockProviderList().toString())
+ }
+
+ @Test
+ fun `GIVEN non-empty cacheResponse WHEN initializeProviderList is called THEN update providerList`(): Unit =
+ runBlocking {
+ val localResponse = JSONObject(rawJson)
+ val cacheResponse: Pair<ULong, List<SearchProviderModel>> = Pair(123u, createMockProviderList())
+
+ `when`(mockRepo.loadProvidersFromCache()).thenReturn(cacheResponse)
+ doAnswer {
+ localResponse
+ }.`when`(mockReadJson)()
+ doReturn(Unit).`when`(mockRepo).fetchRemoteResponse(any())
+
+ baseTelemetry.setProviderList(mockRepo.updateProviderList())
+
+ assertEquals(baseTelemetry.providerList.toString(), createMockProviderList().toString())
+ }
+
+ fun getProviderForUrl(url: String): SearchProviderModel? {
+ return createMockProviderList().find { provider ->
+ provider.searchPageRegexp.pattern.toRegex().containsMatchIn(url)
+ }
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SearchProviderModelTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SearchProviderModelTest.kt
new file mode 100644
index 0000000000..89e4620101
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SearchProviderModelTest.kt
@@ -0,0 +1,40 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+import org.junit.Assert
+import org.junit.Test
+
+class SearchProviderModelTest {
+ private val testSearchProvider =
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("mzl", "813cf1dd", "16eeffc4"),
+ telemetryId = "test",
+ organicCodes = listOf(),
+ codeParamName = "tt",
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https://www\\.ecosia\\.org/",
+ expectedOrganicCodes = listOf(),
+ extraAdServersRegexps = listOf(
+ "^https:\\/\\/www\\.bing\\.com\\/acli?c?k",
+ "^https:\\/\\/www\\.bing\\.com\\/fd\\/ls\\/GLinkPingPost\\.aspx.*acli?c?k",
+ ),
+ )
+
+ @Test
+ fun `test search provider contains ads`() {
+ val ad = "https://www.bing.com/aclick"
+ val nonAd = "https://www.bing.com/notanad"
+ Assert.assertTrue(testSearchProvider.containsAdLinks(listOf(ad, nonAd)))
+ }
+
+ @Test
+ fun `test search provider does not contain ads`() {
+ val nonAd1 = "https://www.yahoo.com/notanad"
+ val nonAd2 = "https://www.google.com/"
+ Assert.assertFalse(testSearchProvider.containsAdLinks(listOf(nonAd1, nonAd2)))
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepositoryTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepositoryTest.kt
new file mode 100644
index 0000000000..50ae3c6aee
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/SerpTelemetryRepositoryTest.kt
@@ -0,0 +1,92 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry
+
+import kotlinx.coroutines.runBlocking
+import mozilla.appservices.remotesettings.RemoteSettingsRecord
+import mozilla.appservices.remotesettings.RemoteSettingsResponse
+import mozilla.components.support.remotesettings.RemoteSettingsClient
+import mozilla.components.support.remotesettings.RemoteSettingsResult
+import mozilla.components.support.test.mock
+import org.json.JSONObject
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertTrue
+import org.junit.Before
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.mockito.Mock
+import org.mockito.Mockito.`when`
+import org.mockito.MockitoAnnotations
+import org.robolectric.RobolectricTestRunner
+
+@RunWith(RobolectricTestRunner::class)
+class SerpTelemetryRepositoryTest {
+ @Mock
+ private lateinit var mockRemoteSettingsClient: RemoteSettingsClient
+
+ private lateinit var serpTelemetryRepository: SerpTelemetryRepository
+
+ @Before
+ fun setup() {
+ MockitoAnnotations.openMocks(this)
+ serpTelemetryRepository = SerpTelemetryRepository(
+ rootStorageDirectory = mock(),
+ readJson = mock(),
+ collectionName = "",
+ serverUrl = "https://test.server",
+ bucketName = "",
+ )
+
+ serpTelemetryRepository.remoteSettingsClient = mockRemoteSettingsClient
+ }
+
+ @Test
+ fun `GIVEN non-empty response WHEN writeToCache is called THEN the result is a success`() = runBlocking {
+ val records = listOf(
+ RemoteSettingsRecord("1", 123u, false, null, JSONObject()),
+ RemoteSettingsRecord("2", 456u, true, null, JSONObject()),
+ )
+ val response = RemoteSettingsResponse(records, 125614567U)
+
+ `when`(mockRemoteSettingsClient.write(response))
+ .thenReturn(RemoteSettingsResult.Success(response))
+
+ val result = serpTelemetryRepository.writeToCache(response)
+
+ assertTrue(result is RemoteSettingsResult.Success)
+ assertEquals(response, (result as RemoteSettingsResult.Success).response)
+ }
+
+ @Test
+ fun `GIVEN non-empty response WHEN fetchRemoteResponse is called THEN the result is equal to the response`() = runBlocking {
+ val records = listOf(
+ RemoteSettingsRecord("1", 123u, false, null, JSONObject()),
+ RemoteSettingsRecord("2", 456u, true, null, JSONObject()),
+ )
+ val response = RemoteSettingsResponse(records, 125614567U)
+ `when`(mockRemoteSettingsClient.fetch())
+ .thenReturn(RemoteSettingsResult.Success(response))
+
+ val result = serpTelemetryRepository.fetchRemoteResponse()
+
+ assertEquals(response, result)
+ }
+
+ @Test
+ fun `GIVEN non-empty response WHEN loadProvidersFromCache is called THEN the result is equal to the response`() = runBlocking {
+ val records = listOf(
+ RemoteSettingsRecord("1", 123u, false, null, JSONObject()),
+ RemoteSettingsRecord("2", 456u, true, null, JSONObject()),
+ )
+ val response = RemoteSettingsResponse(records, 125614567U)
+ `when`(mockRemoteSettingsClient.read())
+ .thenReturn(RemoteSettingsResult.Success(response))
+
+ val result = serpTelemetryRepository.loadProvidersFromCache()
+
+ assertEquals(response.lastModified, result.first)
+ assertEquals(response.records.mapNotNull { it.fields.toSearchProviderModel() }, result.second)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetryTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetryTest.kt
new file mode 100644
index 0000000000..d804ad98fa
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/ads/AdsTelemetryTest.kt
@@ -0,0 +1,271 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry.ads
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import kotlinx.coroutines.runBlocking
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.Engine
+import mozilla.components.feature.search.telemetry.ExtensionInfo
+import mozilla.components.feature.search.telemetry.SearchProviderCookie
+import mozilla.components.feature.search.telemetry.SearchProviderModel
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry.Companion.ADS_EXTENSION_ID
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry.Companion.ADS_EXTENSION_RESOURCE_URL
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry.Companion.ADS_MESSAGE_COOKIES_KEY
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry.Companion.ADS_MESSAGE_DOCUMENT_URLS_KEY
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry.Companion.ADS_MESSAGE_ID
+import mozilla.components.feature.search.telemetry.ads.AdsTelemetry.Companion.ADS_MESSAGE_SESSION_URL_KEY
+import mozilla.components.support.base.Component
+import mozilla.components.support.base.facts.Action
+import mozilla.components.support.base.facts.Fact
+import mozilla.components.support.base.facts.FactProcessor
+import mozilla.components.support.base.facts.Facts
+import mozilla.components.support.test.any
+import mozilla.components.support.test.argumentCaptor
+import mozilla.components.support.test.eq
+import mozilla.components.support.test.mock
+import org.json.JSONArray
+import org.json.JSONObject
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertTrue
+import org.junit.Before
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.mockito.Mockito.doNothing
+import org.mockito.Mockito.spy
+import org.mockito.Mockito.verify
+
+@RunWith(AndroidJUnit4::class)
+class AdsTelemetryTest {
+ private lateinit var telemetry: AdsTelemetry
+
+ fun createMockProviderList(): List<SearchProviderModel> = listOf(
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("monline_7_dg", "monline_4_dg", "monline_3_dg", "monline_dg"),
+ telemetryId = "baidu",
+ organicCodes = emptyList(),
+ codeParamName = "tn",
+ followOnParamNames = listOf("oq"),
+ queryParamNames = listOf("wd", "word"),
+ searchPageRegexp = "^https://(?:m|www)\\.baidu\\.com/(?:s|baidu)",
+ extraAdServersRegexps = listOf("^https?://www\\.baidu\\.com/baidu\\.php?"),
+ expectedOrganicCodes = emptyList(),
+ ),
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("firefox-b-m", "fpas", "lm"),
+ telemetryId = "duckduckgo",
+ organicCodes = emptyList(),
+ codeParamName = "t",
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https:\\/\\/duckduckgo\\.com\\/",
+ extraAdServersRegexps = listOf("^https://duckduckgo.com/y\\.js?.*ad_provider\\="),
+ expectedOrganicCodes = emptyList(),
+ ),
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("firefox-b-m", "fpas", "def"),
+ telemetryId = "google",
+ organicCodes = emptyList(),
+ codeParamName = "client",
+ followOnParamNames = listOf("oq", "ved", "ei"),
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https://www\\.google\\.(?:.+)/search",
+ extraAdServersRegexps = listOf("^https?://www\\.google(?:adservices)?\\.com/(?:pagead/)?aclk"),
+ expectedOrganicCodes = emptyList(),
+ ),
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("MOZMBA", "MOZL", "def"),
+ telemetryId = "bing",
+ organicCodes = emptyList(),
+ codeParamName = "pc",
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https://www\\.bing\\.com/search",
+ extraAdServersRegexps = listOf("^https://www\\.bing\\.com/acli?c?k"),
+ followOnCookies = listOf(
+ SearchProviderCookie(
+ extraCodeParamName = "form",
+ extraCodePrefixes = listOf("QBRE"),
+ host = "www.bing.com",
+ name = "SRCHS",
+ codeParamName = "PC",
+ ),
+ ),
+ expectedOrganicCodes = emptyList(),
+ ),
+ )
+
+ @Before
+ fun setUp() {
+ telemetry = spy(AdsTelemetry())
+ }
+
+ @Test
+ fun `WHEN installWebExtension is called THEN install a properly configured extension`() {
+ val engine: Engine = mock()
+ val store: BrowserStore = mock()
+ val extensionCaptor = argumentCaptor<ExtensionInfo>()
+
+ runBlocking {
+ doNothing().`when`(telemetry).setProviderList(any())
+ telemetry.install(engine, store, mock())
+ }
+
+ verify(telemetry).installWebExtension(eq(engine), eq(store), extensionCaptor.capture())
+ assertEquals(ADS_EXTENSION_ID, extensionCaptor.value.id)
+ assertEquals(ADS_EXTENSION_RESOURCE_URL, extensionCaptor.value.resourceUrl)
+ assertEquals(ADS_MESSAGE_ID, extensionCaptor.value.messageId)
+ }
+
+ @Test
+ fun `WHEN checkIfAddWasClicked is called with a null session URL THEN don't emit a Fact`() {
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.checkIfAddWasClicked(null, listOf())
+
+ assertTrue(facts.isEmpty())
+ }
+
+ @Test
+ fun `GIVEN no ads in the redirect path WHEN checkIfAddWasClicked is called THEN don't emit a Fact`() {
+ val sessionUrl = "https://www.google.com/search?q=aaa"
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.checkIfAddWasClicked(sessionUrl, listOf("https://www.aaa.com"))
+
+ assertTrue(facts.isEmpty())
+ }
+
+ @Test
+ fun `GIVEN ads are in the redirect path WHEN checkIfAddWasClicked is called THEN emit an appropriate SERP_ADD_CLICKED Fact`() {
+ val sessionUrl = "https://www.google.com/search?q=aaa"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.checkIfAddWasClicked(
+ sessionUrl,
+ listOf("https://www.google.com/aclk", "https://www.aaa.com"),
+ )
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(AdsTelemetry.SERP_ADD_CLICKED, facts[0].item)
+ assertEquals("google.in-content.organic.none", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a message containing ad links from the extension WHEN processMessage is called THEN track a SERP_SHOWN_WITH_ADDS Fact`() {
+ val first = "https://www.google.com/aclk"
+ val second = "https://www.google.com/aaa"
+ val urls = JSONArray()
+ urls.put(first)
+ urls.put(second)
+ val cookies = JSONArray()
+ val message = JSONObject()
+ message.put(ADS_MESSAGE_DOCUMENT_URLS_KEY, urls)
+ message.put(ADS_MESSAGE_SESSION_URL_KEY, "https://www.google.com/search?q=aaa")
+ message.put(ADS_MESSAGE_COOKIES_KEY, cookies)
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.processMessage(message)
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(AdsTelemetry.SERP_SHOWN_WITH_ADDS, facts[0].item)
+ assertEquals("google.in-content.organic.none", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a message not containing ad links from the extension WHEN processMessage is called THEN don't emit any Fact`() {
+ val first = "https://www.google.com/aaaaaa"
+ val second = "https://www.google.com/aaa"
+ val urls = JSONArray()
+ urls.put(first)
+ urls.put(second)
+ val cookies = JSONArray()
+ val message = JSONObject()
+ message.put(ADS_MESSAGE_DOCUMENT_URLS_KEY, urls)
+ message.put(ADS_MESSAGE_SESSION_URL_KEY, "https://www.google.com/search?q=aaa")
+ message.put(ADS_MESSAGE_COOKIES_KEY, cookies)
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.processMessage(message)
+
+ assertTrue(facts.isEmpty())
+ }
+
+ @Test
+ fun `GIVEN a Bing sap-follow-on with cookies WHEN checkIfAddWasClicked is called THEN emit an appropriate SERP_ADD_CLICKED Fact`() {
+ val url = "https://www.bing.com/search?q=aaa&form=QBRERANDOM"
+ telemetry.providerList = createMockProviderList()
+ telemetry.cachedCookies = createCookieList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.checkIfAddWasClicked(url, listOf("https://www.bing.com/aclik", "https://www.aaa.com"))
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(AdsTelemetry.SERP_ADD_CLICKED, facts[0].item)
+ assertEquals("bing.in-content.sap-follow-on.mozl", facts[0].value)
+ }
+
+ private fun createCookieList(): List<JSONObject> {
+ val first = JSONObject()
+ first.put("name", "SRCHS")
+ first.put("value", "PC=MOZL")
+ val second = JSONObject()
+ second.put("name", "RANDOM")
+ second.put("value", "RANDOM")
+ return listOf(first, second)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetryTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetryTest.kt
new file mode 100644
index 0000000000..2de381f11d
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/telemetry/incontent/InContentTelemetryTest.kt
@@ -0,0 +1,467 @@
+/* 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/. */
+
+package mozilla.components.feature.search.telemetry.incontent
+
+import androidx.test.ext.junit.runners.AndroidJUnit4
+import kotlinx.coroutines.runBlocking
+import mozilla.components.browser.state.store.BrowserStore
+import mozilla.components.concept.engine.Engine
+import mozilla.components.feature.search.telemetry.ExtensionInfo
+import mozilla.components.feature.search.telemetry.SearchProviderCookie
+import mozilla.components.feature.search.telemetry.SearchProviderModel
+import mozilla.components.feature.search.telemetry.incontent.InContentTelemetry.Companion.SEARCH_EXTENSION_ID
+import mozilla.components.feature.search.telemetry.incontent.InContentTelemetry.Companion.SEARCH_EXTENSION_RESOURCE_URL
+import mozilla.components.feature.search.telemetry.incontent.InContentTelemetry.Companion.SEARCH_MESSAGE_ID
+import mozilla.components.feature.search.telemetry.incontent.InContentTelemetry.Companion.SEARCH_MESSAGE_LIST_KEY
+import mozilla.components.feature.search.telemetry.incontent.InContentTelemetry.Companion.SEARCH_MESSAGE_SESSION_URL_KEY
+import mozilla.components.support.base.Component
+import mozilla.components.support.base.facts.Action
+import mozilla.components.support.base.facts.Fact
+import mozilla.components.support.base.facts.FactProcessor
+import mozilla.components.support.base.facts.Facts
+import mozilla.components.support.test.any
+import mozilla.components.support.test.argumentCaptor
+import mozilla.components.support.test.eq
+import mozilla.components.support.test.mock
+import org.json.JSONArray
+import org.json.JSONObject
+import org.junit.Assert.assertEquals
+import org.junit.Before
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.mockito.Mockito.doNothing
+import org.mockito.Mockito.spy
+import org.mockito.Mockito.verify
+
+@RunWith(AndroidJUnit4::class)
+class InContentTelemetryTest {
+ private lateinit var telemetry: InContentTelemetry
+
+ fun createMockProviderList(): List<SearchProviderModel> = listOf(
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("monline_7_dg", "monline_4_dg", "monline_3_dg", "monline_dg"),
+ telemetryId = "baidu",
+ organicCodes = emptyList(),
+ codeParamName = "tn",
+ followOnParamNames = listOf("oq"),
+ queryParamNames = listOf("wd", "word"),
+ searchPageRegexp = "^https://(?:m|www)\\.baidu\\.com/(?:s|baidu)",
+ extraAdServersRegexps = listOf("^https?://www\\.baidu\\.com/baidu\\.php?"),
+ expectedOrganicCodes = emptyList(),
+ ),
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("firefox-b-m", "fpas", "lm"),
+ telemetryId = "example",
+ organicCodes = listOf("foo"),
+ codeParamName = "pc",
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https:\\/\\/example\\.com\\/",
+ extraAdServersRegexps = listOf("^https://example.com/y\\\\.js?.*ad_provider\\\\="),
+ expectedOrganicCodes = emptyList(),
+ ),
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("firefox-b-m", "fpas", "lm"),
+ telemetryId = "duckduckgo",
+ organicCodes = emptyList(),
+ codeParamName = "t",
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https:\\/\\/duckduckgo\\.com\\/",
+ extraAdServersRegexps = listOf("^https://duckduckgo.com/y\\\\.js?.*ad_provider\\\\="),
+ expectedOrganicCodes = listOf("ha"),
+ ),
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("firefox-b-m", "fpas", "def"),
+ telemetryId = "google",
+ organicCodes = emptyList(),
+ codeParamName = "client",
+ followOnParamNames = listOf("oq", "ved", "ei"),
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https://www\\.google\\.(?:.+)/search",
+ extraAdServersRegexps = listOf("^https?://www\\\\.google(?:adservices)?\\\\.com/(?:pagead/)?aclk"),
+ expectedOrganicCodes = emptyList(),
+ ),
+ SearchProviderModel(
+ schema = 1671479978127,
+ taggedCodes = listOf("MOZ2", "MOZL", "def"),
+ telemetryId = "bing",
+ organicCodes = emptyList(),
+ codeParamName = "pc",
+ queryParamNames = listOf("q"),
+ searchPageRegexp = "^https://www\\.bing\\.com/search",
+ extraAdServersRegexps = listOf("^https://www\\\\.bing\\\\.com/acli?c?k"),
+ followOnCookies = listOf(
+ SearchProviderCookie(
+ extraCodeParamName = "form",
+ extraCodePrefixes = listOf("QBRE"),
+ host = "name",
+ name = "SRCHS",
+ codeParamName = "PC",
+ ),
+ ),
+ expectedOrganicCodes = emptyList(),
+ ),
+ )
+
+ @Before
+ fun setup() {
+ telemetry = spy(InContentTelemetry())
+ }
+
+ @Test
+ fun `WHEN installWebExtension is called THEN install a properly configured extension`() {
+ val engine: Engine = mock()
+ val store: BrowserStore = mock()
+ val extensionCaptor = argumentCaptor<ExtensionInfo>()
+
+ runBlocking {
+ doNothing().`when`(telemetry).setProviderList(any())
+ telemetry.install(engine, store, mock())
+ }
+
+ verify(telemetry).installWebExtension(eq(engine), eq(store), extensionCaptor.capture())
+ assertEquals(SEARCH_EXTENSION_ID, extensionCaptor.value.id)
+ assertEquals(SEARCH_EXTENSION_RESOURCE_URL, extensionCaptor.value.resourceUrl)
+ assertEquals(SEARCH_MESSAGE_ID, extensionCaptor.value.messageId)
+ }
+
+ @Test
+ fun `GIVEN a message from the extension WHEN processMessage is called THEN track the search`() {
+ val first = JSONObject()
+ val second = JSONObject()
+ val array = JSONArray()
+ array.put(first)
+ array.put(second)
+ val message = JSONObject()
+ val url = "https://www.google.com/search?q=aaa"
+ message.put(SEARCH_MESSAGE_LIST_KEY, array)
+ message.put(SEARCH_MESSAGE_SESSION_URL_KEY, url)
+
+ telemetry.processMessage(message)
+
+ verify(telemetry).trackPartnerUrlTypeMetric(url, listOf(first, second))
+ }
+
+ @Test
+ fun `GIVEN a Example search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://example.com/?q=aaa&pc=foo"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("example.in-content.organic.foo", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a Google search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.google.com/search?q=aaa&client=firefox-b-m"
+ telemetry.providerList = createMockProviderList()
+
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("google.in-content.sap.firefox-b-m", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a DuckDuckGo search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://duckduckgo.com/?q=aaa&t=fpas"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("duckduckgo.in-content.sap.fpas", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN an invalid Bing search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.bing.com/search?q=aaa&pc=MOZMBA"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("bing.in-content.organic.other", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a Google sap-follow-on WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.google.com/search?q=aaa&client=firefox-b-m&oq=random"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("google.in-content.sap-follow-on.firefox-b-m", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN an invalid Google sap-follow-on WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.google.com/search?q=aaa&client=firefox-b-mTesting&oq=random"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("google.in-content.organic.other", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a Google sap-follow-on from topSite WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.google.com/search?q=aaa&client=firefox-b-m&channel=ts&oq=random"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("google.in-content.sap-follow-on.firefox-b-m.ts", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN an invalid Google channel from topSite WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.google.com/search?q=aaa&client=firefox-b-m&channel=tsTest&oq=random"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("google.in-content.sap-follow-on.firefox-b-m", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a Bing sap-follow-on with cookies WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.bing.com/search?q=aaa&form=QBRERANDOM"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, createCookieList())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("bing.in-content.sap-follow-on.mozl", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a Google organic search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.google.com/search?q=aaa"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("google.in-content.organic.none", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a DuckDuckGo organic search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://duckduckgo.com/?q=aaa"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("duckduckgo.in-content.organic.none", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a DuckDuckGo organic search with expected organic code WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://duckduckgo.com/?t=ha&q=aaa"
+ val facts = mutableListOf<Fact>()
+ telemetry.providerList = createMockProviderList()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("duckduckgo.in-content.organic.none", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a Bing organic search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://www.bing.com/search?q=aaa"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("bing.in-content.organic.none", facts[0].value)
+ }
+
+ @Test
+ fun `GIVEN a Baidu organic search WHEN trackPartnerUrlTypeMetric is called THEN emit an appropriate IN_CONTENT_SEARCH fact`() {
+ val url = "https://m.baidu.com/s?word=aaa"
+ telemetry.providerList = createMockProviderList()
+ val facts = mutableListOf<Fact>()
+ Facts.registerProcessor(
+ object : FactProcessor {
+ override fun process(fact: Fact) {
+ facts.add(fact)
+ }
+ },
+ )
+
+ telemetry.trackPartnerUrlTypeMetric(url, listOf())
+
+ assertEquals(1, facts.size)
+ assertEquals(Component.FEATURE_SEARCH, facts[0].component)
+ assertEquals(Action.INTERACTION, facts[0].action)
+ assertEquals(InContentTelemetry.IN_CONTENT_SEARCH, facts[0].item)
+ assertEquals("baidu.in-content.organic.none", facts[0].value)
+ }
+
+ private fun createCookieList(): List<JSONObject> {
+ val first = JSONObject()
+ first.put("name", "SRCHS")
+ first.put("value", "PC=MOZL")
+ val second = JSONObject()
+ second.put("name", "RANDOM")
+ second.put("value", "RANDOM")
+ return listOf(first, second)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/AppSearchWidgetProviderTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/AppSearchWidgetProviderTest.kt
new file mode 100644
index 0000000000..b0b74bec65
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/AppSearchWidgetProviderTest.kt
@@ -0,0 +1,159 @@
+/* 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/. */
+
+package mozilla.components.feature.search.widget
+
+import android.content.Context
+import mozilla.components.feature.search.R
+import mozilla.components.feature.search.widget.AppSearchWidgetProvider.Companion.getLayout
+import mozilla.components.support.test.mock
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNull
+import org.junit.Test
+import org.mockito.Mockito.doReturn
+
+class AppSearchWidgetProviderTest {
+
+ private val testContext: Context = mock()
+
+ @Test
+ fun testGetLayoutSize() {
+ val sizes = mapOf(
+ 0 to SearchWidgetProviderSize.EXTRA_SMALL_V1,
+ 10 to SearchWidgetProviderSize.EXTRA_SMALL_V1,
+ 63 to SearchWidgetProviderSize.EXTRA_SMALL_V1,
+ 64 to SearchWidgetProviderSize.EXTRA_SMALL_V2,
+ 99 to SearchWidgetProviderSize.EXTRA_SMALL_V2,
+ 100 to SearchWidgetProviderSize.SMALL,
+ 191 to SearchWidgetProviderSize.SMALL,
+ 192 to SearchWidgetProviderSize.MEDIUM,
+ 255 to SearchWidgetProviderSize.MEDIUM,
+ 256 to SearchWidgetProviderSize.LARGE,
+ 1000 to SearchWidgetProviderSize.LARGE,
+ )
+
+ for ((dp, layoutSize) in sizes) {
+ assertEquals(layoutSize, AppSearchWidgetProvider.getLayoutSize(dp))
+ }
+ }
+
+ @Test
+ fun testGetLargeLayout() {
+ assertEquals(
+ R.layout.mozac_search_widget_large,
+ getLayout(SearchWidgetProviderSize.LARGE, showMic = false),
+ )
+ assertEquals(
+ R.layout.mozac_search_widget_large,
+ getLayout(SearchWidgetProviderSize.LARGE, showMic = true),
+ )
+ }
+
+ @Test
+ fun testGetMediumLayout() {
+ assertEquals(
+ R.layout.mozac_search_widget_medium,
+ getLayout(SearchWidgetProviderSize.MEDIUM, showMic = false),
+ )
+ assertEquals(
+ R.layout.mozac_search_widget_medium,
+ getLayout(SearchWidgetProviderSize.MEDIUM, showMic = true),
+ )
+ }
+
+ @Test
+ fun testGetSmallLayout() {
+ assertEquals(
+ R.layout.mozac_search_widget_small_no_mic,
+ getLayout(SearchWidgetProviderSize.SMALL, showMic = false),
+ )
+ assertEquals(
+ R.layout.mozac_search_widget_small,
+ getLayout(SearchWidgetProviderSize.SMALL, showMic = true),
+ )
+ }
+
+ @Test
+ fun testGetExtraSmall2Layout() {
+ assertEquals(
+ R.layout.mozac_search_widget_extra_small_v2,
+ getLayout(
+ SearchWidgetProviderSize.EXTRA_SMALL_V2,
+ showMic = false,
+ ),
+ )
+ assertEquals(
+ R.layout.mozac_search_widget_extra_small_v2,
+ getLayout(
+ SearchWidgetProviderSize.EXTRA_SMALL_V2,
+ showMic = true,
+ ),
+ )
+ }
+
+ @Test
+ fun testGetExtraSmall1Layout() {
+ assertEquals(
+ R.layout.mozac_search_widget_extra_small_v1,
+ getLayout(
+ SearchWidgetProviderSize.EXTRA_SMALL_V1,
+ showMic = false,
+ ),
+ )
+ assertEquals(
+ R.layout.mozac_search_widget_extra_small_v1,
+ getLayout(
+ SearchWidgetProviderSize.EXTRA_SMALL_V1,
+ showMic = true,
+ ),
+ )
+ }
+
+ @Test
+ fun testGetText() {
+ assertEquals(
+ testContext.getString(R.string.search_widget_text_long),
+ AppSearchWidgetProvider.getText(
+ SearchWidgetProviderSize.LARGE,
+ testContext,
+ ),
+ )
+ assertEquals(
+ testContext.getString(R.string.search_widget_text_short),
+ AppSearchWidgetProvider.getText(
+ SearchWidgetProviderSize.MEDIUM,
+ testContext,
+ ),
+ )
+ assertNull(
+ AppSearchWidgetProvider.getText(
+ SearchWidgetProviderSize.SMALL,
+ testContext,
+ ),
+ )
+ assertNull(
+ AppSearchWidgetProvider.getText(
+ SearchWidgetProviderSize.EXTRA_SMALL_V1,
+ testContext,
+ ),
+ )
+ assertNull(
+ AppSearchWidgetProvider.getText(
+ SearchWidgetProviderSize.EXTRA_SMALL_V2,
+ testContext,
+ ),
+ )
+ }
+
+ @Test
+ fun `GIVEN voice search is disabled WHEN createVoiceSearchIntent is called THEN it returns null`() {
+ val appSearchWidgetProvider: AppSearchWidgetProvider =
+ mock()
+ doReturn(false).`when`(appSearchWidgetProvider).shouldShowVoiceSearch(testContext)
+
+ val result = appSearchWidgetProvider.createVoiceSearchIntent(testContext)
+
+ assertNull(result)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityExtendedForTests.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityExtendedForTests.kt
new file mode 100644
index 0000000000..5777364226
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityExtendedForTests.kt
@@ -0,0 +1,24 @@
+/* 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/. */
+
+package mozilla.components.feature.search.widget
+
+import androidx.core.view.MenuProvider
+import java.util.Locale
+
+class BaseVoiceSearchActivityExtendedForTests : BaseVoiceSearchActivity() {
+
+ override fun getCurrentLocale(): Locale {
+ return Locale.getDefault()
+ }
+
+ override fun onSpeechRecognitionStarted() {
+ }
+
+ override fun onSpeechRecognitionEnded(spokenText: String) {
+ }
+
+ override fun addMenuProvider(provider: MenuProvider) {
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityTest.kt b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityTest.kt
new file mode 100644
index 0000000000..bf865e3313
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/java/mozilla/components/feature/search/widget/BaseVoiceSearchActivityTest.kt
@@ -0,0 +1,134 @@
+/* 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/. */
+
+package mozilla.components.feature.search.widget
+
+import android.app.Activity
+import android.app.Activity.RESULT_OK
+import android.content.ComponentName
+import android.content.Intent
+import android.content.IntentFilter
+import android.os.Bundle
+import android.speech.RecognizerIntent.ACTION_RECOGNIZE_SPEECH
+import android.speech.RecognizerIntent.EXTRA_RESULTS
+import androidx.activity.result.ActivityResult
+import mozilla.components.feature.search.widget.BaseVoiceSearchActivity.Companion.PREVIOUS_INTENT
+import mozilla.components.feature.search.widget.BaseVoiceSearchActivity.Companion.SPEECH_PROCESSING
+import mozilla.components.support.test.robolectric.testContext
+import mozilla.components.support.utils.ext.getParcelableCompat
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertFalse
+import org.junit.Assert.assertNull
+import org.junit.Assert.assertTrue
+import org.junit.Before
+import org.junit.Test
+import org.junit.runner.RunWith
+import org.robolectric.Robolectric
+import org.robolectric.RobolectricTestRunner
+import org.robolectric.Shadows.shadowOf
+import org.robolectric.android.controller.ActivityController
+import org.robolectric.shadows.ShadowActivity
+
+@RunWith(RobolectricTestRunner::class)
+class BaseVoiceSearchActivityTest {
+
+ private lateinit var controller: ActivityController<BaseVoiceSearchActivityExtendedForTests>
+ private lateinit var activity: BaseVoiceSearchActivityExtendedForTests
+ private lateinit var shadow: ShadowActivity
+
+ @Before
+ fun setup() {
+ val intent = Intent()
+ intent.putExtra(SPEECH_PROCESSING, true)
+
+ controller = Robolectric.buildActivity(BaseVoiceSearchActivityExtendedForTests::class.java, intent)
+ activity = controller.get()
+ shadow = shadowOf(activity)
+ }
+
+ private fun allowVoiceIntentToResolveActivity() {
+ val shadowPackageManager = shadowOf(testContext.packageManager)
+ val component = ComponentName("com.test", "Test")
+ shadowPackageManager.addActivityIfNotPresent(component)
+ shadowPackageManager.addIntentFilterForActivity(
+ component,
+ IntentFilter(ACTION_RECOGNIZE_SPEECH).apply { addCategory(Intent.CATEGORY_DEFAULT) },
+ )
+ }
+
+ @Test
+ fun `process intent with speech processing set to true`() {
+ val intent = Intent()
+ intent.putStringArrayListExtra(EXTRA_RESULTS, ArrayList<String>(listOf("hello world")))
+ val activityResult = ActivityResult(RESULT_OK, intent)
+ controller.get().activityResultImplementation(activityResult)
+
+ assertTrue(activity.isFinishing)
+ }
+
+ @Test
+ fun `process intent with speech processing set to false`() {
+ allowVoiceIntentToResolveActivity()
+ val intent = Intent()
+ intent.putExtra(SPEECH_PROCESSING, false)
+
+ val controller = Robolectric.buildActivity(BaseVoiceSearchActivityExtendedForTests::class.java, intent)
+ val activity = controller.get()
+
+ controller.create()
+
+ assertTrue(activity.isFinishing)
+ }
+
+ @Test
+ fun `process null intent`() {
+ allowVoiceIntentToResolveActivity()
+ val controller = Robolectric.buildActivity(BaseVoiceSearchActivityExtendedForTests::class.java, null)
+ val activity = controller.get()
+
+ controller.create()
+
+ assertTrue(activity.isFinishing)
+ }
+
+ @Test
+ fun `save previous intent to instance state`() {
+ allowVoiceIntentToResolveActivity()
+ val previousIntent = Intent().apply {
+ putExtra(SPEECH_PROCESSING, true)
+ }
+ val savedInstanceState = Bundle().apply {
+ putParcelable(PREVIOUS_INTENT, previousIntent)
+ }
+ val outState = Bundle()
+
+ controller.create(savedInstanceState)
+ controller.saveInstanceState(outState)
+
+ assertEquals(previousIntent, outState.getParcelableCompat(PREVIOUS_INTENT, Intent::class.java))
+ }
+
+ @Test
+ fun `process intent with speech processing in previous intent set to true`() {
+ allowVoiceIntentToResolveActivity()
+ val savedInstanceState = Bundle()
+ val previousIntent = Intent().apply {
+ putExtra(SPEECH_PROCESSING, true)
+ }
+ savedInstanceState.putParcelable(PREVIOUS_INTENT, previousIntent)
+
+ controller.create(savedInstanceState)
+
+ assertFalse(activity.isFinishing)
+ assertNull(shadow.peekNextStartedActivityForResult())
+ }
+
+ @Test
+ fun `handle invalid result code`() {
+ val activityResult = ActivityResult(Activity.RESULT_CANCELED, Intent())
+ controller.get().activityResultImplementation(activityResult)
+
+ assertTrue(activity.isFinishing)
+ }
+}
diff --git a/mobile/android/android-components/components/feature/search/src/test/resources/mockito-extensions/org.mockito.plugins.MockMaker b/mobile/android/android-components/components/feature/search/src/test/resources/mockito-extensions/org.mockito.plugins.MockMaker
new file mode 100644
index 0000000000..cf1c399ea8
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/resources/mockito-extensions/org.mockito.plugins.MockMaker
@@ -0,0 +1,2 @@
+mock-maker-inline
+// This allows mocking final classes (classes are final by default in Kotlin)
diff --git a/mobile/android/android-components/components/feature/search/src/test/resources/robolectric.properties b/mobile/android/android-components/components/feature/search/src/test/resources/robolectric.properties
new file mode 100644
index 0000000000..932b01b9eb
--- /dev/null
+++ b/mobile/android/android-components/components/feature/search/src/test/resources/robolectric.properties
@@ -0,0 +1 @@
+sdk=28