settings_defaults.py 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. # SPDX-License-Identifier: AGPL-3.0-or-later
  2. # lint: pylint
  3. # pylint: disable=missing-function-docstring
  4. """Implementation of the default settings.
  5. """
  6. import typing
  7. import numbers
  8. import errno
  9. import os
  10. import logging
  11. from os.path import dirname, abspath
  12. from searx.languages import language_codes as languages
  13. searx_dir = abspath(dirname(__file__))
  14. logger = logging.getLogger('searx')
  15. OUTPUT_FORMATS = ['html', 'csv', 'json', 'rss']
  16. LANGUAGE_CODES = ('', 'all') + tuple(l[0] for l in languages)
  17. OSCAR_STYLE = ('logicodev', 'logicodev-dark', 'pointhi')
  18. CATEGORY_ORDER = [
  19. 'general',
  20. 'images',
  21. 'videos',
  22. 'news',
  23. 'map',
  24. 'music',
  25. 'it',
  26. 'science',
  27. 'files',
  28. 'social media',
  29. ]
  30. STR_TO_BOOL = {
  31. '0': False,
  32. 'false': False,
  33. 'off': False,
  34. '1': True,
  35. 'true': True,
  36. 'on': True,
  37. }
  38. _UNDEFINED = object()
  39. class SettingsValue:
  40. """Check and update a setting value
  41. """
  42. def __init__(self,
  43. type_definition: typing.Union[None, typing.Any, typing.Tuple[typing.Any]]=None,
  44. default: typing.Any=None,
  45. environ_name: str=None):
  46. self.type_definition = (
  47. type_definition
  48. if type_definition is None or isinstance(type_definition, tuple)
  49. else (type_definition,)
  50. )
  51. self.default = default
  52. self.environ_name = environ_name
  53. @property
  54. def type_definition_repr(self):
  55. types_str = [
  56. t.__name__ if isinstance(t, type) else repr(t)
  57. for t in self.type_definition
  58. ]
  59. return ', '.join(types_str)
  60. def check_type_definition(self, value: typing.Any) -> None:
  61. if value in self.type_definition:
  62. return
  63. type_list = tuple(t for t in self.type_definition if isinstance(t, type))
  64. if not isinstance(value, type_list):
  65. raise ValueError(
  66. 'The value has to be one of these types/values: {}'.format(
  67. self.type_definition_repr))
  68. def __call__(self, value: typing.Any) -> typing.Any:
  69. if value == _UNDEFINED:
  70. value = self.default
  71. # override existing value with environ
  72. if self.environ_name and self.environ_name in os.environ:
  73. value = os.environ[self.environ_name]
  74. if self.type_definition == (bool,):
  75. value = STR_TO_BOOL[value.lower()]
  76. self.check_type_definition(value)
  77. return value
  78. class SettingsDirectoryValue(SettingsValue):
  79. """Check and update a setting value that is a directory path
  80. """
  81. def check_type_definition(self, value: typing.Any) -> typing.Any:
  82. super().check_type_definition(value)
  83. if not os.path.isdir(value):
  84. raise FileNotFoundError(errno.ENOENT, os.strerror(errno.ENOENT), value)
  85. def __call__(self, value: typing.Any) -> typing.Any:
  86. if value == '':
  87. value = self.default
  88. return super().__call__(value)
  89. def apply_schema(settings, schema, path_list):
  90. error = False
  91. for key, value in schema.items():
  92. if isinstance(value, SettingsValue):
  93. try:
  94. settings[key] = value(settings.get(key, _UNDEFINED))
  95. except Exception as e: # pylint: disable=broad-except
  96. # don't stop now: check other values
  97. logger.error('%s: %s', '.'.join([*path_list, key]), e)
  98. error = True
  99. elif isinstance(value, dict):
  100. error = error or apply_schema(settings.setdefault(key, {}), schema[key], [*path_list, key])
  101. else:
  102. settings.setdefault(key, value)
  103. if len(path_list) == 0 and error:
  104. raise ValueError('Invalid settings.yml')
  105. return error
  106. SCHEMA = {
  107. 'general': {
  108. 'debug': SettingsValue(bool, False, 'SEARX_DEBUG'),
  109. 'instance_name': SettingsValue(str, 'searxng'),
  110. 'contact_url': SettingsValue((None, False, str), None),
  111. },
  112. 'brand': {
  113. 'issue_url': SettingsValue(str, None),
  114. 'new_issue_url': SettingsValue(str, None),
  115. 'docs_url': SettingsValue(str, None),
  116. 'public_instances': SettingsValue(str, None),
  117. 'wiki_url': SettingsValue(str, None),
  118. },
  119. 'search': {
  120. 'safe_search': SettingsValue((0,1,2), 0),
  121. 'autocomplete': SettingsValue(str, ''),
  122. 'default_lang': SettingsValue(LANGUAGE_CODES, ''),
  123. 'ban_time_on_fail': SettingsValue(numbers.Real, 5),
  124. 'max_ban_time_on_fail': SettingsValue(numbers.Real, 120),
  125. 'formats': SettingsValue(list, OUTPUT_FORMATS),
  126. },
  127. 'server': {
  128. 'port': SettingsValue((int,str), 8888, 'SEARX_PORT'),
  129. 'bind_address': SettingsValue(str, '127.0.0.1', 'SEARX_BIND_ADDRESS'),
  130. 'secret_key': SettingsValue(str, environ_name='SEARX_SECRET'),
  131. 'base_url': SettingsValue((False, str), False),
  132. 'image_proxy': SettingsValue(bool, False),
  133. 'http_protocol_version': SettingsValue(('1.0', '1.1'), '1.0'),
  134. 'method': SettingsValue(('POST', 'GET'), 'POST'),
  135. 'default_http_headers': SettingsValue(dict, {}),
  136. },
  137. 'ui': {
  138. 'static_path': SettingsDirectoryValue(str, os.path.join(searx_dir, 'static')),
  139. 'templates_path': SettingsDirectoryValue(str, os.path.join(searx_dir, 'templates')),
  140. 'default_theme': SettingsValue(str, 'oscar'),
  141. 'default_locale': SettingsValue(str, ''),
  142. 'theme_args': {
  143. 'oscar_style': SettingsValue(OSCAR_STYLE, 'logicodev'),
  144. },
  145. 'results_on_new_tab': SettingsValue(bool, False),
  146. 'advanced_search': SettingsValue(bool, False),
  147. 'categories_order': SettingsValue(list, CATEGORY_ORDER),
  148. },
  149. 'preferences': {
  150. 'lock': SettingsValue(list, []),
  151. },
  152. 'outgoing': {
  153. 'useragent_suffix': SettingsValue(str, ''),
  154. 'request_timeout': SettingsValue(numbers.Real, 3.0),
  155. 'enable_http2': SettingsValue(bool, True),
  156. 'max_request_timeout': SettingsValue((None, numbers.Real), None),
  157. # Magic number kept from previous code
  158. 'pool_connections': SettingsValue(int, 100),
  159. # Picked from constructor
  160. 'pool_maxsize': SettingsValue(int, 10),
  161. 'keepalive_expiry': SettingsValue(numbers.Real, 5.0),
  162. # default maximum redirect
  163. # from https://github.com/psf/requests/blob/8c211a96cdbe9fe320d63d9e1ae15c5c07e179f8/requests/models.py#L55
  164. 'max_redirects': SettingsValue(int, 30),
  165. 'retries': SettingsValue(int, 0),
  166. 'proxies': SettingsValue((None, str, dict), None),
  167. 'source_ips': SettingsValue((None, str, list), None),
  168. # Tor configuration
  169. 'using_tor_proxy': SettingsValue(bool, False),
  170. 'extra_proxy_timeout': SettingsValue(int, 0),
  171. 'networks': {
  172. },
  173. },
  174. 'plugins': SettingsValue((None, list), None),
  175. 'enabled_plugins': SettingsValue(list, []),
  176. 'checker': {
  177. 'off_when_debug': SettingsValue(bool, True),
  178. },
  179. 'engines': SettingsValue(list, []),
  180. 'locales': SettingsValue(dict, {'en': 'English'}),
  181. 'doi_resolvers': {
  182. },
  183. }
  184. def settings_set_defaults(settings):
  185. apply_schema(settings, SCHEMA, [])
  186. return settings