[routing] Road category to speed conversion refactoring

This commit is contained in:
Vladimir Byko-Ianko 2018-10-22 17:56:56 +03:00 committed by mpimenov
parent dc2ce92d62
commit 78e9633742
12 changed files with 170 additions and 74 deletions

View file

@ -19,6 +19,8 @@ set(SRC
camera_info_collector.hpp
camera_node_processor.cpp
camera_node_processor.hpp
category_to_speed.cpp
category_to_speed.hpp
centers_table_builder.cpp
centers_table_builder.hpp
check_model.cpp

View file

@ -1,5 +1,9 @@
#include "generator/camera_node_processor.hpp"
#include "generator/category_to_speed.hpp"
#include "platform/measurement_utils.hpp"
#include "base/assert.hpp"
#include "base/control_flow.hpp"
#include "base/logging.hpp"
@ -8,74 +12,6 @@
namespace generator
{
// Look to https://wiki.openstreetmap.org/wiki/Speed_limits
// Remember about updating this table! Last update: 21.08.18
std::unordered_map<std::string, std::string> const kRoadTypeToSpeedKMpH = {
{"AT:urban", "50"},
{"AT:rural", "100"},
{"AT:trunk", "100"},
{"AT:motorway", "130"},
{"BE:urban", "50"},
{"BE:zone", "30"},
{"BE:motorway", "120"},
{"BE:zone30", "30"},
{"BE:rural", "70"},
{"BE:school", "30"},
{"CZ:motorway", "130"},
{"CZ:trunk", "110"},
{"CZ:rural", "90"},
{"CZ:urban_motorway", "80"},
{"CZ:urban_trunk", "80"},
{"CZ:urban", "50"},
{"DE:rural", "100"},
{"DE:urban", "50"},
{"DE:bicycle_road", "30"},
{"FR:motorway", "130"},
{"FR:rural", "80"},
{"FR:urban", "50"},
{"HU:living_street", "20"},
{"HU:motorway", "130"},
{"HU:rural", "90"},
{"HU:trunk", "110"},
{"HU:urban", "50"},
{"IT:rural", "90"},
{"IT:motorway", "130"},
{"IT:urban", "50"},
{"JP:nsl", "60"},
{"JP:express", "100"},
{"LT:rural", "90"},
{"LT:urban", "50"},
{"NO:rural", "80"},
{"NO:urban", "50"},
{"ON:urban", "50"},
{"ON:rural", "80"},
{"PT:motorway", "120"},
{"PT:rural", "90"},
{"PT:trunk", "100"},
{"PT:urban", "50"},
{"RO:motorway", "130"},
{"RO:rural", "90"},
{"RO:trunk", "100"},
{"RO:urban", "50"},
{"RU:living_street", "20"},
{"RU:urban", "60"},
{"RU:rural", "90"},
{"RU:motorway", "110"},
{"GB:motorway", "112"}, // 70 mph = 112.65408 kmph
{"GB:nsl_dual", "112"}, // 70 mph = 112.65408 kmph
{"GB:nsl_single", "96"}, // 60 mph = 96.56064 kmph
{"UK:motorway", "112"}, // 70 mph
{"UK:nsl_dual", "112"}, // 70 mph
{"UK:nsl_single", "96"}, // 60 mph
{"UZ:living_street", "30"},
{"UZ:urban", "70"},
{"UZ:rural", "100"},
{"UZ:motorway", "110"},
};
size_t const CameraNodeIntermediateDataProcessor::kMaxSpeedSpeedStringLength = 32;
} // namespace generator
@ -178,9 +114,10 @@ void CameraNodeIntermediateDataProcessor::ProcessWay(uint64_t id, WayElement con
std::string CameraNodeIntermediateDataProcessor::ValidateMaxSpeedString(std::string const & maxSpeedString)
{
auto const it = kRoadTypeToSpeedKMpH.find(maxSpeedString);
if (it != kRoadTypeToSpeedKMpH.cend())
return it->second;
double speed = 0.0;
measurement_utils::Units units = measurement_utils::Units::Metric;
if (RoadCategoryToSpeed(maxSpeedString, speed, units))
return strings::to_string(static_cast<int32_t>(measurement_utils::ToSpeedKmPH(speed, units)));
// strings::to_int doesn't work here because of bad errno.
std::string result;
@ -205,7 +142,7 @@ std::string CameraNodeIntermediateDataProcessor::ValidateMaxSpeedString(std::str
if (!strings::to_int(result.c_str(), mph))
return string();
auto const kmh = static_cast<int32_t>(routing::MilesPH2KMPH(mph));
auto const kmh = static_cast<int32_t>(measurement_utils::MphToKmph(mph));
return strings::to_string(kmh);
}

View file

@ -19,7 +19,6 @@
#include <memory>
#include <set>
#include <string>
#include <unordered_map>
#include <vector>
// TODO (@gmoryes) move members of m_routingTagsProcessor to generator

View file

@ -0,0 +1,97 @@
#include "generator/category_to_speed.hpp"
#include <unordered_map>
namespace
{
using namespace measurement_utils;
struct Speed
{
Speed() = delete;
uint16_t m_speed;
Units m_units;
};
std::unordered_map<std::string, Speed> const kRoadCategoryToSpeedKMpH = {
{"AT:urban", {50, Units::Metric}},
{"AT:rural", {100, Units::Metric}},
{"AT:trunk", {100, Units::Metric}},
{"AT:motorway", {130, Units::Metric}},
{"BE:urban", {50, Units::Metric}},
{"BE:zone", {30, Units::Metric}},
{"BE:motorway", {120, Units::Metric}},
{"BE:zone30", {30, Units::Metric}},
{"BE:rural", {70, Units::Metric}},
{"BE:school", {30, Units::Metric}},
{"CZ:motorway", {130, Units::Metric}},
{"CZ:trunk", {110, Units::Metric}},
{"CZ:rural", {90, Units::Metric}},
{"CZ:urban_motorway", {80, Units::Metric}},
{"CZ:urban_trunk", {80, Units::Metric}},
{"CZ:urban", {50, Units::Metric}},
{"DE:rural", {100, Units::Metric}},
{"DE:urban", {50, Units::Metric}},
{"DE:bicycle_road", {30, Units::Metric}},
{"FR:motorway", {130, Units::Metric}},
{"FR:rural", {80, Units::Metric}},
{"FR:urban", {50, Units::Metric}},
{"HU:living_street", {20, Units::Metric}},
{"HU:motorway", {130, Units::Metric}},
{"HU:rural", {90, Units::Metric}},
{"HU:trunk", {110, Units::Metric}},
{"HU:urban", {50, Units::Metric}},
{"IT:rural", {90, Units::Metric}},
{"IT:motorway", {130, Units::Metric}},
{"IT:urban", {50, Units::Metric}},
{"JP:nsl", {60, Units::Metric}},
{"JP:express", {100, Units::Metric}},
{"LT:rural", {90, Units::Metric}},
{"LT:urban", {50, Units::Metric}},
{"NO:rural", {80, Units::Metric}},
{"NO:urban", {50, Units::Metric}},
{"ON:urban", {50, Units::Metric}},
{"ON:rural", {80, Units::Metric}},
{"PT:motorway", {120, Units::Metric}},
{"PT:rural", {90, Units::Metric}},
{"PT:trunk", {100, Units::Metric}},
{"PT:urban", {50, Units::Metric}},
{"RO:motorway", {130, Units::Metric}},
{"RO:rural", {90, Units::Metric}},
{"RO:trunk", {100, Units::Metric}},
{"RO:urban", {50, Units::Metric}},
{"RU:living_street", {20, Units::Metric}},
{"RU:urban", {60, Units::Metric}},
{"RU:rural", {90, Units::Metric}},
{"RU:motorway", {110, Units::Metric}},
{"GB:motorway", {70, Units::Imperial}}, // 70 mph = 112.65408 kmph
{"GB:nsl_dual", {70, Units::Imperial}}, // 70 mph = 112.65408 kmph
{"GB:nsl_single", {60, Units::Imperial}}, // 60 mph = 96.56064 kmph
{"UK:motorway", {70, Units::Imperial}}, // 70 mph
{"UK:nsl_dual", {70, Units::Imperial}}, // 70 mph
{"UK:nsl_single", {60, Units::Imperial}}, // 60 mph
{"UZ:living_street", {30, Units::Metric}},
{"UZ:urban", {70, Units::Metric}},
{"UZ:rural", {100, Units::Metric}},
{"UZ:motorway", {110, Units::Metric}},
};
} // namespace
namespace generator
{
bool RoadCategoryToSpeed(std::string const & category, double & speed,
measurement_utils::Units & units)
{
auto const it = kRoadCategoryToSpeedKMpH.find(category);
if (it == kRoadCategoryToSpeedKMpH.cend())
return false;
speed = it->second.m_speed;
units = it->second.m_units;
return true;
}
} // generator

View file

@ -0,0 +1,18 @@
#pragma once
#include "platform/measurement_utils.hpp"
#include <string>
namespace generator
{
/// \brief Obtains |speed| and |units| by road category based on
/// the table in https://wiki.openstreetmap.org/wiki/Speed_limits
/// This method should be updated regularly. Last update: 22.10.18.
/// \note For example by passing string "RU:urban", you'll get
/// speed = 60 and units = Units::Metric.
/// \returns true if |speed| and |units| is found for |category| and these fields are filled
/// and false otherwise.
bool RoadCategoryToSpeed(std::string const & category, double & speed,
measurement_utils::Units & units);
} // generator

View file

@ -52,6 +52,16 @@ bool FormatDistanceImpl(double m, string & res,
return true;
}
double ToSpeedKmPH(double speed, measurement_utils::Units units)
{
switch (units)
{
case Units::Imperial: return MphToKmph(speed);
case Units::Metric: return speed;
}
CHECK_SWITCH();
}
bool FormatDistance(double m, string & res)
{
auto units = Units::Metric;

View file

@ -26,12 +26,16 @@ inline std::string DebugPrint(Units units)
inline double MetersToMiles(double m) { return m * 0.000621371192; }
inline double MilesToMeters(double mi) { return mi * 1609.344; }
inline double MphToKmph(double mph) { return MilesToMeters(mph) / 1000.0; }
inline double KmphToMph(double kmph) { return MetersToMiles(kmph) * 1000.0; }
inline double MetersToFeet(double m) { return m * 3.2808399; }
inline double FeetToMeters(double ft) { return ft * 0.3048; }
inline double FeetToMiles(double ft) { return ft * 5280; }
inline double InchesToMeters(double in) { return in / 39.370; }
inline double NauticalMilesToMeters(double nmi) { return nmi * 1852; }
double ToSpeedKmPH(double speed, measurement_utils::Units units);
/// Takes into an account user settings [metric, imperial]
/// @param[in] m meters
/// @param[out] res formatted std::string for search

View file

@ -3,6 +3,8 @@
#include "platform/measurement_utils.hpp"
#include "platform/settings.hpp"
#include "base/math.hpp"
#include "std/utility.hpp"
using namespace measurement_utils;
@ -191,3 +193,16 @@ UNIT_TEST(OSMDistanceToMetersString)
TEST_EQUAL(OSMDistanceToMetersString("15.123", false, 4), "15.123", ());
TEST_EQUAL(OSMDistanceToMetersString("15.654321", true, 1), "15.7", ());
}
UNIT_TEST(ConversionUnits)
{
double const kEps = 0.00001;
TEST(base::AlmostEqualAbs(MilesToMeters(MetersToMiles(1000.0)), 1000.0, kEps), ());
TEST(base::AlmostEqualAbs(MilesToMeters(1.0), 1609.344, kEps), ());
TEST(base::AlmostEqualAbs(MphToKmph(KmphToMph(100.0)), 100.0, kEps), ());
TEST(base::AlmostEqualAbs(MphToKmph(100.0), 160.934, kEps), ());
TEST(base::AlmostEqualAbs(ToSpeedKmPH(100.0, Units::Imperial), 160.934, kEps), ());
TEST(base::AlmostEqualAbs(ToSpeedKmPH(100.0, Units::Metric), 100.0, kEps), ());
}

View file

@ -63,6 +63,8 @@ set(
joint_index.cpp
joint_index.hpp
loaded_path_segment.hpp
maxspeed_conversion.cpp
maxspeed_conversion.hpp
maxspeed_serialization.cpp
maxspeed_serialization.hpp
nearest_edge_finder.cpp

View file

@ -0,0 +1,5 @@
#include "routing/maxspeed_conversion.hpp"
namespace routing
{
} // namespace routing

View file

@ -0,0 +1,8 @@
#pragma once
#include <string>
namespace routing
{
} // namespace routing

View file

@ -21,7 +21,6 @@
namespace routing
{
inline double KMPH2MPS(double kmph) { return kmph * 1000.0 / (60 * 60); }
inline double MilesPH2KMPH(double mph) { return mph / 0.621371192237334; }
template <typename Types>
bool IsCarRoad(Types const & types)