From d92718afd5650b1052e04be823f2a11a3bae3287 Mon Sep 17 00:00:00 2001 From: Siarhei Fedartsou Date: Tue, 30 Aug 2022 21:08:52 +0200 Subject: [PATCH] Optimize RestrictionParser performance (#6344) --- CHANGELOG.md | 1 + include/extractor/restriction_parser.hpp | 7 +- src/extractor/restriction_parser.cpp | 152 ++++++++++++----------- 3 files changed, 83 insertions(+), 77 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1e5dba51137..8b6f8bb7a20 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,7 @@ - NodeJS: - FIXED: Support `skip_waypoints` in Node bindings [#6060](https://github.com/Project-OSRM/osrm-backend/pull/6060) - Misc: + - CHANGED: Optimize RestrictionParser performance. [#6344](https://github.com/Project-OSRM/osrm-backend/pull/6344) - ADDED: Support floats for speed value in traffic updates CSV. [#6327](https://github.com/Project-OSRM/osrm-backend/pull/6327) - CHANGED: Use Lua 5.4 in Docker image. [#6346](https://github.com/Project-OSRM/osrm-backend/pull/6346) - CHANGED: Remove redundant nullptr check. [#6326](https://github.com/Project-OSRM/osrm-backend/pull/6326) diff --git a/include/extractor/restriction_parser.hpp b/include/extractor/restriction_parser.hpp index 58539685726..869ff9b8404 100644 --- a/include/extractor/restriction_parser.hpp +++ b/include/extractor/restriction_parser.hpp @@ -5,6 +5,8 @@ #include +#include +#include #include #include @@ -43,7 +45,7 @@ class RestrictionParser public: RestrictionParser(bool use_turn_restrictions, bool parse_conditionals, - std::vector &restrictions); + const std::vector &restrictions); std::vector TryParse(const osmium::Relation &relation) const; private: @@ -51,7 +53,8 @@ class RestrictionParser bool use_turn_restrictions; bool parse_conditionals; - std::vector restrictions; + std::set restrictions; + osmium::tags::KeyFilter filter; }; } // namespace extractor } // namespace osrm diff --git a/src/extractor/restriction_parser.cpp b/src/extractor/restriction_parser.cpp index 2e4328acfe9..5daab9e7440 100644 --- a/src/extractor/restriction_parser.cpp +++ b/src/extractor/restriction_parser.cpp @@ -2,6 +2,7 @@ #include "extractor/profile_properties.hpp" #include "util/conditional_restrictions.hpp" +#include "util/integer_range.hpp" #include "util/log.hpp" #include @@ -9,7 +10,6 @@ #include #include -#include #include @@ -20,9 +20,9 @@ namespace extractor RestrictionParser::RestrictionParser(bool use_turn_restrictions_, bool parse_conditionals_, - std::vector &restrictions_) + const std::vector &restrictions_) : use_turn_restrictions(use_turn_restrictions_), parse_conditionals(parse_conditionals_), - restrictions(restrictions_) + restrictions(restrictions_.begin(), restrictions_.end()), filter(false) { if (use_turn_restrictions) { @@ -40,11 +40,28 @@ RestrictionParser::RestrictionParser(bool use_turn_restrictions_, util::Log() << "Found no turn restriction tags"; } } + + filter.add(true, "restriction"); + if (parse_conditionals) + { + filter.add(true, "restriction:conditional"); + for (const auto &namespaced : restrictions_) + { + filter.add(true, "restriction:" + namespaced + ":conditional"); + } + } + + // Not only use restriction= but also e.g. restriction:motorcar= + // Include restriction:{mode}:conditional if flagged + for (const auto &namespaced : restrictions_) + { + filter.add(true, "restriction:" + namespaced); + } } /** * Tries to parse a relation as a turn restriction. This can fail for a number of - * reasons. The return type is a boost::optional. + * reasons. The return type is a std::vector. * * Some restrictions can also be ignored: See the ```get_restrictions``` function * in the corresponding profile. We use it for both namespacing restrictions, as in @@ -59,31 +76,13 @@ RestrictionParser::TryParse(const osmium::Relation &relation) const return {}; } - osmium::tags::KeyFilter filter(false); - filter.add(true, "restriction"); - if (parse_conditionals) - { - filter.add(true, "restriction:conditional"); - for (const auto &namespaced : restrictions) - { - filter.add(true, "restriction:" + namespaced + ":conditional"); - } - } - - // Not only use restriction= but also e.g. restriction:motorcar= - // Include restriction:{mode}:conditional if flagged - for (const auto &namespaced : restrictions) - { - filter.add(true, "restriction:" + namespaced); - } - const osmium::TagList &tag_list = relation.tags(); osmium::tags::KeyFilter::iterator fi_begin(filter, tag_list.begin(), tag_list.end()); osmium::tags::KeyFilter::iterator fi_end(filter, tag_list.end(), tag_list.end()); // if it's not a restriction, continue; - if (std::distance(fi_begin, fi_end) == 0) + if (fi_begin == fi_end) { return {}; } @@ -99,18 +98,20 @@ RestrictionParser::TryParse(const osmium::Relation &relation) const bool is_multi_from = false; bool is_multi_to = false; + std::vector condition; + for (; fi_begin != fi_end; ++fi_begin) { - const std::string key(fi_begin->key()); - const std::string value(fi_begin->value()); + auto value = fi_begin->value(); // documented OSM restriction tags start either with only_* or no_*; // check and return on these values, and ignore no_*_on_red or unrecognized values - if (value.find("only_") == 0) + if (boost::algorithm::starts_with(value, "only_")) { is_only_restriction = true; } - else if (value.find("no_") == 0 && !boost::algorithm::ends_with(value, "_on_red")) + else if (boost::algorithm::starts_with(value, "no_") && + !boost::algorithm::ends_with(value, "_on_red")) { is_only_restriction = false; if (boost::algorithm::starts_with(value, "no_exit")) @@ -126,6 +127,25 @@ RestrictionParser::TryParse(const osmium::Relation &relation) const { return {}; } + + if (parse_conditionals) + { + // Parse condition and add independent value/condition pairs + const auto &parsed = osrm::util::ParseConditionalRestrictions(value); + + if (parsed.empty()) + continue; + + for (const auto &p : parsed) + { + std::vector hours = util::ParseOpeningHours(p.condition); + // found unrecognized condition, continue + if (hours.empty()) + return {}; + + condition = std::move(hours); + } + } } constexpr auto INVALID_OSM_ID = std::numeric_limits::max(); @@ -138,7 +158,11 @@ RestrictionParser::TryParse(const osmium::Relation &relation) const for (const auto &member : relation.members()) { const char *role = member.role(); - if (strcmp("from", role) != 0 && strcmp("to", role) != 0 && strcmp("via", role) != 0) + const bool is_from_role = strcmp("from", role) == 0; + const bool is_to_role = strcmp("to", role) == 0; + const bool is_via_role = strcmp("via", role) == 0; + + if (!is_from_role && !is_to_role && !is_via_role) { continue; } @@ -149,28 +173,27 @@ RestrictionParser::TryParse(const osmium::Relation &relation) const { // Make sure nodes appear only in the role if a via node - if (0 == strcmp("from", role) || 0 == strcmp("to", role)) + if (is_from_role || is_to_role) { continue; } - BOOST_ASSERT(0 == strcmp("via", role)); + BOOST_ASSERT(is_via_role); via_node = static_cast(member.ref()); is_node_restriction = true; // set via node id break; } case osmium::item_type::way: - BOOST_ASSERT(0 == strcmp("from", role) || 0 == strcmp("to", role) || - 0 == strcmp("via", role)); - if (0 == strcmp("from", role)) + BOOST_ASSERT(is_from_role || is_to_role || is_via_role); + if (is_from_role) { from_ways.push_back({static_cast(member.ref())}); } - else if (0 == strcmp("to", role)) + else if (is_to_role) { to_ways.push_back({static_cast(member.ref())}); } - else if (0 == strcmp("via", role)) + else if (is_via_role) { via_ways.push_back({static_cast(member.ref())}); is_node_restriction = false; @@ -185,35 +208,6 @@ RestrictionParser::TryParse(const osmium::Relation &relation) const } } - std::vector condition; - // parse conditional tags - if (parse_conditionals) - { - osmium::tags::KeyFilter::iterator fi_begin(filter, tag_list.begin(), tag_list.end()); - osmium::tags::KeyFilter::iterator fi_end(filter, tag_list.end(), tag_list.end()); - for (; fi_begin != fi_end; ++fi_begin) - { - const std::string key(fi_begin->key()); - const std::string value(fi_begin->value()); - - // Parse condition and add independent value/condition pairs - const auto &parsed = osrm::util::ParseConditionalRestrictions(value); - - if (parsed.empty()) - continue; - - for (const auto &p : parsed) - { - std::vector hours = util::ParseOpeningHours(p.condition); - // found unrecognized condition, continue - if (hours.empty()) - return {}; - - condition = std::move(hours); - } - } - } - std::vector restriction_containers; if (!from_ways.empty() && (via_node != INVALID_OSM_ID || !via_ways.empty()) && !to_ways.empty()) { @@ -270,17 +264,25 @@ bool RestrictionParser::ShouldIgnoreRestriction(const std::string &except_tag_st return false; } - // Be warned, this is quadratic work here, but we assume that - // only a few exceptions are actually defined. - const std::regex delimiter_re("[;][ ]*"); - std::sregex_token_iterator except_tags_begin( - except_tag_string.begin(), except_tag_string.end(), delimiter_re, -1); - std::sregex_token_iterator except_tags_end; - - return std::any_of(except_tags_begin, except_tags_end, [&](const std::string ¤t_string) { - return std::end(restrictions) != - std::find(std::begin(restrictions), std::end(restrictions), current_string); - }); + // split `except_tag_string` by semicolon and check if any of items is in `restrictions` + std::string current_string; + for (auto index : util::irange(0, except_tag_string.size())) + { + const auto ch = except_tag_string[index]; + if (ch != ';') + { + current_string += ch; + } + else + { + if (restrictions.find(current_string) != restrictions.end()) + { + return true; + } + current_string.clear(); + } + } + return restrictions.find(current_string) != restrictions.end(); } } // namespace extractor } // namespace osrm