.oO SearXNG Developer Documentation Oo.
Loading...
Searching...
No Matches
searx.plugins.unit_converter Namespace Reference

Classes

class  SXNGPlugin
 

Functions

 symbol_to_si ()
 
str|None _parse_text_and_convert (from_query, to_query)
 

Variables

str name = ""
 
 description = gettext("")
 
str plugin_id = ""
 
str preference_section = ""
 
list CONVERT_KEYWORDS = ["in", "to", "as"]
 
str RE_MEASURE
 
list ADDITIONAL_UNITS
 
dict ALIAS_SYMBOLS
 
list SYMBOL_TO_SI = []
 

Detailed Description

A plugin for converting measured values from one unit to another unit (a
unit converter).

The plugin looks up the symbols (given in the query term) in a list of
converters, each converter is one item in the list (compare
:py:obj:`ADDITIONAL_UNITS`).  If the symbols are ambiguous, the matching units
of measurement are evaluated.  The weighting in the evaluation results from the
sorting of the :py:obj:`list of unit converters<symbol_to_si>`.

Function Documentation

◆ _parse_text_and_convert()

str | None searx.plugins.unit_converter._parse_text_and_convert ( from_query,
to_query )
protected

Definition at line 215 of file unit_converter.py.

215def _parse_text_and_convert(from_query, to_query) -> str | None:
216
217 # pylint: disable=too-many-branches, too-many-locals
218
219 if not (from_query and to_query):
220 return None
221
222 measured = re.match(RE_MEASURE, from_query, re.VERBOSE)
223 if not (measured and measured.group('number'), measured.group('unit')):
224 return None
225
226 # Symbols are not unique, if there are several hits for the from-unit, then
227 # the correct one must be determined by comparing it with the to-unit
228 # https://github.com/searxng/searxng/pull/3378#issuecomment-2080974863
229
230 # first: collecting possible units
231
232 source_list, target_list = [], []
233
234 for symbol, si_name, from_si, to_si, orig_symbol in symbol_to_si():
235
236 if symbol == measured.group('unit'):
237 source_list.append((si_name, to_si))
238 if symbol == to_query:
239 target_list.append((si_name, from_si, orig_symbol))
240
241 if not (source_list and target_list):
242 return None
243
244 source_to_si = target_from_si = target_symbol = None
245
246 # second: find the right unit by comparing list of from-units with list of to-units
247
248 for source in source_list:
249 for target in target_list:
250 if source[0] == target[0]: # compare si_name
251 source_to_si = source[1]
252 target_from_si = target[1]
253 target_symbol = target[2]
254
255 if not (source_to_si and target_from_si):
256 return None
257
258 _locale = get_locale() or 'en_US'
259
260 value = measured.group('sign') + measured.group('number') + (measured.group('E') or '')
261 value = babel.numbers.parse_decimal(value, locale=_locale)
262
263 # convert value to SI unit
264
265 if isinstance(source_to_si, (float, int)):
266 value = float(value) * source_to_si
267 else:
268 value = source_to_si(float(value))
269
270 # convert value from SI unit to target unit
271
272 if isinstance(target_from_si, (float, int)):
273 value = float(value) * target_from_si
274 else:
275 value = target_from_si(float(value))
276
277 if measured.group('E'):
278 # when incoming notation is scientific, outgoing notation is scientific
279 result = babel.numbers.format_scientific(value, locale=_locale)
280 else:
281 result = babel.numbers.format_decimal(value, locale=_locale, format='#,##0.##########;-#')
282
283 return f'{result} {target_symbol}'

References symbol_to_si().

Referenced by searx.plugins.unit_converter.SXNGPlugin.post_search().

+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ symbol_to_si()

searx.plugins.unit_converter.symbol_to_si ( )
Generates a list of tuples, each tuple is a measure unit and the fields
in the tuple are:

0. Symbol of the measure unit (e.g. 'mi' for measure unit 'miles' Q253276)

1. SI name of the measure unit (e.g. Q11573 for SI unit 'metre')

2. Factor to get SI value from measure unit (e.g. 1mi is equal to SI 1m
   multiplied by 1609.344)

3. Factor to get measure value from from SI value (e.g. SI 100m is equal to
   100mi divided by 1609.344)

The returned list is sorted, the first items are created from
``WIKIDATA_UNITS``, the second group of items is build from
:py:obj:`ADDITIONAL_UNITS` and items created from :py:obj:`ALIAS_SYMBOLS`.

If you search this list for a symbol, then a match with a symbol from
Wikidata has the highest weighting (first hit in the list), followed by the
symbols from the :py:obj:`ADDITIONAL_UNITS` and the lowest weighting is
given to the symbols resulting from the aliases :py:obj:`ALIAS_SYMBOLS`.

Definition at line 143 of file unit_converter.py.

