]> git.openstreetmap.org Git - nominatim.git/blob - src/nominatim_db/tokenizer/sanitizers/clean_postcodes.py
Merge pull request #3715 from lonvia/demote-tags-to-fallbacks
[nominatim.git] / src / nominatim_db / tokenizer / sanitizers / clean_postcodes.py
1 # SPDX-License-Identifier: GPL-3.0-or-later
2 #
3 # This file is part of Nominatim. (https://nominatim.org)
4 #
5 # Copyright (C) 2024 by the Nominatim developer community.
6 # For a full list of authors see the git log.
7 """
8 Sanitizer that filters postcodes by their officially allowed pattern.
9
10 Arguments:
11     convert-to-address: If set to 'yes' (the default), then postcodes that do
12                         not conform with their country-specific pattern are
13                         converted to an address component. That means that
14                         the postcode does not take part when computing the
15                         postcode centroids of a country but is still searchable.
16                         When set to 'no', non-conforming postcodes are not
17                         searchable either.
18     default-pattern:    Pattern to use, when there is none available for the
19                         country in question. Warning: will not be used for
20                         objects that have no country assigned. These are always
21                         assumed to have no postcode.
22 """
23 from typing import Callable, Optional, Tuple
24
25 from ...data.postcode_format import PostcodeFormatter
26 from .base import ProcessInfo
27 from .config import SanitizerConfig
28
29
30 class _PostcodeSanitizer:
31
32     def __init__(self, config: SanitizerConfig) -> None:
33         self.convert_to_address = config.get_bool('convert-to-address', True)
34         self.matcher = PostcodeFormatter()
35
36         default_pattern = config.get('default-pattern')
37         if default_pattern is not None and isinstance(default_pattern, str):
38             self.matcher.set_default_pattern(default_pattern)
39
40     def __call__(self, obj: ProcessInfo) -> None:
41         if not obj.address:
42             return
43
44         postcodes = ((i, o) for i, o in enumerate(obj.address) if o.kind == 'postcode')
45
46         for pos, postcode in postcodes:
47             formatted = self.scan(postcode.name, obj.place.country_code)
48
49             if formatted is None:
50                 if self.convert_to_address:
51                     postcode.kind = 'unofficial_postcode'
52                 else:
53                     obj.address.pop(pos)
54             else:
55                 postcode.name = formatted[0]
56                 postcode.set_attr('variant', formatted[1])
57
58     def scan(self, postcode: str, country: Optional[str]) -> Optional[Tuple[str, str]]:
59         """ Check the postcode for correct formatting and return the
60             normalized version. Returns None if the postcode does not
61             correspond to the official format of the given country.
62         """
63         match = self.matcher.match(country, postcode)
64         if match is None:
65             return None
66
67         assert country is not None
68
69         return self.matcher.normalize(country, match), \
70             ' '.join(filter(lambda p: p is not None, match.groups()))
71
72
73 def create(config: SanitizerConfig) -> Callable[[ProcessInfo], None]:
74     """ Create a function that filters postcodes by their officially allowed pattern.
75     """
76
77     return _PostcodeSanitizer(config)