1 # SPDX-License-Identifier: GPL-2.0-only
3 # This file is part of Nominatim. (https://nominatim.org)
5 # Copyright (C) 2022 by the Nominatim developer community.
6 # For a full list of authors see the git log.
8 Configuration for Sanitizers.
10 from typing import Sequence, Union, Optional, Pattern, Callable, Any, TYPE_CHECKING
11 from collections import UserDict
14 from nominatim.errors import UsageError
16 # working around missing generics in Python < 3.8
17 # See https://github.com/python/typing/issues/60#issuecomment-869757075
19 _BaseUserDict = UserDict[str, Any]
21 _BaseUserDict = UserDict
23 class SanitizerConfig(_BaseUserDict):
24 """ The `SanitizerConfig` class is a read-only dictionary
25 with configuration options for the sanitizer.
26 In addition to the usual dictionary functions, the class provides
27 accessors to standard sanitizer options that are used by many of the
31 def get_string_list(self, param: str, default: Sequence[str] = tuple()) -> Sequence[str]:
32 """ Extract a configuration parameter as a string list.
35 param: Name of the configuration parameter.
36 default: Takes a tuple or list of strings which will
37 be returned if the parameter is missing in the
38 sanitizer configuration.
39 Note that if this default parameter is not
40 provided then an empty list is returned.
43 If the parameter value is a simple string, it is returned as a
44 one-item list. If the parameter value does not exist, the given
45 default is returned. If the parameter value is a list, it is
46 checked to contain only strings before being returned.
48 values = self.data.get(param, None)
53 if isinstance(values, str):
54 return [values] if values else []
56 if not isinstance(values, (list, tuple)):
57 raise UsageError(f"Parameter '{param}' must be string or list of strings.")
59 if any(not isinstance(value, str) for value in values):
60 raise UsageError(f"Parameter '{param}' must be string or list of strings.")
65 def get_bool(self, param: str, default: Optional[bool] = None) -> bool:
66 """ Extract a configuration parameter as a boolean.
69 param: Name of the configuration parameter. The parameter must
70 contain one of the yaml boolean values or an
71 UsageError will be raised.
72 default: Value to return, when the parameter is missing.
73 When set to `None`, the parameter must be defined.
76 Boolean value of the given parameter.
78 value = self.data.get(param, default)
80 if not isinstance(value, bool):
81 raise UsageError(f"Parameter '{param}' must be a boolean value ('yes' or 'no').")
86 def get_delimiter(self, default: str = ',;') -> Pattern[str]:
87 """ Return the 'delimiters' parameter in the configuration as a
88 compiled regular expression that can be used to split strings on
92 default: Delimiters to be used when 'delimiters' parameter
93 is not explicitly configured.
96 A regular expression pattern which can be used to
97 split a string. The regular expression makes sure that the
98 resulting names are stripped and that repeated delimiters
99 are ignored. It may still create empty fields on occasion. The
100 code needs to filter those.
102 delimiter_set = set(self.data.get('delimiters', default))
103 if not delimiter_set:
104 raise UsageError("Empty 'delimiter' parameter not allowed for sanitizer.")
106 return re.compile('\\s*[{}]+\\s*'.format(''.join('\\' + d for d in delimiter_set)))
109 def get_filter(self, param: str, default: Union[str, Sequence[str]] = 'PASS_ALL'
110 ) -> Callable[[str], bool]:
111 """ Returns a filter function for the given parameter of the sanitizer
114 The value provided for the parameter in sanitizer configuration
115 should be a string or list of strings, where each string is a regular
116 expression. These regular expressions will later be used by the
117 filter function to filter strings.
120 param: The parameter for which the filter function
122 default: Defines the behaviour of filter function if
123 parameter is missing in the sanitizer configuration.
124 Takes a string(PASS_ALL or FAIL_ALL) or a list of strings.
125 Any other value of string or an empty list is not allowed,
126 and will raise a ValueError. If the value is PASS_ALL, the filter
127 function will let all strings to pass, if the value is FAIL_ALL,
128 filter function will let no strings to pass.
129 If value provided is a list of strings each string
130 is treated as a regular expression. In this case these regular
131 expressions will be used by the filter function.
132 By default allow filter function to let all strings pass.
135 A filter function that takes a target string as the argument and
136 returns True if it fully matches any of the regular expressions
137 otherwise returns False.
139 filters = self.get_string_list(param) or default
141 if filters == 'PASS_ALL':
142 return lambda _: True
143 if filters == 'FAIL_ALL':
144 return lambda _: False
146 if filters and isinstance(filters, (list, tuple)):
147 regexes = [re.compile(regex) for regex in filters]
148 return lambda target: any(regex.fullmatch(target) for regex in regexes)
150 raise ValueError("Default parameter must be a non-empty list or a string value \
151 ('PASS_ALL' or 'FAIL_ALL').")