X-Git-Url: https://git.openstreetmap.org./nominatim.git/blobdiff_plain/f5f0c197be28f8eb80e47eaf00dea12d25aa917e..da7218350b42f6470d575b88b9e6bb72b6e65ee5:/sql/functions/ranking.sql?ds=sidebyside diff --git a/sql/functions/ranking.sql b/sql/functions/ranking.sql index ecd31e90..98e70a42 100644 --- a/sql/functions/ranking.sql +++ b/sql/functions/ranking.sql @@ -114,3 +114,79 @@ BEGIN END; $$ LANGUAGE plpgsql IMMUTABLE; + + +-- Get standard search and address rank for an object. +-- +-- \param country Two-letter country code where the object is in. +-- \param extended_type OSM type (N, W, R) or area type (A). +-- \param place_class Class (or tag key) of object. +-- \param place_type Type (or tag value) of object. +-- \param admin_level Value of admin_level tag. +-- \param is_major If true, boost search rank by one. +-- \param postcode Value of addr:postcode tag. +-- \param[out] search_rank Computed search rank. +-- \param[out] address_rank Computed address rank. +-- +CREATE OR REPLACE FUNCTION compute_place_rank(country VARCHAR(2), + extended_type VARCHAR(1), + place_class TEXT, place_type TEXT, + admin_level SMALLINT, + is_major BOOLEAN, + postcode TEXT, + OUT search_rank SMALLINT, + OUT address_rank SMALLINT) +AS $$ +DECLARE + classtype TEXT; +BEGIN + IF place_class in ('place','boundary') + and place_type in ('postcode','postal_code') + THEN + SELECT * INTO search_rank, address_rank + FROM get_postcode_rank(country, postcode); + + IF NOT extended_type = 'A' THEN + address_rank := 0; + END IF; + ELSEIF extended_type = 'N' AND place_class = 'highway' THEN + search_rank = 30; + address_rank = 0; + ELSEIF place_class = 'landuse' AND extended_type != 'A' THEN + search_rank = 30; + address_rank = 0; + ELSE + IF place_class = 'boundary' and place_type = 'administrative' THEN + classtype = place_type || admin_level::TEXT; + ELSE + classtype = place_type; + END IF; + + SELECT l.rank_search, l.rank_address INTO search_rank, address_rank + FROM address_levels l + WHERE (l.country_code = country or l.country_code is NULL) + AND l.class = place_class AND (l.type = classtype or l.type is NULL) + ORDER BY l.country_code, l.class, l.type LIMIT 1; + + IF search_rank is NULL THEN + search_rank := 30; + END IF; + + IF address_rank is NULL THEN + address_rank := 30; + END IF; + + -- some postcorrections + IF place_class = 'waterway' AND extended_type = 'R' THEN + -- Slightly promote waterway relations so that they are processed + -- before their members. + search_rank := search_rank - 1; + END IF; + + IF is_major THEN + search_rank := search_rank - 1; + END IF; + END IF; +END; +$$ +LANGUAGE plpgsql IMMUTABLE;