2"""Engine's traits are fetched from the origin engines and stored in a JSON file
3in the *data folder*. Most often traits are languages and region codes and
4their mapping from SearXNG's representation to the representation in the origin
5search engine. For new traits new properties can be added to the class
6:py:class:`EngineTraits`.
8To load traits from the persistence :py:obj:`EngineTraitsMap.from_data` can be
12from __future__
import annotations
18from typing
import Dict, Literal, Iterable, Union, Callable, Optional, TYPE_CHECKING
20from searx
import locales
28 """Encodes :class:`EngineTraits` to a serializable object, see
29 :class:`json.JSONEncoder`."""
32 """Return dictionary of a :class:`EngineTraits` object."""
33 if isinstance(o, EngineTraits):
40 """The class is intended to be instantiated for each engine."""
42 regions: Dict[str, str] = dataclasses.field(default_factory=dict)
43 """Maps SearXNG's internal representation of a region to the one of the engine.
45 SearXNG's internal representation can be parsed by babel and the value is
51 'fr-BE' : <engine's region name>,
54 for key, egnine_region regions.items():
55 searxng_region = babel.Locale.parse(key, sep='-')
59 languages: Dict[str, str] = dataclasses.field(default_factory=dict)
60 """Maps SearXNG's internal representation of a language to the one of the engine.
62 SearXNG's internal representation can be parsed by babel and the value is
68 'ca' : <engine's language name>,
71 for key, egnine_lang in languages.items():
72 searxng_lang = babel.Locale.parse(key)
76 all_locale: Optional[str] =
None
77 """To which locale value SearXNG's ``all`` language is mapped (shown a "Default
81 data_type: Literal[
'traits_v1'] =
'traits_v1'
82 """Data type, default is 'traits_v1'.
85 custom: Dict[str, Union[Dict[str, Dict], Iterable[str]]] = dataclasses.field(default_factory=dict)
86 """A place to store engine's custom traits, not related to the SearXNG core.
90 """Return engine's language string that *best fits* to SearXNG's locale.
92 :param searxng_locale: SearXNG's internal representation of locale
95 :param default: engine's default language
97 The *best fits* rules are implemented in
98 :py:obj:`searx.locales.get_engine_locale`. Except for the special value ``all``
99 which is determined from :py:obj:`EngineTraits.all_locale`.
101 if searxng_locale ==
'all' and self.
all_locale is not None:
103 return locales.get_engine_locale(searxng_locale, self.
languages, default=default)
106 """Return engine's region string that best fits to SearXNG's locale.
108 :param searxng_locale: SearXNG's internal representation of locale
109 selected by the user.
111 :param default: engine's default region
113 The *best fits* rules are implemented in
114 :py:obj:`searx.locales.get_engine_locale`. Except for the special value ``all``
115 which is determined from :py:obj:`EngineTraits.all_locale`.
117 if searxng_locale ==
'all' and self.
all_locale is not None:
119 return locales.get_engine_locale(searxng_locale, self.
regions, default=default)
122 """A *locale* (SearXNG's internal representation) is considered to be
123 supported by the engine if the *region* or the *language* is supported
126 For verification the functions :py:func:`EngineTraits.get_region` and
127 :py:func:`EngineTraits.get_language` are used.
132 raise TypeError(
'engine traits of type %s is unknown' % self.
data_type)
135 """Create a copy of the dataclass object."""
140 """Call a function ``fetch_traits(engine_traits)`` from engines namespace to fetch
141 and set properties from the origin engine in the object ``engine_traits``. If
142 function does not exists, ``None`` is returned.
145 fetch_traits = getattr(engine,
'fetch_traits',
None)
149 engine_traits = cls()
154 """Set traits from self object in a :py:obj:`.Engine` namespace.
156 :param engine: engine instance build by :py:func:`searx.engines.load_engine`
162 raise TypeError(
'engine traits of type %s is unknown' % self.
data_type)
175 _msg =
"settings.yml - engine: '%s' / %s: '%s' not supported"
177 languages = traits.languages
178 if hasattr(engine,
'language'):
179 if engine.language
not in languages:
180 raise ValueError(_msg % (engine.name,
'language', engine.language))
181 traits.languages = {engine.language: languages[engine.language]}
183 regions = traits.regions
184 if hasattr(engine,
'region'):
185 if engine.region
not in regions:
186 raise ValueError(_msg % (engine.name,
'region', engine.region))
187 traits.regions = {engine.region: regions[engine.region]}
189 engine.language_support = bool(traits.languages
or traits.regions)
192 engine.traits = traits
196 """A python dictionary to map :class:`EngineTraits` by engine name."""
198 ENGINE_TRAITS_FILE = (data_dir /
'engine_traits.json').resolve()
199 """File with persistence of the :py:obj:`EngineTraitsMap`."""
202 """Store EngineTraitsMap in in file :py:obj:`self.ENGINE_TRAITS_FILE`"""
204 json.dump(self, f, indent=2, sort_keys=
True, cls=EngineTraitsEncoder)
208 """Instantiate :class:`EngineTraitsMap` object from :py:obj:`ENGINE_TRAITS`"""
210 for k, v
in ENGINE_TRAITS.items():
216 from searx
import engines
218 names = list(engines.engines)
222 for engine_name
in names:
223 engine = engines.engines[engine_name]
228 traits = EngineTraits.fetch_traits(engine)
229 except Exception
as exc:
230 log(
"FATAL: while fetch_traits %s: %s" % (engine_name, exc))
231 if os.environ.get(
'FORCE',
'').lower()
not in [
'on',
'true',
'1']:
233 v = ENGINE_TRAITS.get(engine_name)
235 log(
"FORCE: re-use old values from fetch_traits - ENGINE_TRAITS[%s]" % engine_name)
238 if traits
is not None:
239 log(
"%-20s: SearXNG languages --> %s " % (engine_name, len(traits.languages)))
240 log(
"%-20s: SearXNG regions --> %s" % (engine_name, len(traits.regions)))
241 obj[engine_name] = traits
246 """Set traits in a :py:obj:`Engine` namespace.
248 :param engine: engine instance build by :py:func:`searx.engines.load_engine`
252 if engine.name
in self.keys():
253 engine_traits = self[engine.name]
255 elif engine.engine
in self.keys():
262 engine_traits = self[engine.engine]
264 engine_traits.set_traits(engine)
'EngineTraitsMap' fetch_traits(cls, Callable log)
'EngineTraitsMap' from_data(cls)
set_traits(self, Engine|types.ModuleType engine)
_set_traits_v1(self, Engine engine)
get_region(self, str searxng_locale, default=None)
bool is_locale_supported(self, str searxng_locale)
get_language(self, str searxng_locale, default=None)
Union[ 'EngineTraits', None] fetch_traits(cls, Engine engine)
set_traits(self, Engine engine)