settings_defaults.py 6.9 KB

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