2020-11-27 18:32:45 +00:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-or-later
|
2024-03-11 13:06:26 +00:00
|
|
|
# pylint: disable=missing-module-docstring, too-many-branches
|
2020-11-27 18:32:45 +00:00
|
|
|
|
2022-08-27 11:41:14 +00:00
|
|
|
from typing import Optional
|
2020-11-27 18:32:45 +00:00
|
|
|
from os import environ
|
|
|
|
from os.path import dirname, join, abspath, isfile
|
|
|
|
from collections.abc import Mapping
|
|
|
|
from itertools import filterfalse
|
|
|
|
|
|
|
|
import yaml
|
|
|
|
|
|
|
|
from searx.exceptions import SearxSettingsException
|
|
|
|
|
|
|
|
|
|
|
|
searx_dir = abspath(dirname(__file__))
|
|
|
|
|
|
|
|
|
2022-08-27 11:41:14 +00:00
|
|
|
def existing_filename_or_none(file_name: str) -> Optional[str]:
|
2020-11-27 18:32:45 +00:00
|
|
|
if isfile(file_name):
|
|
|
|
return file_name
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
def load_yaml(file_name):
|
|
|
|
try:
|
|
|
|
with open(file_name, 'r', encoding='utf-8') as settings_yaml:
|
|
|
|
return yaml.safe_load(settings_yaml)
|
|
|
|
except IOError as e:
|
2020-12-17 08:57:57 +00:00
|
|
|
raise SearxSettingsException(e, file_name) from e
|
2020-11-27 18:32:45 +00:00
|
|
|
except yaml.YAMLError as e:
|
2020-12-17 08:57:57 +00:00
|
|
|
raise SearxSettingsException(e, file_name) from e
|
2020-11-27 18:32:45 +00:00
|
|
|
|
|
|
|
|
2024-05-05 21:17:35 +00:00
|
|
|
def get_yaml_file(file_name):
|
|
|
|
path = existing_filename_or_none(join(searx_dir, file_name))
|
|
|
|
if path is None:
|
|
|
|
raise FileNotFoundError(f"File {file_name} does not exist!")
|
|
|
|
|
|
|
|
return load_yaml(path)
|
|
|
|
|
|
|
|
|
2020-11-27 18:32:45 +00:00
|
|
|
def get_default_settings_path():
|
2022-08-27 11:41:14 +00:00
|
|
|
return existing_filename_or_none(join(searx_dir, 'settings.yml'))
|
2020-11-27 18:32:45 +00:00
|
|
|
|
|
|
|
|
2022-08-27 11:41:14 +00:00
|
|
|
def get_user_settings_path() -> Optional[str]:
|
|
|
|
"""Get an user settings file.
|
|
|
|
By descending priority:
|
|
|
|
1. ``environ['SEARXNG_SETTINGS_PATH']``
|
|
|
|
2. ``/etc/searxng/settings.yml`` except if ``SEARXNG_DISABLE_ETC_SETTINGS`` is ``true`` or ``1``
|
|
|
|
3. ``None``
|
|
|
|
"""
|
|
|
|
|
|
|
|
# check the environment variable SEARXNG_SETTINGS_PATH
|
|
|
|
# if the environment variable is defined, this is the last check
|
2021-10-02 10:21:02 +00:00
|
|
|
if 'SEARXNG_SETTINGS_PATH' in environ:
|
2022-08-27 11:41:14 +00:00
|
|
|
return existing_filename_or_none(environ['SEARXNG_SETTINGS_PATH'])
|
2020-11-27 18:32:45 +00:00
|
|
|
|
2023-09-15 07:53:03 +00:00
|
|
|
# if SEARXNG_DISABLE_ETC_SETTINGS don't look any further
|
2021-10-02 10:21:02 +00:00
|
|
|
if environ.get('SEARXNG_DISABLE_ETC_SETTINGS', '').lower() in ('1', 'true'):
|
2021-05-18 15:23:21 +00:00
|
|
|
return None
|
|
|
|
|
2022-08-27 11:41:14 +00:00
|
|
|
# check /etc/searxng/settings.yml
|
|
|
|
# (continue with other locations if the file is not found)
|
|
|
|
return existing_filename_or_none('/etc/searxng/settings.yml')
|
2020-11-27 18:32:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def update_dict(default_dict, user_dict):
|
|
|
|
for k, v in user_dict.items():
|
|
|
|
if isinstance(v, Mapping):
|
|
|
|
default_dict[k] = update_dict(default_dict.get(k, {}), v)
|
|
|
|
else:
|
|
|
|
default_dict[k] = v
|
|
|
|
return default_dict
|
|
|
|
|
|
|
|
|
|
|
|
def update_settings(default_settings, user_settings):
|
|
|
|
# merge everything except the engines
|
|
|
|
for k, v in user_settings.items():
|
|
|
|
if k not in ('use_default_settings', 'engines'):
|
[fix] settings_loader.py - use update_dict only for mapping types
I can't set `default_doi_resolver` in `settings.yml` if I'm using
`use_default_settings`. Searx seems to try to interpret all settings at root
level in `settings.yml` as dict, which is correct except for
`default_doi_resolver` which is at root level and a string::
File "/usr/lib/python3.9/site-packages/searx/settings_loader.py", line 125, in load_settings
update_settings(default_settings, user_settings)
File "/usr/lib/python3.9/site-packages/searx/settings_loader.py", line 61, in update_settings
update_dict(default_settings[k], v)
File "/usr/lib/python3.9/site-packages/searx/settings_loader.py", line 48, in update_dict
for k, v in user_dict.items():
AttributeError: 'str' object has no attribute 'items'
Signed-off-by: Markus Heiser <markus@darmarit.de>
Suggested-by: @0xhtml https://github.com/searx/searx/issues/2722#issuecomment-813391659
2021-04-05 14:33:48 +00:00
|
|
|
if k in default_settings and isinstance(v, Mapping):
|
2020-12-03 10:35:12 +00:00
|
|
|
update_dict(default_settings[k], v)
|
|
|
|
else:
|
|
|
|
default_settings[k] = v
|
2020-11-27 18:32:45 +00:00
|
|
|
|
2022-07-23 16:46:05 +00:00
|
|
|
categories_as_tabs = user_settings.get('categories_as_tabs')
|
|
|
|
if categories_as_tabs:
|
|
|
|
default_settings['categories_as_tabs'] = categories_as_tabs
|
|
|
|
|
2020-11-27 18:32:45 +00:00
|
|
|
# parse the engines
|
|
|
|
remove_engines = None
|
|
|
|
keep_only_engines = None
|
|
|
|
use_default_settings = user_settings.get('use_default_settings')
|
|
|
|
if isinstance(use_default_settings, dict):
|
|
|
|
remove_engines = use_default_settings.get('engines', {}).get('remove')
|
|
|
|
keep_only_engines = use_default_settings.get('engines', {}).get('keep_only')
|
|
|
|
|
|
|
|
if 'engines' in user_settings or remove_engines is not None or keep_only_engines is not None:
|
|
|
|
engines = default_settings['engines']
|
|
|
|
|
|
|
|
# parse "use_default_settings.engines.remove"
|
|
|
|
if remove_engines is not None:
|
|
|
|
engines = list(filterfalse(lambda engine: (engine.get('name')) in remove_engines, engines))
|
|
|
|
|
|
|
|
# parse "use_default_settings.engines.keep_only"
|
|
|
|
if keep_only_engines is not None:
|
|
|
|
engines = list(filter(lambda engine: (engine.get('name')) in keep_only_engines, engines))
|
|
|
|
|
|
|
|
# parse "engines"
|
|
|
|
user_engines = user_settings.get('engines')
|
|
|
|
if user_engines:
|
|
|
|
engines_dict = dict((definition['name'], definition) for definition in engines)
|
|
|
|
for user_engine in user_engines:
|
|
|
|
default_engine = engines_dict.get(user_engine['name'])
|
|
|
|
if default_engine:
|
|
|
|
update_dict(default_engine, user_engine)
|
|
|
|
else:
|
|
|
|
engines.append(user_engine)
|
|
|
|
|
|
|
|
# store the result
|
|
|
|
default_settings['engines'] = engines
|
|
|
|
|
|
|
|
return default_settings
|
|
|
|
|
|
|
|
|
|
|
|
def is_use_default_settings(user_settings):
|
|
|
|
use_default_settings = user_settings.get('use_default_settings')
|
|
|
|
if use_default_settings is True:
|
|
|
|
return True
|
|
|
|
if isinstance(use_default_settings, dict):
|
|
|
|
return True
|
|
|
|
if use_default_settings is False or use_default_settings is None:
|
|
|
|
return False
|
|
|
|
raise ValueError('Invalid value for use_default_settings')
|
|
|
|
|
|
|
|
|
2022-09-27 15:01:00 +00:00
|
|
|
def load_settings(load_user_settings=True):
|
2020-11-27 18:32:45 +00:00
|
|
|
default_settings_path = get_default_settings_path()
|
|
|
|
user_settings_path = get_user_settings_path()
|
2022-09-27 15:01:00 +00:00
|
|
|
if user_settings_path is None or not load_user_settings:
|
2020-11-27 18:32:45 +00:00
|
|
|
# no user settings
|
2021-12-27 08:26:22 +00:00
|
|
|
return (load_yaml(default_settings_path), 'load the default settings from {}'.format(default_settings_path))
|
2020-11-27 18:32:45 +00:00
|
|
|
|
|
|
|
# user settings
|
|
|
|
user_settings = load_yaml(user_settings_path)
|
|
|
|
if is_use_default_settings(user_settings):
|
|
|
|
# the user settings are merged with the default configuration
|
|
|
|
default_settings = load_yaml(default_settings_path)
|
|
|
|
update_settings(default_settings, user_settings)
|
2021-12-27 08:26:22 +00:00
|
|
|
return (
|
|
|
|
default_settings,
|
2022-09-27 15:01:00 +00:00
|
|
|
'merge the default settings ( {} ) and the user settings ( {} )'.format(
|
2021-12-27 08:26:22 +00:00
|
|
|
default_settings_path, user_settings_path
|
|
|
|
),
|
|
|
|
)
|
2020-11-27 18:32:45 +00:00
|
|
|
|
|
|
|
# the user settings, fully replace the default configuration
|
2021-12-27 08:26:22 +00:00
|
|
|
return (user_settings, 'load the user settings from {}'.format(user_settings_path))
|