mirror of
https://github.com/searxng/searxng.git
synced 2025-11-26 08:15:06 -05:00
brave web:
xpath selectors needed to be justified
brave images & videos:
The JS code with the JS object was read incorrectly; not always, but quite
often, it led to exceptions when the Python data structure was created from it.
BTW: A complete review was conducted and corrections or additions were made to
the type definitions.
To test all brave engines in once::
!br !brimg !brvid !brnews weather
Signed-off-by: Markus Heiser <markus.heiser@darmarit.de>
479 lines
15 KiB
Python
479 lines
15 KiB
Python
# SPDX-License-Identifier: AGPL-3.0-or-later
|
|
"""Brave supports the categories listed in :py:obj:`brave_category` (General,
|
|
news, videos, images). The support of :py:obj:`paging` and :py:obj:`time range
|
|
<time_range_support>` is limited (see remarks).
|
|
|
|
Configured ``brave`` engines:
|
|
|
|
.. code:: yaml
|
|
|
|
- name: brave
|
|
engine: brave
|
|
...
|
|
brave_category: search
|
|
time_range_support: true
|
|
paging: true
|
|
|
|
- name: brave.images
|
|
engine: brave
|
|
...
|
|
brave_category: images
|
|
|
|
- name: brave.videos
|
|
engine: brave
|
|
...
|
|
brave_category: videos
|
|
|
|
- name: brave.news
|
|
engine: brave
|
|
...
|
|
brave_category: news
|
|
|
|
- name: brave.goggles
|
|
time_range_support: true
|
|
paging: true
|
|
...
|
|
brave_category: goggles
|
|
|
|
|
|
.. _brave regions:
|
|
|
|
Brave regions
|
|
=============
|
|
|
|
Brave uses two-digit tags for the regions like ``ca`` while SearXNG deals with
|
|
locales. To get a mapping, all *officiat de-facto* languages of the Brave
|
|
region are mapped to regions in SearXNG (see :py:obj:`babel
|
|
<babel.languages.get_official_languages>`):
|
|
|
|
.. code:: python
|
|
|
|
"regions": {
|
|
..
|
|
"en-CA": "ca",
|
|
"fr-CA": "ca",
|
|
..
|
|
}
|
|
|
|
|
|
.. note::
|
|
|
|
The language (aka region) support of Brave's index is limited to very basic
|
|
languages. The search results for languages like Chinese or Arabic are of
|
|
low quality.
|
|
|
|
|
|
.. _brave googles:
|
|
|
|
Brave Goggles
|
|
=============
|
|
|
|
.. _list of Goggles: https://search.brave.com/goggles/discover
|
|
.. _Goggles Whitepaper: https://brave.com/static-assets/files/goggles.pdf
|
|
.. _Goggles Quickstart: https://github.com/brave/goggles-quickstart
|
|
|
|
Goggles allow you to choose, alter, or extend the ranking of Brave Search
|
|
results (`Goggles Whitepaper`_). Goggles are openly developed by the community
|
|
of Brave Search users.
|
|
|
|
Select from the `list of Goggles`_ people have published, or create your own
|
|
(`Goggles Quickstart`_).
|
|
|
|
|
|
.. _brave languages:
|
|
|
|
Brave languages
|
|
===============
|
|
|
|
Brave's language support is limited to the UI (menus, area local notations,
|
|
etc). Brave's index only seems to support a locale, but it does not seem to
|
|
support any languages in its index. The choice of available languages is very
|
|
small (and its not clear to me where the difference in UI is when switching
|
|
from en-us to en-ca or en-gb).
|
|
|
|
In the :py:obj:`EngineTraits object <searx.enginelib.traits.EngineTraits>` the
|
|
UI languages are stored in a custom field named ``ui_lang``:
|
|
|
|
.. code:: python
|
|
|
|
"custom": {
|
|
"ui_lang": {
|
|
"ca": "ca",
|
|
"de-DE": "de-de",
|
|
"en-CA": "en-ca",
|
|
"en-GB": "en-gb",
|
|
"en-US": "en-us",
|
|
"es": "es",
|
|
"fr-CA": "fr-ca",
|
|
"fr-FR": "fr-fr",
|
|
"ja-JP": "ja-jp",
|
|
"pt-BR": "pt-br",
|
|
"sq-AL": "sq-al"
|
|
}
|
|
},
|
|
|
|
Implementations
|
|
===============
|
|
|
|
"""
|
|
|
|
import typing as t
|
|
|
|
from urllib.parse import (
|
|
urlencode,
|
|
urlparse,
|
|
)
|
|
|
|
import json
|
|
from dateutil import parser
|
|
from lxml import html
|
|
|
|
from searx import locales
|
|
from searx.utils import (
|
|
extract_text,
|
|
eval_xpath_list,
|
|
eval_xpath_getindex,
|
|
js_obj_str_to_python,
|
|
js_obj_str_to_json_str,
|
|
get_embeded_stream_url,
|
|
)
|
|
from searx.enginelib.traits import EngineTraits
|
|
from searx.result_types import EngineResults
|
|
from searx.extended_types import SXNG_Response
|
|
|
|
about = {
|
|
"website": "https://search.brave.com/",
|
|
"wikidata_id": "Q22906900",
|
|
"official_api_documentation": None,
|
|
"use_official_api": False,
|
|
"require_api_key": False,
|
|
"results": "HTML",
|
|
}
|
|
|
|
base_url = "https://search.brave.com/"
|
|
categories = []
|
|
brave_category: t.Literal["search", "videos", "images", "news", "goggles"] = "search"
|
|
"""Brave supports common web-search, videos, images, news, and goggles search.
|
|
|
|
- ``search``: Common WEB search
|
|
- ``videos``: search for videos
|
|
- ``images``: search for images
|
|
- ``news``: search for news
|
|
- ``goggles``: Common WEB search with custom rules, requires a :py:obj:`Goggles` URL.
|
|
"""
|
|
|
|
Goggles: str = ""
|
|
"""This should be a URL ending in ``.goggle``"""
|
|
|
|
brave_spellcheck = False
|
|
"""Brave supports some kind of spell checking. When activated, Brave tries to
|
|
fix typos, e.g. it searches for ``food`` when the user queries for ``fooh``. In
|
|
the UI of Brave the user gets warned about this, since we can not warn the user
|
|
in SearXNG, the spellchecking is disabled by default.
|
|
"""
|
|
|
|
send_accept_language_header = True
|
|
paging = False
|
|
"""Brave only supports paging in :py:obj:`brave_category` ``search`` (UI
|
|
category All) and in the goggles category."""
|
|
max_page = 10
|
|
"""Tested 9 pages maximum (``&offset=8``), to be save max is set to 10. Trying
|
|
to do more won't return any result and you will most likely be flagged as a bot.
|
|
"""
|
|
|
|
safesearch = True
|
|
safesearch_map = {2: "strict", 1: "moderate", 0: "off"} # cookie: safesearch=off
|
|
|
|
time_range_support = False
|
|
"""Brave only supports time-range in :py:obj:`brave_category` ``search`` (UI
|
|
category All) and in the goggles category."""
|
|
|
|
time_range_map: dict[str, str] = {
|
|
"day": "pd",
|
|
"week": "pw",
|
|
"month": "pm",
|
|
"year": "py",
|
|
}
|
|
|
|
|
|
def request(query: str, params: dict[str, t.Any]) -> None:
|
|
|
|
args: dict[str, t.Any] = {
|
|
"q": query,
|
|
"source": "web",
|
|
}
|
|
if brave_spellcheck:
|
|
args["spellcheck"] = "1"
|
|
|
|
if brave_category in ("search", "goggles"):
|
|
if params.get("pageno", 1) - 1:
|
|
args["offset"] = params.get("pageno", 1) - 1
|
|
if time_range_map.get(params["time_range"]):
|
|
args["tf"] = time_range_map.get(params["time_range"])
|
|
|
|
if brave_category == "goggles":
|
|
args["goggles_id"] = Goggles
|
|
|
|
params["url"] = f"{base_url}{brave_category}?{urlencode(args)}"
|
|
logger.debug("url %s", params["url"])
|
|
|
|
# set properties in the cookies
|
|
|
|
params["cookies"]["safesearch"] = safesearch_map.get(params["safesearch"], "off")
|
|
# the useLocation is IP based, we use cookie "country" for the region
|
|
params["cookies"]["useLocation"] = "0"
|
|
params["cookies"]["summarizer"] = "0"
|
|
|
|
engine_region = traits.get_region(params["searxng_locale"], "all")
|
|
params["cookies"]["country"] = engine_region.split("-")[-1].lower() # type: ignore
|
|
|
|
ui_lang = locales.get_engine_locale(params["searxng_locale"], traits.custom["ui_lang"], "en-us")
|
|
params["cookies"]["ui_lang"] = ui_lang
|
|
logger.debug("cookies %s", params["cookies"])
|
|
|
|
|
|
def _extract_published_date(published_date_raw: str | None):
|
|
if published_date_raw is None:
|
|
return None
|
|
try:
|
|
return parser.parse(published_date_raw)
|
|
except parser.ParserError:
|
|
return None
|
|
|
|
|
|
def extract_json_data(text: str) -> dict[str, t.Any]:
|
|
# Example script source containing the data:
|
|
#
|
|
# kit.start(app, element, {
|
|
# node_ids: [0, 19],
|
|
# data: [{type:"data",data: .... ["q","goggles_id"],route:1,url:1}}]
|
|
# ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
|
text = text[text.index("<script") : text.index("</script")]
|
|
if not text:
|
|
raise ValueError("can't find JS/JSON data in the given text")
|
|
start = text.index("data: [{")
|
|
end = text.rindex("}}]")
|
|
js_obj_str = text[start:end]
|
|
js_obj_str = "{" + js_obj_str + "}}]}"
|
|
# js_obj_str = js_obj_str.replace("\xa0", "") # remove ASCII for
|
|
# js_obj_str = js_obj_str.replace(r"\u003C", "<").replace(r"\u003c", "<") # fix broken HTML tags in strings
|
|
json_str = js_obj_str_to_json_str(js_obj_str)
|
|
data: dict[str, t.Any] = json.loads(json_str)
|
|
return data
|
|
|
|
|
|
def response(resp: SXNG_Response) -> EngineResults:
|
|
|
|
if brave_category in ('search', 'goggles'):
|
|
return _parse_search(resp)
|
|
|
|
if brave_category in ('news'):
|
|
return _parse_news(resp)
|
|
|
|
# Example script source containing the data:
|
|
#
|
|
# kit.start(app, element, {
|
|
# node_ids: [0, 19],
|
|
# data: [{type:"data",data: .... ["q","goggles_id"],route:1,url:1}}]
|
|
# ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
|
json_data: dict[str, t.Any] = extract_json_data(resp.text)
|
|
json_resp: dict[str, t.Any] = json_data['data'][1]["data"]['body']['response']
|
|
|
|
if brave_category == 'images':
|
|
return _parse_images(json_resp)
|
|
if brave_category == 'videos':
|
|
return _parse_videos(json_resp)
|
|
|
|
raise ValueError(f"Unsupported brave category: {brave_category}")
|
|
|
|
|
|
def _parse_search(resp: SXNG_Response) -> EngineResults:
|
|
res = EngineResults()
|
|
dom = html.fromstring(resp.text)
|
|
|
|
for result in eval_xpath_list(dom, "//div[contains(@class, 'snippet ')]"):
|
|
|
|
url: str | None = eval_xpath_getindex(result, ".//a/@href", 0, default=None)
|
|
title_tag = eval_xpath_getindex(result, ".//div[contains(@class, 'title')]", 0, default=None)
|
|
if url is None or title_tag is None or not urlparse(url).netloc: # partial url likely means it's an ad
|
|
continue
|
|
|
|
content: str = ""
|
|
pub_date = None
|
|
|
|
_content = eval_xpath_getindex(result, ".//div[contains(@class, 'content')]", 0, default="")
|
|
if len(_content):
|
|
content = extract_text(_content) # type: ignore
|
|
_pub_date = extract_text(
|
|
eval_xpath_getindex(_content, ".//span[contains(@class, 't-secondary')]", 0, default="")
|
|
)
|
|
if _pub_date:
|
|
pub_date = _extract_published_date(_pub_date)
|
|
content = content.lstrip(_pub_date).strip("- \n\t")
|
|
|
|
thumbnail: str = eval_xpath_getindex(result, ".//a[contains(@class, 'thumbnail')]//img/@src", 0, default="")
|
|
|
|
item = res.types.LegacyResult(
|
|
template="default.html",
|
|
url=url,
|
|
title=extract_text(title_tag),
|
|
content=content,
|
|
publishedDate=pub_date,
|
|
thumbnail=thumbnail,
|
|
)
|
|
res.add(item)
|
|
|
|
video_tag = eval_xpath_getindex(
|
|
result, ".//div[contains(@class, 'video-snippet') and @data-macro='video']", 0, default=[]
|
|
)
|
|
if len(video_tag):
|
|
# In my tests a video tag in the WEB search was most often not a
|
|
# video, except the ones from youtube ..
|
|
iframe_src = get_embeded_stream_url(url)
|
|
if iframe_src:
|
|
item["iframe_src"] = iframe_src
|
|
item["template"] = "videos.html"
|
|
|
|
return res
|
|
|
|
|
|
def _parse_news(resp: SXNG_Response) -> EngineResults:
|
|
res = EngineResults()
|
|
dom = html.fromstring(resp.text)
|
|
|
|
for result in eval_xpath_list(dom, "//div[contains(@class, 'results')]//div[@data-type='news']"):
|
|
|
|
url = eval_xpath_getindex(result, ".//a[contains(@class, 'result-header')]/@href", 0, default=None)
|
|
if url is None:
|
|
continue
|
|
|
|
title = eval_xpath_list(result, ".//span[contains(@class, 'snippet-title')]")
|
|
content = eval_xpath_list(result, ".//p[contains(@class, 'desc')]")
|
|
thumbnail = eval_xpath_getindex(result, ".//div[contains(@class, 'image-wrapper')]//img/@src", 0, default="")
|
|
|
|
item = res.types.LegacyResult(
|
|
template="default.html",
|
|
url=url,
|
|
title=extract_text(title),
|
|
thumbnail=thumbnail,
|
|
content=extract_text(content),
|
|
)
|
|
res.add(item)
|
|
|
|
return res
|
|
|
|
|
|
def _parse_images(json_resp: dict[str, t.Any]) -> EngineResults:
|
|
res = EngineResults()
|
|
|
|
for result in json_resp["results"]:
|
|
item = res.types.LegacyResult(
|
|
template="images.html",
|
|
url=result["url"],
|
|
title=result["title"],
|
|
source=result["source"],
|
|
img_src=result["properties"]["url"],
|
|
thumbnail_src=result["thumbnail"]["src"],
|
|
)
|
|
res.add(item)
|
|
|
|
return res
|
|
|
|
|
|
def _parse_videos(json_resp: dict[str, t.Any]) -> EngineResults:
|
|
res = EngineResults()
|
|
|
|
for result in json_resp["results"]:
|
|
item = res.types.LegacyResult(
|
|
template="videos.html",
|
|
url=result["url"],
|
|
title=result["title"],
|
|
content=result["description"],
|
|
length=result["video"]["duration"],
|
|
duration=result["video"]["duration"],
|
|
publishedDate=_extract_published_date(result["age"]),
|
|
)
|
|
if result["thumbnail"] is not None:
|
|
item["thumbnail"] = result["thumbnail"]["src"]
|
|
iframe_src = get_embeded_stream_url(result["url"])
|
|
if iframe_src:
|
|
item["iframe_src"] = iframe_src
|
|
|
|
res.add(item)
|
|
|
|
return res
|
|
|
|
|
|
def fetch_traits(engine_traits: EngineTraits):
|
|
"""Fetch :ref:`languages <brave languages>` and :ref:`regions <brave
|
|
regions>` from Brave."""
|
|
|
|
# pylint: disable=import-outside-toplevel, too-many-branches
|
|
|
|
import babel.languages
|
|
from searx.locales import region_tag, language_tag
|
|
from searx.network import get # see https://github.com/searxng/searxng/issues/762
|
|
|
|
engine_traits.custom["ui_lang"] = {}
|
|
|
|
lang_map = {'no': 'nb'} # norway
|
|
|
|
# languages (UI)
|
|
|
|
resp = get('https://search.brave.com/settings')
|
|
|
|
if not resp.ok:
|
|
print("ERROR: response from Brave is not OK.")
|
|
dom = html.fromstring(resp.text)
|
|
|
|
for option in dom.xpath("//section//option[@value='en-us']/../option"):
|
|
|
|
ui_lang = option.get("value")
|
|
try:
|
|
l = babel.Locale.parse(ui_lang, sep="-")
|
|
if l.territory:
|
|
sxng_tag = region_tag(babel.Locale.parse(ui_lang, sep="-"))
|
|
else:
|
|
sxng_tag = language_tag(babel.Locale.parse(ui_lang, sep="-"))
|
|
|
|
except babel.UnknownLocaleError:
|
|
print("ERROR: can't determine babel locale of Brave's (UI) language %s" % ui_lang)
|
|
continue
|
|
|
|
conflict = engine_traits.custom["ui_lang"].get(sxng_tag) # type: ignore
|
|
if conflict:
|
|
if conflict != ui_lang:
|
|
print("CONFLICT: babel %s --> %s, %s" % (sxng_tag, conflict, ui_lang))
|
|
continue
|
|
engine_traits.custom["ui_lang"][sxng_tag] = ui_lang
|
|
|
|
# search regions of brave
|
|
|
|
resp = get("https://cdn.search.brave.com/serp/v2/_app/immutable/chunks/parameters.734c106a.js")
|
|
|
|
if not resp.ok:
|
|
print("ERROR: response from Brave is not OK.")
|
|
|
|
country_js = resp.text[resp.text.index("options:{all") + len("options:") :]
|
|
country_js = country_js[: country_js.index("},k={default")]
|
|
country_tags = js_obj_str_to_python(country_js)
|
|
|
|
for k, v in country_tags.items():
|
|
if k == "all":
|
|
engine_traits.all_locale = "all"
|
|
continue
|
|
country_tag = v["value"]
|
|
|
|
# add official languages of the country ..
|
|
for lang_tag in babel.languages.get_official_languages(country_tag, de_facto=True):
|
|
lang_tag = lang_map.get(lang_tag, lang_tag)
|
|
sxng_tag = region_tag(babel.Locale.parse("%s_%s" % (lang_tag, country_tag.upper())))
|
|
# print("%-20s: %s <-- %s" % (v["label"], country_tag, sxng_tag))
|
|
|
|
conflict = engine_traits.regions.get(sxng_tag)
|
|
if conflict:
|
|
if conflict != country_tag:
|
|
print("CONFLICT: babel %s --> %s, %s" % (sxng_tag, conflict, country_tag))
|
|
continue
|
|
engine_traits.regions[sxng_tag] = country_tag
|