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 -- Functions related to search and address ranks
 
  10 -- Return an approximate search radius according to the search rank.
 
  11 CREATE OR REPLACE FUNCTION reverse_place_diameter(rank_search SMALLINT)
 
  15   IF rank_search <= 4 THEN
 
  17   ELSIF rank_search <= 8 THEN
 
  19   ELSIF rank_search <= 12 THEN
 
  21   ELSIF rank_search <= 17 THEN
 
  23   ELSIF rank_search <= 18 THEN
 
  25   ELSIF rank_search <= 19 THEN
 
  32 LANGUAGE plpgsql IMMUTABLE;
 
  35 -- Return an approximate update radius according to the search rank.
 
  36 CREATE OR REPLACE FUNCTION update_place_diameter(rank_search SMALLINT)
 
  41   IF rank_search = 11 or rank_search = 5 THEN
 
  43   -- anything higher than city is effectively ignored (polygon required)
 
  44   ELSIF rank_search < 16 THEN
 
  46   ELSIF rank_search < 18 THEN
 
  48   ELSIF rank_search < 20 THEN
 
  50   ELSIF rank_search = 21 THEN
 
  52   ELSIF rank_search < 24 THEN
 
  54   ELSIF rank_search < 26 THEN
 
  56   ELSIF rank_search < 28 THEN
 
  63 LANGUAGE plpgsql IMMUTABLE;
 
  65 -- Compute a base address rank from the extent of the given geometry.
 
  67 -- This is all simple guess work. We don't need particularly good estimates
 
  68 -- here. This just avoids to have very high ranked address parts in features
 
  69 -- that span very large areas (or vice versa).
 
  70 CREATE OR REPLACE FUNCTION geometry_to_rank(search_rank SMALLINT, geometry GEOMETRY, country_code TEXT)
 
  76   IF ST_GeometryType(geometry) in ('ST_Polygon','ST_MultiPolygon') THEN
 
  77       area := ST_Area(geometry);
 
  78   ELSIF ST_GeometryType(geometry) in ('ST_LineString','ST_MultiLineString') THEN
 
  79       area := (ST_Length(geometry)^2) * 0.1;
 
  84   -- adjust for the fact that countries come in different sizes
 
  85   IF country_code IN ('ca', 'au', 'ru') THEN
 
  87   ELSIF country_code IN ('br', 'kz', 'cn', 'us', 'ne', 'gb', 'za', 'sa', 'id', 'eh', 'ml', 'tm') THEN
 
  89   ELSIF country_code IN ('bo', 'ar', 'sd', 'mn', 'in', 'et', 'cd', 'mz', 'ly', 'cl', 'zm') THEN
 
  97   ELSIF area > 0.01 THEN
 
  99   ELSIF area > 0.001 THEN
 
 101   ELSIF area > 0.0001 THEN
 
 103   ELSIF area > 0.000005 THEN
 
 110 LANGUAGE plpgsql IMMUTABLE;
 
 113 -- Guess a ranking for postcodes from country and postcode format.
 
 114 CREATE OR REPLACE FUNCTION get_postcode_rank(country_code VARCHAR(2), postcode TEXT,
 
 115                                              OUT rank_search SMALLINT,
 
 116                                              OUT rank_address SMALLINT)
 
 123     postcode := upper(postcode);
 
 125     IF country_code = 'gb' THEN
 
 126         IF postcode ~ '^([A-Z][A-Z]?[0-9][0-9A-Z]? [0-9][A-Z][A-Z])$' THEN
 
 129         ELSEIF postcode ~ '^([A-Z][A-Z]?[0-9][0-9A-Z]? [0-9])$' THEN
 
 132         ELSEIF postcode ~ '^([A-Z][A-Z]?[0-9][0-9A-Z])$' THEN
 
 137     ELSEIF country_code = 'sg' THEN
 
 138         IF postcode ~ '^([0-9]{6})$' THEN
 
 143     ELSEIF country_code = 'de' THEN
 
 144         IF postcode ~ '^([0-9]{5})$' THEN
 
 150         -- Guess at the postcode format and coverage (!)
 
 151         IF postcode ~ '^[A-Z0-9]{1,5}$' THEN -- Probably too short to be very local
 
 155             -- Does it look splitable into and area and local code?
 
 156             part := substring(postcode from '^([- :A-Z0-9]+)([- :][A-Z0-9]+)$');
 
 158             IF part IS NOT NULL THEN
 
 161             ELSEIF postcode ~ '^[- :A-Z0-9]{6,}$' THEN
 
 170 LANGUAGE plpgsql IMMUTABLE;
 
 173 -- Get standard search and address rank for an object.
 
 175 -- \param country        Two-letter country code where the object is in.
 
 176 -- \param extended_type  OSM type (N, W, R) or area type (A).
 
 177 -- \param place_class    Class (or tag key) of object.
 
 178 -- \param place_type     Type (or tag value) of object.
 
 179 -- \param admin_level    Value of admin_level tag.
 
 180 -- \param is_major       If true, boost search rank by one.
 
 181 -- \param postcode       Value of addr:postcode tag.
 
 182 -- \param[out] search_rank   Computed search rank.
 
 183 -- \param[out] address_rank  Computed address rank.
 
 185 CREATE OR REPLACE FUNCTION compute_place_rank(country VARCHAR(2),
 
 186                                               extended_type VARCHAR(1),
 
 187                                               place_class TEXT, place_type TEXT,
 
 188                                               admin_level SMALLINT,
 
 191                                               OUT search_rank SMALLINT,
 
 192                                               OUT address_rank SMALLINT)
 
 197   IF place_class in ('place','boundary')
 
 198      and place_type in ('postcode','postal_code')
 
 200     SELECT * INTO search_rank, address_rank
 
 201       FROM get_postcode_rank(country, postcode);
 
 202   ELSEIF extended_type = 'N' AND place_class = 'highway' THEN
 
 205   ELSEIF place_class = 'landuse' AND extended_type != 'A' THEN
 
 209     IF place_class = 'boundary' and place_type = 'administrative' THEN
 
 210       classtype = place_type || admin_level::TEXT;
 
 212       classtype = place_type;
 
 215     SELECT l.rank_search, l.rank_address INTO search_rank, address_rank
 
 216       FROM address_levels l
 
 217      WHERE (l.country_code = country or l.country_code is NULL)
 
 218            AND l.class = place_class AND (l.type = classtype or l.type is NULL)
 
 219      ORDER BY l.country_code, l.class, l.type LIMIT 1;
 
 221     IF search_rank is NULL OR address_rank is NULL THEN
 
 226     -- some postcorrections
 
 227     IF place_class = 'waterway' AND extended_type = 'R' THEN
 
 228         -- Slightly promote waterway relations so that they are processed
 
 229         -- before their members.
 
 230         search_rank := search_rank - 1;
 
 234       search_rank := search_rank - 1;
 
 239 LANGUAGE plpgsql IMMUTABLE;
 
 241 CREATE OR REPLACE FUNCTION get_addr_tag_rank(key TEXT, country TEXT,
 
 242                                              OUT from_rank SMALLINT,
 
 243                                              OUT to_rank SMALLINT,
 
 253       (SELECT l.rank_search, l.rank_address FROM address_levels l
 
 254         WHERE (l.country_code = country or l.country_code is NULL)
 
 255                AND l.class = 'place' AND l.type = key
 
 256         ORDER BY l.country_code LIMIT 1) r
 
 257       WHERE rank_address > 0
 
 259     extent := reverse_place_diameter(ranks.rank_search);
 
 261     IF ranks.rank_address <= 4 THEN
 
 264     ELSEIF ranks.rank_address <= 9 THEN
 
 267     ELSEIF ranks.rank_address <= 12 THEN
 
 270     ELSEIF ranks.rank_address <= 16 THEN
 
 273     ELSEIF ranks.rank_address <= 21 THEN
 
 276     ELSEIF ranks.rank_address <= 24 THEN
 
 286 LANGUAGE plpgsql IMMUTABLE;