]> git.openstreetmap.org Git - nominatim.git/blob - nominatim/data/postcode_format.py
move postcode matcher in a separate file
[nominatim.git] / nominatim / data / postcode_format.py
1 # SPDX-License-Identifier: GPL-2.0-only
2 #
3 # This file is part of Nominatim. (https://nominatim.org)
4 #
5 # Copyright (C) 2022 by the Nominatim developer community.
6 # For a full list of authors see the git log.
7 """
8 Functions for formatting postcodes according to their country-specific
9 format.
10 """
11 import re
12
13 from nominatim.errors import UsageError
14 from nominatim.tools import country_info
15
16 class CountryPostcodeMatcher:
17     """ Matches and formats a postcode according to a format definition
18         of the given country.
19     """
20     def __init__(self, country_code, config):
21         if 'pattern' not in config:
22             raise UsageError("Field 'pattern' required for 'postcode' "
23                              f"for country '{country_code}'")
24
25         pc_pattern = config['pattern'].replace('d', '[0-9]').replace('l', '[A-Z]')
26
27         self.norm_pattern = re.compile(f'\\s*(?:{country_code.upper()}[ -]?)?(.*)\\s*')
28         self.pattern = re.compile(pc_pattern)
29
30         self.output = config.get('output', r'\g<0>')
31
32
33     def match(self, postcode):
34         """ Match the given postcode against the postcode pattern for this
35             matcher. Returns a `re.Match` object if the match was successful
36             and None otherwise.
37         """
38         # Upper-case, strip spaces and leading country code.
39         normalized = self.norm_pattern.fullmatch(postcode.upper())
40
41         if normalized:
42             return self.pattern.fullmatch(normalized.group(1))
43
44         return None
45
46
47     def normalize(self, match):
48         """ Return the default format of the postcode for the given match.
49             `match` must be a `re.Match` object previously returned by
50             `match()`
51         """
52         return match.expand(self.output)
53
54
55 class PostcodeFormatter:
56     """ Container for different postcode formats of the world and
57         access functions.
58     """
59     def __init__(self):
60         # Objects without a country code can't have a postcode per definition.
61         self.country_without_postcode = {None}
62         self.country_matcher = {}
63         self.default_matcher = CountryPostcodeMatcher('', {'pattern': '.*'})
64
65         for ccode, prop in country_info.iterate('postcode'):
66             if prop is False:
67                 self.country_without_postcode.add(ccode)
68             elif isinstance(prop, dict):
69                 self.country_matcher[ccode] = CountryPostcodeMatcher(ccode, prop)
70             else:
71                 raise UsageError(f"Invalid entry 'postcode' for country '{ccode}'")
72
73
74     def set_default_pattern(self, pattern):
75         """ Set the postcode match pattern to use, when a country does not
76             have a specific pattern or is marked as country without postcode.
77         """
78         self.default_matcher = CountryPostcodeMatcher('', {'pattern': pattern})
79
80
81     def match(self, country_code, postcode):
82         """ Match the given postcode against the postcode pattern for this
83             matcher. Returns a `re.Match` object if the country has a pattern
84             and the match was successful or None if the match failed.
85         """
86         if country_code in self.country_without_postcode:
87             return None
88
89         return self.country_matcher.get(country_code, self.default_matcher).match(postcode)
90
91
92     def normalize(self, country_code, match):
93         """ Return the default format of the postcode for the given match.
94             `match` must be a `re.Match` object previously returned by
95             `match()`
96         """
97         return self.country_matcher.get(country_code, self.default_matcher).normalize(match)