143def symbol_to_si():
144 """Generates a list of tuples, each tuple is a measure unit and the fields
145 in the tuple are:
146
147 0. Symbol of the measure unit (e.g. 'mi' for measure unit 'miles' Q253276)
148
149 1. SI name of the measure unit (e.g. Q11573 for SI unit 'metre')
150
151 2. Factor to get SI value from measure unit (e.g. 1mi is equal to SI 1m
152 multiplied by 1609.344)
153
154 3. Factor to get measure value from from SI value (e.g. SI 100m is equal to
155 100mi divided by 1609.344)
156
157 The returned list is sorted, the first items are created from
158 ``WIKIDATA_UNITS``, the second group of items is build from
159 :py:obj:`ADDITIONAL_UNITS` and items created from :py:obj:`ALIAS_SYMBOLS`.
160
161 If you search this list for a symbol, then a match with a symbol from
162 Wikidata has the highest weighting (first hit in the list), followed by the
163 symbols from the :py:obj:`ADDITIONAL_UNITS` and the lowest weighting is
164 given to the symbols resulting from the aliases :py:obj:`ALIAS_SYMBOLS`.
165
166 """
167
168 global SYMBOL_TO_SI # pylint: disable=global-statement
169 if SYMBOL_TO_SI:
170 return SYMBOL_TO_SI
171
172 # filter out units which can't be normalized to a SI unit and filter out
173 # units without a symbol / arcsecond does not have a symbol
174 # https://www.wikidata.org/wiki/Q829073
175
176 for item in data.WIKIDATA_UNITS.values():
177 if item['to_si_factor'] and item['symbol']:
178 SYMBOL_TO_SI.append(
179 (
180 item['symbol'],
181 item['si_name'],
182 1 / item['to_si_factor'], # from_si
183 item['to_si_factor'], # to_si
184 item['symbol'],
185 )
186 )
187
188 for item in ADDITIONAL_UNITS:
189 SYMBOL_TO_SI.append(
190 (
191 item['symbol'],
192 item['si_name'],
193 item['from_si'],
194 item['to_si'],
195 item['symbol'],
196 )
197 )
198
199 alias_items = []
200 for item in SYMBOL_TO_SI:
201 for alias in ALIAS_SYMBOLS.get(item[0], ()):
202 alias_items.append(
203 (
204 alias,
205 item[1],
206 item[2], # from_si
207 item[3], # to_si
208 item[0], # origin unit
209 )
210 )
211 SYMBOL_TO_SI = SYMBOL_TO_SI + alias_items
212 return SYMBOL_TO_SI
213
214

Referenced by _parse_text_and_convert().

+ Here is the caller graph for this function:

Variable Documentation

◆ ADDITIONAL_UNITS

list searx.plugins.unit_converter.ADDITIONAL_UNITS
Initial value:
1= [
2 {
3 "si_name": "Q11579",
4 "symbol": "°C",
5 "to_si": lambda val: val + 273.15,
6 "from_si": lambda val: val - 273.15,
7 },
8 {
9 "si_name": "Q11579",
10 "symbol": "°F",
11 "to_si": lambda val: (val + 459.67) * 5 / 9,
12 "from_si": lambda val: (val * 9 / 5) - 459.67,
13 },
14]

Definition at line 89 of file unit_converter.py.

◆ ALIAS_SYMBOLS

dict searx.plugins.unit_converter.ALIAS_SYMBOLS
Initial value:
1= {
2 '°C': ('C',),
3 '°F': ('F',),
4 'mi': ('L',),
5}

Definition at line 127 of file unit_converter.py.

◆ CONVERT_KEYWORDS

list searx.plugins.unit_converter.CONVERT_KEYWORDS = ["in", "to", "as"]

Definition at line 34 of file unit_converter.py.

◆ description

searx.plugins.unit_converter.description = gettext("")

Definition at line 29 of file unit_converter.py.

◆ name

str searx.plugins.unit_converter.name = ""

Definition at line 28 of file unit_converter.py.

◆ plugin_id

str searx.plugins.unit_converter.plugin_id = ""

Definition at line 31 of file unit_converter.py.

◆ preference_section

str searx.plugins.unit_converter.preference_section = ""

Definition at line 32 of file unit_converter.py.

◆ RE_MEASURE

str searx.plugins.unit_converter.RE_MEASURE
Initial value:
1= r'''
2(?P<sign>[-+]?) # +/- or nothing for positive
3(\s*) # separator: white space or nothing
4(?P<number>[\d\.,]*) # number: 1,000.00 (en) or 1.000,00 (de)
5(?P<E>[eE][-+]?\d+)? # scientific notation: e(+/-)2 (*10^2)
6(\s*) # separator: white space or nothing
7(?P<unit>\S+) # unit of measure
8'''

Definition at line 79 of file unit_converter.py.

◆ SYMBOL_TO_SI

list searx.plugins.unit_converter.SYMBOL_TO_SI = []

Definition at line 140 of file unit_converter.py.