Merge pull request #5167 from bykoianko/master-restrictions-and-uturns

[routing] Dynamic restrictions and UTurns on invert graph.
This commit is contained in:
mpimenov 2017-01-30 13:58:41 +04:00 committed by GitHub
commit fec34814b1
27 changed files with 1595 additions and 81 deletions

View file

@ -5,7 +5,7 @@
#include "generator/osm_id.hpp"
#include "generator/restriction_collector.hpp"
#include "routing/routing_serialization.hpp"
#include "routing/restrictions_serialization.hpp"
#include "platform/platform_tests_support/scoped_dir.hpp"
#include "platform/platform_tests_support/scoped_file.hpp"

View file

@ -45,6 +45,28 @@ void BuildEmptyMwm(LocalCountryFile & country)
generator::tests_support::TestMwmBuilder builder(country, feature::DataHeader::country);
}
void LoadRestrictions(MwmValue const & mwmValue, RestrictionVec & restrictions)
{
if (!mwmValue.m_cont.IsExist(RESTRICTIONS_FILE_TAG))
return;
try
{
FilesContainerR::TReader const reader = mwmValue.m_cont.GetReader(RESTRICTIONS_FILE_TAG);
ReaderSource<FilesContainerR::TReader> src(reader);
RestrictionHeader header;
header.Deserialize(src);
RestrictionVec restrictionsOnly;
RestrictionSerializer::Deserialize(header, restrictions, restrictionsOnly, src);
restrictions.insert(restrictions.end(), restrictionsOnly.cbegin(), restrictionsOnly.cend());
}
catch (Reader::OpenException const & e)
{
LOG(LERROR, ("Error while reading", RESTRICTIONS_FILE_TAG, "section.", e.Msg()));
}
}
/// \brief Generates a restriction section, adds it to an empty mwm,
/// loads the restriction section and test loaded restrictions.
/// \param restrictionContent comma separated text with restrictions in osm id terms.
@ -83,10 +105,12 @@ void TestRestrictionBuilding(string const & restrictionContent, string const & m
MwmSet::MwmHandle mwmHandle = index.GetMwmHandleById(regResult.first);
TEST(mwmHandle.IsAlive(), ());
RestrictionLoader const restrictionLoader(*mwmHandle.GetValue<MwmValue>());
RestrictionVec restrictionsFromMwm;
LoadRestrictions(*mwmHandle.GetValue<MwmValue>(), restrictionsFromMwm);
RestrictionCollector const restrictionCollector(restrictionFullPath, mappingFullPath);
TEST_EQUAL(restrictionLoader.GetRestrictions(), restrictionCollector.GetRestrictions(), ());
TEST_EQUAL(restrictionsFromMwm, restrictionCollector.GetRestrictions(), ());
}
UNIT_TEST(RestrictionGenerationTest_NoRestriction)

View file

@ -1,6 +1,6 @@
#pragma once
#include "routing/routing_serialization.hpp"
#include "routing/restrictions_serialization.hpp"
#include "std/functional.hpp"
#include "std/limits.hpp"

View file

@ -33,9 +33,11 @@ bool BuildRoadRestrictions(string const & mwmPath, string const & restrictionPat
auto const firstOnlyIt =
lower_bound(restrictions.cbegin(), restrictions.cend(),
Restriction(Restriction::Type::Only, {} /* links */), my::LessBy(&Restriction::m_type));
RoutingHeader header;
RestrictionHeader header;
header.m_noRestrictionCount = base::checked_cast<uint32_t>(distance(restrictions.cbegin(), firstOnlyIt));
header.m_onlyRestrictionCount = base::checked_cast<uint32_t>(restrictions.size() - header.m_noRestrictionCount);
LOG(LINFO, ("Header info. There are", header.m_noRestrictionCount, "of type No restrictions and",
header.m_onlyRestrictionCount, "of type Only restrictions"));

View file

@ -4,7 +4,7 @@
#include "generator/osm_id.hpp"
#include "generator/restriction_collector.hpp"
#include "routing/routing_serialization.hpp"
#include "routing/restrictions_serialization.hpp"
#include "base/logging.hpp"

View file

@ -70,6 +70,8 @@ set(
pedestrian_model.hpp
restriction_loader.cpp
restriction_loader.hpp
restrictions_serialization.cpp
restrictions_serialization.hpp
road_graph.cpp
road_graph.hpp
road_graph_router.cpp
@ -92,8 +94,6 @@ set(
routing_mapping.cpp
routing_mapping.hpp
routing_result_graph.hpp
routing_serialization.cpp
routing_serialization.hpp
routing_session.cpp
routing_session.hpp
routing_settings.hpp

View file

@ -43,6 +43,7 @@ public:
void Finish() override;
double CalcSegmentWeight(Segment const & segment, RoadGeometry const & road) const override;
double CalcHeuristic(m2::PointD const & from, m2::PointD const & to) const override;
double GetUTurnPenalty() const override;
private:
TrafficCache const & m_trafficCache;
@ -101,6 +102,14 @@ double CarEdgeEstimator::CalcHeuristic(m2::PointD const & from, m2::PointD const
{
return TimeBetweenSec(from, to, m_maxSpeedMPS);
}
double CarEdgeEstimator::GetUTurnPenalty() const
{
// Adds 2 minutes penalty for U-turn. The value is quite arbitrary
// and needs to be properly selected after a number of real-world
// experiments.
return 2 * 60; // seconds
}
} // namespace
namespace routing

View file

@ -24,6 +24,7 @@ public:
virtual void Finish() = 0;
virtual double CalcSegmentWeight(Segment const & segment, RoadGeometry const & road) const = 0;
virtual double CalcHeuristic(m2::PointD const & from, m2::PointD const & to) const = 0;
virtual double GetUTurnPenalty() const = 0;
static shared_ptr<EdgeEstimator> CreateForCar(IVehicleModel const & vehicleModel,
traffic::TrafficCache const & trafficCache);

View file

@ -42,6 +42,12 @@ public:
// In such cases RoadGeometry is not valid.
bool IsValid() const { return m_valid; }
bool IsEndPointId(uint32_t pointId) const
{
ASSERT_LESS(pointId, m_points.size(), ());
return pointId == 0 || pointId + 1 == GetPointsCount();
}
private:
Points m_points;
double m_speed = 0.0;

View file

@ -1,9 +1,59 @@
#include "index_graph.hpp"
#include "routing/index_graph.hpp"
#include "routing/restrictions_serialization.hpp"
#include "base/assert.hpp"
#include "base/checked_cast.hpp"
#include "base/exception.hpp"
#include "std/limits.hpp"
#include <algorithm>
#include <limits>
namespace
{
using namespace base;
using namespace routing;
using namespace std;
bool IsUTurn(Segment const & u, Segment const & v)
{
return u.GetFeatureId() == v.GetFeatureId() && u.GetSegmentIdx() == v.GetSegmentIdx() &&
u.IsForward() != v.IsForward();
}
bool IsRestricted(RestrictionVec const & restrictions, Segment const & u, Segment const & v,
bool isOutgoing)
{
uint32_t const featureIdFrom = isOutgoing ? u.GetFeatureId() : v.GetFeatureId();
uint32_t const featureIdTo = isOutgoing ? v.GetFeatureId() : u.GetFeatureId();
if (!binary_search(restrictions.cbegin(), restrictions.cend(),
Restriction(Restriction::Type::No, {featureIdFrom, featureIdTo})))
{
return false;
}
if (featureIdFrom != featureIdTo)
return true;
// @TODO(bykoianko) According to current code if a feature id is marked as a feature with
// restrictricted U-turn it's restricted to make a U-turn on the both ends of the feature.
// Generally speaking it's wrong. In osm there's information about the end of the feature
// where the U-turn is restricted. It's necessary to pass the data to mwm and to use it here.
// Please see test LineGraph_RestrictionF1F1No for details.
//
// Another example when it's necessary to be aware about feature end participated in restriction
// is
// *---F1---*
// | |
// *--F3--A B--F4--*
// | |
// *---F2---*
// In case of restriction F1-A-F2 or F1-B-F2 of any type (No, Only) the important information
// is lost.
return IsUTurn(u, v);
}
} // namespace
namespace routing
{
@ -36,7 +86,13 @@ void IndexGraph::Import(vector<Joint> const & joints)
{
m_roadIndex.Import(joints);
CHECK_LESS_OR_EQUAL(joints.size(), numeric_limits<uint32_t>::max(), ());
Build(static_cast<uint32_t>(joints.size()));
Build(checked_cast<uint32_t>(joints.size()));
}
void IndexGraph::SetRestrictions(RestrictionVec && restrictions)
{
ASSERT(is_sorted(restrictions.cbegin(), restrictions.cend()), ());
m_restrictions = move(restrictions);
}
double IndexGraph::CalcSegmentWeight(Segment const & segment)
@ -70,17 +126,27 @@ void IndexGraph::GetNeighboringEdges(Segment const & from, RoadPoint const & rp,
void IndexGraph::GetNeighboringEdge(Segment const & from, Segment const & to, bool isOutgoing,
vector<SegmentEdge> & edges)
{
// Blocking U-turns on internal feature points.
RoadPoint const rp = from.GetRoadPoint(isOutgoing);
if (from.GetFeatureId() == to.GetFeatureId() && from.GetSegmentIdx() == to.GetSegmentIdx()
&& from.IsForward() != to.IsForward()
&& m_roadIndex.GetJointId(rp) == Joint::kInvalidId
&& rp.GetPointId() != 0
&& rp.GetPointId() + 1 != m_geometry.GetRoad(from.GetFeatureId()).GetPointsCount())
if (IsUTurn(from, to) && m_roadIndex.GetJointId(rp) == Joint::kInvalidId
&& !m_geometry.GetRoad(from.GetFeatureId()).IsEndPointId(rp.GetPointId()))
{
return;
}
double const weight = CalcSegmentWeight(isOutgoing ? to : from);
if (IsRestricted(m_restrictions, from, to, isOutgoing))
return;
double const weight = CalcSegmentWeight(isOutgoing ? to : from) +
GetPenalties(isOutgoing ? from : to, isOutgoing ? to : from);
edges.emplace_back(to, weight);
}
double IndexGraph::GetPenalties(Segment const & u, Segment const & v) const
{
if (IsUTurn(u, v))
return GetEstimator().GetUTurnPenalty();
return 0.0;
}
} // namespace routing

View file

@ -4,6 +4,7 @@
#include "routing/geometry.hpp"
#include "routing/joint.hpp"
#include "routing/joint_index.hpp"
#include "routing/restrictions_serialization.hpp"
#include "routing/road_index.hpp"
#include "routing/road_point.hpp"
#include "routing/segment.hpp"
@ -31,6 +32,7 @@ public:
Geometry & GetGeometry() { return m_geometry; }
EdgeEstimator const & GetEstimator() const { return *m_estimator; }
bool IsRoad(uint32_t featureId) const { return m_roadIndex.IsRoad(featureId); }
RoadJointIds const & GetRoad(uint32_t featureId) const { return m_roadIndex.GetRoad(featureId); }
uint32_t GetNumRoads() const { return m_roadIndex.GetSize(); }
@ -40,6 +42,8 @@ public:
void Build(uint32_t numJoints);
void Import(vector<Joint> const & joints);
void SetRestrictions(RestrictionVec && restrictions);
void PushFromSerializer(Joint::Id jointId, RoadPoint const & rp)
{
m_roadIndex.PushFromSerializer(jointId, rp);
@ -51,16 +55,24 @@ public:
m_roadIndex.ForEachRoad(forward<F>(f));
}
template <typename F>
void ForEachPoint(Joint::Id jointId, F && f) const
{
m_jointIndex.ForEachPoint(jointId, forward<F>(f));
}
private:
double CalcSegmentWeight(Segment const & segment);
void GetNeighboringEdges(Segment const & from, RoadPoint const & rp, bool isOutgoing,
vector<SegmentEdge> & edges);
void GetNeighboringEdge(Segment const & from, Segment const & to, bool isOutgoing,
vector<SegmentEdge> & edges);
double GetPenalties(Segment const & u, Segment const & v) const;
Geometry m_geometry;
shared_ptr<EdgeEstimator> m_estimator;
RoadIndex m_roadIndex;
JointIndex m_jointIndex;
RestrictionVec m_restrictions;
};
} // namespace routing

View file

@ -88,8 +88,9 @@ const Segment & IndexRoadGraph::GetSegment(Junction const & junction, bool isOut
{
auto const & junctionToSegment = isOutgoing ? m_endToSegment : m_beginToSegment;
auto it = junctionToSegment.find(junction);
CHECK(it != junctionToSegment.cend(), ("junctionToSegment doesn't contains", junction, ", isOutgoing =" + isOutgoing));
auto const it = junctionToSegment.find(junction);
CHECK(it != junctionToSegment.cend(),
("junctionToSegment doesn't contain", junction, ", isOutgoing =", isOutgoing));
return it->second;
}
} // namespace routing

View file

@ -1,9 +1,40 @@
#include "routing/restriction_loader.hpp"
#include "routing/routing_serialization.hpp"
#include "routing/restrictions_serialization.hpp"
#include "routing/road_index.hpp"
#include "base/stl_helpers.hpp"
namespace
{
using namespace routing;
/// \returns if features |r1| and |r2| have a common end returns its joint id.
/// If not, returns Joint::kInvalidId.
/// \note It's possible that the both ends of |r1| and |r2| have common joint ids.
/// In that case returns any of them.
/// \note In general case ends of features don't have to be joints. For example all
/// loose feature ends aren't joints. But if ends of r1 and r2 are connected at this
/// point there has to be a joint. So the method is valid.
Joint::Id GetCommonEndJoint(RoadJointIds const & r1, RoadJointIds const & r2)
{
auto const & j11 = r1.GetJointId(0 /* point id */);
auto const & j12 = r1.GetEndingJointId();
auto const & j21 = r2.GetJointId(0 /* point id */);
auto const & j22 = r2.GetEndingJointId();
if (j11 == j21 || j11 == j22)
return j11;
if (j12 == j21 || j12 == j22)
return j12;
return Joint::kInvalidId;
}
} // namespace
namespace routing
{
RestrictionLoader::RestrictionLoader(MwmValue const & mwmValue)
RestrictionLoader::RestrictionLoader(MwmValue const & mwmValue, IndexGraph const & graph)
: m_countryFileName(mwmValue.GetCountryFileName())
{
if (!mwmValue.m_cont.IsExist(RESTRICTIONS_FILE_TAG))
@ -15,13 +46,48 @@ RestrictionLoader::RestrictionLoader(MwmValue const & mwmValue)
ReaderSource<FilesContainerR::TReader> src(*m_reader);
m_header.Deserialize(src);
RestrictionSerializer::Deserialize(m_header, m_restrictions, src);
RestrictionVec restrictionsOnly;
RestrictionSerializer::Deserialize(m_header, m_restrictions /* restriction No */,
restrictionsOnly, src);
ConvertRestrictionsOnlyToNoAndSort(graph, restrictionsOnly, m_restrictions);
}
catch (Reader::OpenException const & e)
{
m_header.Reset();
LOG(LERROR,
("File", m_countryFileName, "Error while reading", RESTRICTIONS_FILE_TAG, "section.", e.Msg()));
throw;
}
}
void ConvertRestrictionsOnlyToNoAndSort(IndexGraph const & graph,
RestrictionVec const & restrictionsOnly,
RestrictionVec & restrictionsNo)
{
for (Restriction const & o : restrictionsOnly)
{
if (o.m_featureIds.size() != 2)
continue;
if (!graph.IsRoad(o.m_featureIds[0]) || !graph.IsRoad(o.m_featureIds[1]))
continue;
// Looking for a joint of an intersection of |o| features.
Joint::Id const common =
GetCommonEndJoint(graph.GetRoad(o.m_featureIds[0]), graph.GetRoad(o.m_featureIds[1]));
if (common == Joint::kInvalidId)
continue;
// Adding restriction of type No for all features of joint |common| except for
// the second feature of restriction |o|.
graph.ForEachPoint(common, [&](RoadPoint const & rp) {
if (rp.GetFeatureId() != o.m_featureIds[1 /* to */])
{
restrictionsNo.emplace_back(Restriction::Type::No,
vector<uint32_t>({o.m_featureIds[0 /* from */], rp.GetFeatureId()}));
}
});
}
my::SortUnique(restrictionsNo);
}
} // namespace routing

View file

@ -1,6 +1,7 @@
#pragma once
#include "routing/routing_serialization.hpp"
#include "routing/index_graph.hpp"
#include "routing/restrictions_serialization.hpp"
#include "indexer/index.hpp"
@ -14,15 +15,19 @@ namespace routing
class RestrictionLoader
{
public:
explicit RestrictionLoader(MwmValue const & mwmValue);
explicit RestrictionLoader(MwmValue const & mwmValue, IndexGraph const & graph);
bool HasRestrictions() const { return !m_restrictions.empty(); }
RestrictionVec const & GetRestrictions() const { return m_restrictions; }
RestrictionVec && StealRestrictions() { return move(m_restrictions); }
private:
unique_ptr<FilesContainerR::TReader> m_reader;
RoutingHeader m_header;
RestrictionHeader m_header;
RestrictionVec m_restrictions;
string const m_countryFileName;
};
void ConvertRestrictionsOnlyToNoAndSort(IndexGraph const & graph,
RestrictionVec const & restrictionsOnly,
RestrictionVec & restrictionsNo);
} // namespace routing

View file

@ -1,4 +1,4 @@
#include "routing/routing_serialization.hpp"
#include "routing/restrictions_serialization.hpp"
#include "std/sstream.hpp"
@ -15,8 +15,8 @@ uint32_t const Restriction::kInvalidFeatureId = numeric_limits<uint32_t>::max();
bool Restriction::IsValid() const
{
return !m_featureIds.empty() && find(begin(m_featureIds), end(m_featureIds), kInvalidFeatureId)
== end(m_featureIds);
return !m_featureIds.empty() &&
find(begin(m_featureIds), end(m_featureIds), kInvalidFeatureId) == end(m_featureIds);
}
bool Restriction::operator==(Restriction const & restriction) const

View file

@ -37,7 +37,6 @@ struct Restriction
};
Restriction(Type type, vector<uint32_t> const & links) : m_featureIds(links), m_type(type) {}
bool IsValid() const;
bool operator==(Restriction const & restriction) const;
bool operator<(Restriction const & restriction) const;
@ -53,9 +52,9 @@ string ToString(Restriction::Type const & type);
string DebugPrint(Restriction::Type const & type);
string DebugPrint(Restriction const & restriction);
struct RoutingHeader
struct RestrictionHeader
{
RoutingHeader() { Reset(); }
RestrictionHeader() { Reset(); }
template <class Sink>
void Serialize(Sink & sink) const
@ -89,13 +88,13 @@ struct RoutingHeader
uint32_t m_onlyRestrictionCount;
};
static_assert(sizeof(RoutingHeader) == 12, "Wrong header size of routing section.");
static_assert(sizeof(RestrictionHeader) == 12, "Wrong header size of routing section.");
class RestrictionSerializer
{
public:
template <class Sink>
static void Serialize(RoutingHeader const & header,
static void Serialize(RestrictionHeader const & header,
routing::RestrictionVec::const_iterator begin,
routing::RestrictionVec::const_iterator end, Sink & sink)
{
@ -105,13 +104,14 @@ public:
}
template <class Source>
static void Deserialize(RoutingHeader const & header, routing::RestrictionVec & restrictions,
Source & src)
static void Deserialize(RestrictionHeader const & header,
routing::RestrictionVec & restrictionsNo,
routing::RestrictionVec & restrictionsOnly, Source & src)
{
DeserializeSingleType(routing::Restriction::Type::No, header.m_noRestrictionCount,
restrictions, src);
restrictionsNo, src);
DeserializeSingleType(routing::Restriction::Type::Only, header.m_onlyRestrictionCount,
restrictions, src);
restrictionsOnly, src);
}
private:
@ -139,17 +139,20 @@ private:
routing::Restriction const & restriction = *begin;
CHECK(restriction.IsValid(), ());
CHECK_LESS(1, restriction.m_featureIds.size(), ("No sense in zero or one link restrictions."));
CHECK_LESS(1, restriction.m_featureIds.size(),
("No sense in zero or one link restrictions."));
coding::DeltaCoder::Encode(bits, restriction.m_featureIds.size() - 1 /* number of link is two or more */);
coding::DeltaCoder::Encode(
bits, restriction.m_featureIds.size() - 1 /* number of link is two or more */);
CHECK_LESS_OR_EQUAL(prevFirstLinkFeatureId, restriction.m_featureIds[0], ());
coding::DeltaCoder::Encode(bits,
restriction.m_featureIds[0] - prevFirstLinkFeatureId + 1 /* making it greater than zero */);
coding::DeltaCoder::Encode(bits, restriction.m_featureIds[0] - prevFirstLinkFeatureId +
1 /* making it greater than zero */);
for (size_t i = 1; i < restriction.m_featureIds.size(); ++i)
{
uint32_t const delta = bits::ZigZagEncode(static_cast<int32_t>(restriction.m_featureIds[i]) -
static_cast<int32_t>(restriction.m_featureIds[i - 1]));
uint32_t const delta =
bits::ZigZagEncode(static_cast<int32_t>(restriction.m_featureIds[i]) -
static_cast<int32_t>(restriction.m_featureIds[i - 1]));
coding::DeltaCoder::Encode(bits, delta + 1 /* making it greater than zero */);
}
prevFirstLinkFeatureId = restriction.m_featureIds[0];
@ -172,7 +175,7 @@ private:
}
size_t const numLinks = biasedLinkNumber + 1 /* number of link is two or more */;
routing::Restriction restriction(type, {} /* links */);
routing::Restriction restriction(type, {} /* links */);
restriction.m_featureIds.resize(numLinks);
auto const biasedFirstFeatureId = coding::DeltaCoder::Decode(bits);
if (biasedFirstFeatureId == 0)
@ -180,7 +183,8 @@ private:
LOG(LERROR, ("Decoded first link restriction feature id delta is zero."));
return false;
}
restriction.m_featureIds[0] = prevFirstLinkFeatureId + base::checked_cast<uint32_t>(biasedFirstFeatureId) - 1;
restriction.m_featureIds[0] =
prevFirstLinkFeatureId + base::checked_cast<uint32_t>(biasedFirstFeatureId) - 1;
for (size_t i = 1; i < numLinks; ++i)
{
auto const biasedDelta = coding::DeltaCoder::Decode(bits);
@ -189,9 +193,10 @@ private:
LOG(LERROR, ("Decoded link restriction feature id delta is zero."));
return false;
}
auto const delta = biasedDelta - 1;
restriction.m_featureIds[i] = static_cast<uint32_t>(
bits::ZigZagDecode(delta) + restriction.m_featureIds[i - 1]);
uint32_t const delta = biasedDelta - 1;
restriction.m_featureIds[i] =
static_cast<uint32_t>(bits::ZigZagDecode(delta) + restriction.m_featureIds[i - 1]);
}
prevFirstLinkFeatureId = restriction.m_featureIds[0];

View file

@ -29,6 +29,15 @@ public:
return Joint::kInvalidId;
}
Joint::Id GetEndingJointId() const
{
if (m_jointIds.empty())
return Joint::kInvalidId;
ASSERT_NOT_EQUAL(m_jointIds.back(), Joint::kInvalidId, ());
return m_jointIds.back();
}
void AddJoint(uint32_t pointId, Joint::Id jointId)
{
ASSERT_NOT_EQUAL(jointId, Joint::kInvalidId, ());
@ -113,10 +122,12 @@ public:
m_roads[rp.GetFeatureId()].AddJoint(rp.GetPointId(), jointId);
}
bool IsRoad(uint32_t featureId) const { return m_roads.count(featureId) != 0; }
RoadJointIds const & GetRoad(uint32_t featureId) const
{
auto const & it = m_roads.find(featureId);
CHECK(it != m_roads.cend(), ());
CHECK(it != m_roads.cend(), ("Feature id:", featureId));
return it->second;
}

View file

@ -42,6 +42,7 @@ SOURCES += \
pedestrian_directions.cpp \
pedestrian_model.cpp \
restriction_loader.cpp \
restrictions_serialization.cpp \
road_graph.cpp \
road_graph_router.cpp \
road_index.cpp \
@ -51,7 +52,6 @@ SOURCES += \
routing_algorithm.cpp \
routing_helpers.cpp \
routing_mapping.cpp \
routing_serialization.cpp \
routing_session.cpp \
single_mwm_router.cpp \
speed_camera.cpp \
@ -96,6 +96,7 @@ HEADERS += \
pedestrian_directions.hpp \
pedestrian_model.hpp \
restriction_loader.hpp \
restrictions_serialization.hpp \
road_graph.hpp \
road_graph_router.hpp \
road_index.hpp \
@ -109,7 +110,6 @@ HEADERS += \
routing_helpers.hpp \
routing_mapping.hpp \
routing_result_graph.hpp \
routing_serialization.hpp \
routing_session.hpp \
routing_settings.hpp \
segment.hpp \

View file

@ -9,10 +9,14 @@ set(
astar_router_test.cpp
async_router_test.cpp
cross_routing_tests.cpp
cumulative_restriction_test.cpp
followed_polyline_test.cpp
index_graph_tools.cpp
index_graph_tools.hpp
nearest_edge_finder_tests.cpp
online_cross_fetcher_test.cpp
osrm_router_test.cpp
restriction_test.cpp
road_graph_builder.cpp
road_graph_builder.hpp
road_graph_nearest_edges_test.cpp

View file

@ -0,0 +1,283 @@
#include "testing/testing.hpp"
#include "routing/routing_tests/index_graph_tools.hpp"
#include "routing/car_model.hpp"
#include "routing/geometry.hpp"
#include "traffic/traffic_cache.hpp"
#include "geometry/point2d.hpp"
#include <memory>
#include <vector>
namespace
{
using namespace routing;
using namespace routing_test;
// Finish
// 3 *
// ^
// F5
// |
// 2 * *
// ↖ ↗ ↖
// F2 F3 F4
// ↖ ↗ ↖
// 1 * *
// ↗ ↖
// F0 F1
// ↗ ↖
// 0 * *
// 0 1 2 3
// Start
// Note. This graph contains of 6 one segment directed features.
unique_ptr<IndexGraph> BuildXYGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(1 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(2 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {0.0, 2.0}}));
loader->AddRoad(3 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {2.0, 2.0}}));
loader->AddRoad(4 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 1.0}, {2.0, 2.0}}));
loader->AddRoad(5 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 2.0}, {2.0, 3.0}}));
vector<Joint> const joints = {
MakeJoint({{0 /* feature id */, 0 /* point id */}}), /* joint at point (0, 0) */
MakeJoint({{1, 0}}), /* joint at point (2, 0) */
MakeJoint({{0, 1}, {1, 1}, {2, 0}, {3, 0}}), /* joint at point (1, 1) */
MakeJoint({{2, 1}}), /* joint at point (0, 2) */
MakeJoint({{3, 1}, {4, 1}, {5, 0}}), /* joint at point (2, 2) */
MakeJoint({{4, 0}}), /* joint at point (3, 1) */
MakeJoint({{5, 1}}), /* joint at point (2, 3) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// Route through XY graph without any restrictions.
UNIT_TEST(XYGraph)
{
unique_ptr<IndexGraph> graph = BuildXYGraph();
IndexGraphStarter starter(*graph,
IndexGraphStarter::FakeVertex(1, 0, m2::PointD(2, 0)) /* start */,
IndexGraphStarter::FakeVertex(5, 0, m2::PointD(2, 3)) /* finish */);
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 1}, {2, 2}, {2, 3}};
TestRouteGeometry(starter, AStarAlgorithm<IndexGraphStarter>::Result::OK, expectedGeom);
}
// Route through XY graph with one restriciton (type only) from F1 to F3.
UNIT_CLASS_TEST(RestrictionTest, XYGraph_RestrictionF1F3Only)
{
Init(BuildXYGraph());
RestrictionVec restrictions = {
{Restriction::Type::Only, {1 /* feature from */, 3 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 1}, {2, 2}, {2, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(1, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(2, 3)) /* finish */,
move(restrictions), *this);
}
// Route through XY graph with one restriciton (type only) from F3 to F5.
UNIT_CLASS_TEST(RestrictionTest, XYGraph_RestrictionF3F5Only)
{
Init(BuildXYGraph());
RestrictionVec restrictions = {
{Restriction::Type::Only, {3 /* feature from */, 5 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 1}, {2, 2}, {2, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(1, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(2, 3)) /* finish */,
move(restrictions), *this);
}
// Cumulative case. Route through XY graph with two restricitons (type only) applying
// in all possible orders.
UNIT_CLASS_TEST(RestrictionTest, XYGraph_PermutationsF3F5OnlyF1F3Only)
{
Init(BuildXYGraph());
RestrictionVec restrictions = {
{Restriction::Type::Only, {1 /* feature from */, 3 /* feature to */}},
{Restriction::Type::Only, {3 /* feature from */, 5 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 1}, {2, 2}, {2, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(1, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(2, 3)) /* finish */,
move(restrictions), *this);
}
// Cumulative case. Route through XY graph with two restricitons (type only and type no) applying
// in all possible orders.
UNIT_CLASS_TEST(RestrictionTest, XYGraph_PermutationsF3F5OnlyAndF0F2No)
{
Init(BuildXYGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {1 /* feature from */, 2 /* feature to */}},
{Restriction::Type::Only, {3 /* feature from */, 5 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 1}, {2, 2}, {2, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(1, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(2, 3)) /* finish */,
move(restrictions), *this);
}
// Cumulative case. Trying to build route through XY graph with two restricitons applying
// according to the order. First from F3 to F5 (type only)
// and then and from F1 to F3 (type no).
UNIT_CLASS_TEST(RestrictionTest, XYGraph_RestrictionF3F5OnlyAndF1F3No)
{
Init(BuildXYGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {1 /* feature from */, 3 /* feature to */}},
{Restriction::Type::Only, {3 /* feature from */, 5 /* feature to */}}};
TestRestrictions({} /* expectedGeom */, AStarAlgorithm<IndexGraphStarter>::Result::NoPath,
routing::IndexGraphStarter::FakeVertex(1, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(2, 3)) /* finish */,
move(restrictions), *this);
}
// Finish
// 3 * *
// ↖ ↗
// F5 F6
// ↖ ↗
// 2 * *
// ↖ ↗ ↖
// F2 F3 F4
// ↖ ↗ ↖
// 1 * *
// ↗ ↖ ^
// F0 F1 F8
// ↗ ↖ |
// 0 * *--F7--->*
// ^
// F9
// |
//-1 *
// 0 1 2 3
// Start
// Note. This graph contains of 9 one segment directed features.
unique_ptr<IndexGraph> BuildXXGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(1 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(2 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {0.0, 2.0}}));
loader->AddRoad(3 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {2.0, 2.0}}));
loader->AddRoad(4 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 1.0}, {2.0, 2.0}}));
loader->AddRoad(5 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 2.0}, {1.0, 3.0}}));
loader->AddRoad(6 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 2.0}, {3.0, 3.0}}));
loader->AddRoad(7 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {3.0, 0.0}}));
loader->AddRoad(8 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 0.0}, {3.0, 1.0}}));
loader->AddRoad(9 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, -1.0}, {2.0, 0.0}}));
vector<Joint> const joints = {
MakeJoint({{0 /* feature id */, 0 /* point id */}}), /* joint at point (0, 0) */
MakeJoint({{1, 0}, {7, 0}, {9, 1}}), /* joint at point (2, 0) */
MakeJoint({{0, 1}, {1, 1}, {2, 0}, {3, 0}}), /* joint at point (1, 1) */
MakeJoint({{2, 1}}), /* joint at point (0, 2) */
MakeJoint({{3, 1}, {4, 1}, {5, 0}, {6, 0}}), /* joint at point (2, 2) */
MakeJoint({{4, 0}, {8, 1}}), /* joint at point (3, 1) */
MakeJoint({{5, 1}}), /* joint at point (1, 3) */
MakeJoint({{6, 1}}), /* joint at point (3, 3) */
MakeJoint({{7, 1}, {8, 0}}), /* joint at point (3, 0) */
MakeJoint({{9, 0}}), /* joint at point (2, -1) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// Route through XY graph without any restrictions.
UNIT_CLASS_TEST(RestrictionTest, XXGraph)
{
Init(BuildXXGraph());
RestrictionVec restrictions = {};
vector<m2::PointD> const expectedGeom = {{2 /* x */, -1 /* y */}, {2, 0}, {1, 1}, {2, 2}, {3, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(9, 0, m2::PointD(2, -1)), /* start */
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(3, 3)), /* finish */
move(restrictions), *this);
}
// Cumulative case. Route through XX graph with two restricitons (type only) applying
// in all possible orders.
UNIT_CLASS_TEST(RestrictionTest, XXGraph_PermutationsF1F3OnlyAndF3F6Only)
{
Init(BuildXXGraph());
RestrictionVec restrictions = {
{Restriction::Type::Only, {1 /* feature from */, 3 /* feature to */}},
{Restriction::Type::Only, {3 /* feature from */, 6 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{2 /* x */, -1 /* y */}, {2, 0}, {1, 1}, {2, 2}, {3, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(9, 0, m2::PointD(2, -1)), /* start */
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(3, 3)), /* finish */
move(restrictions), *this);
}
// Route through XX graph with one restriciton (type no) from F1 to F3.
UNIT_CLASS_TEST(RestrictionTest, XXGraph_RestrictionF1F3No)
{
Init(BuildXXGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {1 /* feature from */, 3 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{2 /* x */, -1 /* y */}, {2, 0}, {3, 0}, {3, 1}, {2, 2}, {3, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(9, 0, m2::PointD(2, -1)), /* start */
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(3, 3)), /* finish */
move(restrictions), *this);
}
// Cumulative case. Route through XX graph with four restricitons of different types applying
// in all possible orders.
UNIT_CLASS_TEST(RestrictionTest, XXGraph_PermutationsF1F3NoF7F8OnlyF8F4OnlyF4F6Only)
{
Init(BuildXXGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {1 /* feature from */, 3 /* feature to */}},
{Restriction::Type::Only, {4 /* feature from */, 6 /* feature to */}},
{Restriction::Type::Only, {7 /* feature from */, 8 /* feature to */}},
{Restriction::Type::Only, {8 /* feature from */, 4 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{2 /* x */, -1 /* y */}, {2, 0}, {3, 0}, {3, 1}, {2, 2}, {3, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(9, 0, m2::PointD(2, -1)), /* start */
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(3, 3)), /* finish */
move(restrictions), *this);
}
} // namespace

View file

@ -31,12 +31,10 @@ void TestRoute(IndexGraph & graph, IndexGraphStarter::FakeVertex const & start,
IndexGraphStarter::FakeVertex const & finish, size_t expectedLength,
vector<Segment> const * expectedRoute = nullptr)
{
LOG(LINFO, ("Test route", start.GetFeatureId(), ",", start.GetSegmentIdx(), "=>",
finish.GetFeatureId(), ",", finish.GetSegmentIdx()));
IndexGraphStarter starter(graph, start, finish);
vector<Segment> route;
auto const resultCode = CalculateRoute(starter, route);
double timeSec;
auto const resultCode = CalculateRoute(starter, route, timeSec);
TEST_EQUAL(resultCode, AStarAlgorithm<IndexGraphStarter>::Result::OK, ());
TEST_GREATER(route.size(), 2, ());
@ -386,4 +384,79 @@ UNIT_TEST(SerializeSimpleGraph)
TEST_EQUAL(graph.GetJointId({2, 1}), Joint::kInvalidId, ());
}
}
// Finish
// 0.0004 *
// ^
// |
// F2
// |
// |
// 0.0003 *---------*
// | |
// | |
// | |
// | |
// | |
// 0.0002 * *
// | |
// | |
// 0.00015 * F0 *
// \ /
// \ /
// 0.0001 *---F0-*----*
// ^
// |
// F1
// |
// |
// 0 * Start
// 0 0.0001 0.0002
// F0 is a two-way feature with a loop and F1 and F2 are an one-way one-segment features.
unique_ptr<IndexGraph> BuildLoopGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, false /* one way */, 100.0 /* speed */,
RoadGeometry::Points({{0.0002, 0.0001},
{0.00015, 0.0001},
{0.0001, 0.0001},
{0.00015, 0.00015},
{0.00015, 0.0002},
{0.00015, 0.0003},
{0.00005, 0.0003},
{0.00005, 0.0002},
{0.00005, 0.00015},
{0.0001, 0.0001}}));
loader->AddRoad(1 /* feature id */, true /* one way */, 100.0 /* speed */,
RoadGeometry::Points({{0.0002, 0.0}, {0.0002, 0.0001}}));
loader->AddRoad(2 /* feature id */, true /* one way */, 100.0 /* speed */,
RoadGeometry::Points({{0.00005, 0.0003}, {0.00005, 0.0004}}));
vector<Joint> const joints = {
MakeJoint({{0 /* feature id */, 2 /* point id */}, {0, 9}}), /* joint at point (0.0002, 0) */
MakeJoint({{1, 1}, {0, 0}}), /* joint at point (0.0002, 0.0001) */
MakeJoint({{0, 6}, {2, 0}}), /* joint at point (0.00005, 0.0003) */
MakeJoint({{2, 1}}), /* joint at point (0.00005, 0.0004) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// This test checks that the route from Start to Finish doesn't make an extra loop in F0.
// If it was so the route time had been much more.
UNIT_CLASS_TEST(RestrictionTest, LoopGraph)
{
Init(BuildLoopGraph());
SetStarter(
routing::IndexGraphStarter::FakeVertex(1, 0 /* seg id */, m2::PointD(0.0002, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(2, 0 /* seg id */,
m2::PointD(0.00005, 0.0004)) /* finish */);
double constexpr kExpectedRouteTimeSec = 3.48;
TestRouteTime(*m_starter, AStarAlgorithm<IndexGraphStarter>::Result::OK, kExpectedRouteTimeSec);
}
} // namespace routing_test

View file

@ -41,13 +41,16 @@ shared_ptr<EdgeEstimator> CreateEstimator(traffic::TrafficCache const & trafficC
}
AStarAlgorithm<IndexGraphStarter>::Result CalculateRoute(IndexGraphStarter & starter,
vector<Segment> & roadPoints)
vector<Segment> & roadPoints,
double & timeSec)
{
AStarAlgorithm<IndexGraphStarter> algorithm;
RoutingResult<Segment> routingResult;
auto const resultCode = algorithm.FindPathBidirectional(
starter, starter.GetStart(), starter.GetFinish(), routingResult, {}, {});
timeSec = routingResult.distance;
roadPoints = routingResult.path;
return resultCode;
}
@ -56,12 +59,63 @@ void TestRouteGeometry(IndexGraphStarter & starter,
AStarAlgorithm<IndexGraphStarter>::Result expectedRouteResult,
vector<m2::PointD> const & expectedRouteGeom)
{
vector<Segment> route;
auto const resultCode = CalculateRoute(starter, route);
TEST_EQUAL(resultCode, expectedRouteResult, ());
TEST_EQUAL(IndexGraphStarter::GetRouteNumPoints(route), expectedRouteGeom.size(), ());
vector<Segment> routeSegs;
double timeSec = 0.0;
auto const resultCode = CalculateRoute(starter, routeSegs, timeSec);
for (size_t i = 0; i < IndexGraphStarter::GetRouteNumPoints(route); ++i)
TEST_EQUAL(expectedRouteGeom[i], starter.GetRoutePoint(route, i), ());
if (AStarAlgorithm<IndexGraphStarter>::Result::NoPath == expectedRouteResult &&
expectedRouteGeom == vector<m2::PointD>())
{
// The route goes through a restriction. So there's no choice for building route
// except for going through restriction. So no path.
return;
}
TEST_EQUAL(resultCode, expectedRouteResult, ());
if (resultCode != AStarAlgorithm<IndexGraphStarter>::Result::OK)
return;
CHECK(!routeSegs.empty(), ());
vector<m2::PointD> geom;
auto const pushPoint = [&geom](m2::PointD const & point) {
if (geom.empty() || geom.back() != point)
geom.push_back(point);
};
for (size_t i = 0; i < routeSegs.size(); ++i)
{
m2::PointD const & pnt = starter.GetPoint(routeSegs[i], false /* front */);
// Note. In case of A* router all internal points of route are duplicated.
// So it's necessary to exclude the duplicates.
pushPoint(pnt);
}
pushPoint(starter.GetPoint(routeSegs.back(), false /* front */));
TEST_EQUAL(geom, expectedRouteGeom, ());
}
void TestRouteTime(IndexGraphStarter & starter,
AStarAlgorithm<IndexGraphStarter>::Result expectedRouteResult,
double expectedTime)
{
vector<Segment> routeSegs;
double timeSec = 0.0;
auto const resultCode = CalculateRoute(starter, routeSegs, timeSec);
TEST_EQUAL(resultCode, expectedRouteResult, ());
double const kEpsilon = 1e-5;
TEST(my::AlmostEqualAbs(timeSec, expectedTime, kEpsilon), ());
}
void TestRestrictions(vector<m2::PointD> const & expectedRouteGeom,
AStarAlgorithm<IndexGraphStarter>::Result expectedRouteResult,
routing::IndexGraphStarter::FakeVertex const & start,
routing::IndexGraphStarter::FakeVertex const & finish,
RestrictionVec && restrictions, RestrictionTest & restrictionTest)
{
restrictionTest.SetRestrictions(move(restrictions));
restrictionTest.SetStarter(start, finish);
TestRouteGeometry(*restrictionTest.m_starter, expectedRouteResult, expectedRouteGeom);
}
} // namespace routing_test

View file

@ -3,6 +3,7 @@
#include "routing/edge_estimator.hpp"
#include "routing/index_graph.hpp"
#include "routing/index_graph_starter.hpp"
#include "routing/restrictions_serialization.hpp"
#include "routing/road_point.hpp"
#include "routing/segment.hpp"
@ -10,6 +11,8 @@
#include "traffic/traffic_info.hpp"
#include "indexer/classificator_loader.hpp"
#include "geometry/point2d.hpp"
#include "std/algorithm.hpp"
@ -20,6 +23,27 @@
namespace routing_test
{
using namespace routing;
struct RestrictionTest
{
RestrictionTest() { classificator::Load(); }
void Init(unique_ptr<IndexGraph> graph) { m_graph = move(graph); }
void SetStarter(IndexGraphStarter::FakeVertex const & start,
IndexGraphStarter::FakeVertex const & finish)
{
m_starter = make_unique<IndexGraphStarter>(*m_graph, start, finish);
}
void SetRestrictions(RestrictionVec && restrictions)
{
m_graph->SetRestrictions(move(restrictions));
}
unique_ptr<IndexGraph> m_graph;
unique_ptr<IndexGraphStarter> m_starter;
};
class TestGeometryLoader final : public routing::GeometryLoader
{
public:
@ -38,15 +62,23 @@ routing::Joint MakeJoint(vector<routing::RoadPoint> const & points);
shared_ptr<routing::EdgeEstimator> CreateEstimator(traffic::TrafficCache const & trafficCache);
routing::AStarAlgorithm<routing::IndexGraphStarter>::Result CalculateRoute(
routing::IndexGraphStarter & starter, vector<routing::Segment> & roadPoints);
void TestRouteSegments(
routing::IndexGraphStarter & starter,
routing::AStarAlgorithm<routing::IndexGraphStarter>::Result expectedRouteResult,
vector<routing::RoadPoint> const & expectedRoute);
routing::IndexGraphStarter & starter, vector<routing::Segment> & roadPoints, double & timeSec);
void TestRouteGeometry(
routing::IndexGraphStarter & starter,
routing::AStarAlgorithm<routing::IndexGraphStarter>::Result expectedRouteResult,
vector<m2::PointD> const & expectedRouteGeom);
void TestRouteTime(IndexGraphStarter & starter,
AStarAlgorithm<IndexGraphStarter>::Result expectedRouteResult,
double expectedTime);
/// \brief Applies |restrictions| to graph in |restrictionTest| and
/// tests the resulting route.
/// \note restrictionTest should have a valid |restrictionTest.m_graph|.
void TestRestrictions(vector<m2::PointD> const & expectedRouteGeom,
AStarAlgorithm<IndexGraphStarter>::Result expectedRouteResult,
routing::IndexGraphStarter::FakeVertex const & start,
routing::IndexGraphStarter::FakeVertex const & finish,
RestrictionVec && restrictions, RestrictionTest & restrictionTest);
} // namespace routing_test

View file

@ -0,0 +1,838 @@
#include "testing/testing.hpp"
#include "routing/routing_tests/index_graph_tools.hpp"
#include "routing/car_model.hpp"
#include "routing/geometry.hpp"
#include "routing/restriction_loader.hpp"
#include "indexer/classificator_loader.hpp"
#include "geometry/point2d.hpp"
#include <memory>
#include <vector>
namespace routing_test
{
using namespace routing;
// Finish
// *
// ^
// |
// F7
// |
// *
// ^
// |
// F6
// |
// Start * -- F0 --> * -- F1 --> * <-- F2 --> * -- F3 --> *
// | ^
// | |
// F4 F5
// | |
// ⌄ |
// *
unique_ptr<IndexGraph> BuildCrossGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{-1.0, 0.0}, {0.0, 0.0}}));
loader->AddRoad(1 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {1.0, 0.0}}));
loader->AddRoad(2 /* featureId */, false /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {1.9999, 0.0}}));
loader->AddRoad(3 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.9999, 0.0}, {3.0, 0.0}}));
loader->AddRoad(4 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {1.0, -1.0}}));
loader->AddRoad(5 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, -1.0}, {1.0, 0.0}}));
loader->AddRoad(6 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(7 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {1.0, 2.0}}));
vector<Joint> const joints = {
MakeJoint({{0, 1}, {1, 0}}), MakeJoint({{1, 1}, {2, 0}, {4, 0}, {5, 1}, {6, 0}}),
MakeJoint({{2, 1}, {3, 0}}), MakeJoint({{4, 1}, {5, 0}}), MakeJoint({{6, 1}, {7, 0}})};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
UNIT_CLASS_TEST(RestrictionTest, CrossGraph_NoUTurn)
{
Init(BuildCrossGraph());
SetStarter(routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(-1, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(7, 0, m2::PointD(1, 2)) /* finish */);
vector<m2::PointD> const expectedGeom = {
{-1.0 /* x */, 0.0 /* y */}, {0.0, 0.0}, {1.0, 0.0}, {1.0, 1.0}, {1.0, 2.0}};
TestRouteGeometry(*m_starter, AStarAlgorithm<IndexGraphStarter>::Result::OK, expectedGeom);
}
UNIT_CLASS_TEST(RestrictionTest, CrossGraph_UTurn)
{
Init(BuildCrossGraph());
SetStarter(routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(-1, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(7, 0, m2::PointD(1, 2)) /* finish */);
RestrictionVec restrictions = {
{Restriction::Type::No, {1 /* feature from */, 6 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{-1.0, 0.0}, {0.0, 0.0}, {1.0, 0.0}, {1.0, -1.0},
{1.0, 0.0}, {1.0, 1.0}, {1.0, 2.0}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(-1, 0)), /* start */
routing::IndexGraphStarter::FakeVertex(7, 0, m2::PointD(1, 2)), /* finish */
move(restrictions), *this);
}
// Finish
// 3 *
// ^
// |
// F4
// |
// 2 *
// ^ ↖
// | F1
// | ↖
// 1 | *
// F0 ↖
// | F2
// | ↖
// 0 *<--F3---<--F3---*<--F5--* Start
// 0 1 2 3
// Note. F0, F1 and F2 are one segment features. F3 is a two segments feature.
unique_ptr<IndexGraph> BuildTriangularGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {0.0, 2.0}}));
loader->AddRoad(1 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {0.0, 2.0}}));
loader->AddRoad(2 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(3 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 0.0}, {0.0, 0.0}}));
loader->AddRoad(4 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 2.0}, {0.0, 3.0}}));
loader->AddRoad(5 /* featureId */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 0.0}, {2.0, 0.0}}));
vector<Joint> const joints = {
MakeJoint({{2, 0}, {3, 0}, {5, 1}}), /* joint at point (2, 0) */
MakeJoint({{3, 2}, {0, 0}}), /* joint at point (0, 0) */
MakeJoint({{2, 1}, {1, 0}}), /* joint at point (1, 1) */
MakeJoint({{0, 1}, {1, 1}, {4, 0}}), /* joint at point (0, 2) */
MakeJoint({{5, 0}}), /* joint at point (3, 0) */
MakeJoint({{4, 1}}) /* joint at point (0, 3) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// Route through triangular graph without any restrictions.
UNIT_CLASS_TEST(RestrictionTest, TriangularGraph)
{
Init(BuildTriangularGraph());
vector<m2::PointD> const expectedGeom = {{3 /* x */, 0 /* y */}, {2, 0}, {1, 1}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */, {},
*this);
}
// Route through triangular graph with restriction type no from feature 2 to feature 1.
UNIT_CLASS_TEST(RestrictionTest, TriangularGraph_RestrictionNoF2F1)
{
Init(BuildTriangularGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {2 /* feature from */, 1 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */,
move(restrictions), *this);
}
UNIT_CLASS_TEST(RestrictionTest, TriangularGraph_RestrictionNoF5F2)
{
Init(BuildTriangularGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {5 /* feature from */, 2 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */,
move(restrictions), *this);
}
UNIT_CLASS_TEST(RestrictionTest, TriangularGraph_RestrictionOnlyF5F3)
{
Init(BuildTriangularGraph());
RestrictionVec restrictionsOnly = {
{Restriction::Type::Only, {5 /* feature from */, 3 /* feature to */}}};
RestrictionVec restrictionsNo;
ConvertRestrictionsOnlyToNoAndSort(*m_graph, restrictionsOnly, restrictionsNo);
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */,
move(restrictionsNo), *this);
}
UNIT_CLASS_TEST(RestrictionTest, TriangularGraph_RestrictionNoF5F2RestrictionOnlyF5F3)
{
Init(BuildTriangularGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {5 /* feature from */, 2 /* feature to */}},
{Restriction::Type::Only, {5 /* feature from */, 3 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */,
move(restrictions), *this);
}
// Finish
// 3 *
// |
// F4
// |
// 2 *
// | \
// F0 F2
// | \
// 1 * *
// | \
// F0 F2
// | \
// 0 *---F1--*--F1--*--F3---* Start
// 0 1 2 3
// Note. All features are two setments and two-way.
unique_ptr<IndexGraph> BuildTwowayCornerGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, false /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {0.0, 1.0}, {0.0, 2.0}}));
loader->AddRoad(1 /* feature id */, false /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 0.0}, {0.0, 0.0}}));
loader->AddRoad(2 /* feature id */, false /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 1.0}, {0.0, 2.0}}));
loader->AddRoad(3 /* feature id */, false /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 0.0}, {2.0, 0.0}}));
loader->AddRoad(4 /* feature id */, false /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 2.0}, {0.0, 3.0}}));
vector<Joint> const joints = {
MakeJoint({{1 /* feature id */, 2 /* point id */}, {0, 0}})
/* joint at point (0, 0) */,
MakeJoint({{1, 0}, {2, 0}, {3, 1}}), /* joint at point (2, 0) */
MakeJoint({{2, 2}, {0, 2}, {4, 0}}), /* joint at point (0, 2) */
MakeJoint({{4, 1}}), /* joint at point (0, 3) */
MakeJoint({{3, 0}}), /* joint at point (3, 0) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
UNIT_CLASS_TEST(RestrictionTest, TwowayCornerGraph)
{
Init(BuildTwowayCornerGraph());
vector<m2::PointD> const expectedGeom = {{3 /* x */, 0 /* y */}, {2, 0}, {1, 1}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(3, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */, {},
*this);
}
UNIT_CLASS_TEST(RestrictionTest, TwowayCornerGraph_RestrictionF3F2No)
{
Init(BuildTwowayCornerGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {3 /* feature from */, 2 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}, {0, 1}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(3, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */,
move(restrictions), *this);
}
UNIT_CLASS_TEST(RestrictionTest, TwowayCornerGraph_RestrictionF3F1Only)
{
Init(BuildTwowayCornerGraph());
RestrictionVec restrictionsOnly = {
{Restriction::Type::Only, {3 /* feature from */, 1 /* feature to */}}};
RestrictionVec restrictionsNo;
ConvertRestrictionsOnlyToNoAndSort(*m_graph, restrictionsOnly, restrictionsNo);
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}, {0, 1}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(3, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(4, 0, m2::PointD(0, 3)) /* finish */,
move(restrictionsNo), *this);
}
// Finish
// 3 *
// ^
// |
// F11
// |
// 2 *<---F5----*<---F6---*
// ^ ↖ ^ ↖ ^
// | F7 | F8 |
// | ↖ F1 ↖ F2
// | ↖ | ↖ |
// 1 F0 * *
// | ^ ↖ ^
// | F1 F9 F2
// | | ↖ |
// 0 *<----F4---*<---F3----*<--F10---* Start
// 0 1 2 3
// Note. F1 and F2 are two segments features. The others are one segment ones.
unique_ptr<IndexGraph> BuildTwoSquaresGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {0.0, 2.0}}));
loader->AddRoad(1 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {1.0, 1.0}, {1.0, 2.0}}));
loader->AddRoad(2 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {2.0, 1.0}, {2.0, 2.0}}));
loader->AddRoad(3 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 0.0}}));
loader->AddRoad(4 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {0.0, 0.0}}));
loader->AddRoad(5 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 2.0}, {0.0, 2.0}}));
loader->AddRoad(6 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 2.0}, {1.0, 2.0}}));
loader->AddRoad(7 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {0.0, 2.0}}));
loader->AddRoad(8 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 1.0}, {1.0, 2.0}}));
loader->AddRoad(9 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(10 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 0.0}, {2.0, 0.0}}));
loader->AddRoad(11 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 2.0}, {0.0, 3.0}}));
vector<Joint> const joints = {
MakeJoint({{4 /* featureId */, 1 /* pointId */}, {0, 0}}), /* joint at point (0, 0) */
MakeJoint({{0, 1}, {5, 1}, {7, 1}, {11, 0}}), /* joint at point (0, 2) */
MakeJoint({{4, 0}, {1, 0}, {3, 1}}), /* joint at point (1, 0) */
MakeJoint({{5, 0}, {1, 2}, {6, 1}, {8, 1}}), /* joint at point (1, 2) */
MakeJoint({{3, 0}, {2, 0}, {9, 0}, {10, 1}}), /* joint at point (2, 0) */
MakeJoint({{2, 2}, {6, 0}}), /* joint at point (2, 2) */
MakeJoint({{1, 1}, {9, 1}, {7, 0}}), /* joint at point (1, 1) */
MakeJoint({{2, 1}, {8, 0}}), /* joint at point (2, 1) */
MakeJoint({{10, 0}}), /* joint at point (3, 0) */
MakeJoint({{11, 1}}), /* joint at point (0, 3) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
UNIT_CLASS_TEST(RestrictionTest, TwoSquaresGraph)
{
Init(BuildTwoSquaresGraph());
vector<m2::PointD> const expectedGeom = {{3 /* x */, 0 /* y */}, {2, 0}, {1, 1}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(10, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(11, 0, m2::PointD(0, 3)) /* finish */, {},
*this);
}
UNIT_CLASS_TEST(RestrictionTest, TwoSquaresGraph_RestrictionF10F3Only)
{
Init(BuildTwoSquaresGraph());
RestrictionVec restrictionsOnly = {
{Restriction::Type::Only, {10 /* feature from */, 3 /* feature to */}}};
RestrictionVec restrictionsNo;
ConvertRestrictionsOnlyToNoAndSort(*m_graph, restrictionsOnly, restrictionsNo);
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {1, 1}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(10, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(11, 0, m2::PointD(0, 3)) /* finish */,
move(restrictionsNo), *this);
}
UNIT_CLASS_TEST(RestrictionTest, TwoSquaresGraph_RestrictionF10F3OnlyF3F4Only)
{
Init(BuildTwoSquaresGraph());
RestrictionVec restrictionsOnly = {
{Restriction::Type::Only, {3 /* feature from */, 4 /* feature to */}},
{Restriction::Type::Only, {10 /* feature from */, 3 /* feature to */}}};
RestrictionVec restrictionsNo;
ConvertRestrictionsOnlyToNoAndSort(*m_graph, restrictionsOnly, restrictionsNo);
vector<m2::PointD> const expectedGeom = {
{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(10, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(11, 0, m2::PointD(0, 3)) /* finish */,
move(restrictionsNo), *this);
}
UNIT_CLASS_TEST(RestrictionTest, TwoSquaresGraph_RestrictionF2F8NoRestrictionF9F1Only)
{
Init(BuildTwoSquaresGraph());
RestrictionVec restrictionsNo = {
{Restriction::Type::No, {2 /* feature from */, 8 /* feature to */}}}; // Invalid restriction.
RestrictionVec const restrictionsOnly = {
{Restriction::Type::Only,
{9 /* feature from */, 1 /* feature to */}}}; // Invalid restriction.
ConvertRestrictionsOnlyToNoAndSort(*m_graph, restrictionsOnly, restrictionsNo);
vector<m2::PointD> const expectedGeom = {{3 /* x */, 0 /* y */}, {2, 0}, {1, 1}, {0, 2}, {0, 3}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(10, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(11, 0, m2::PointD(0, 3)) /* finish */,
move(restrictionsNo), *this);
}
// 2 *
// |
// F6
// |Finish
// 1 *-F4-*-F5-*
// | |
// F2 F3
// | |
// 0 *---F1----*---F0---* Start
// 0 1 2
// Note 1. All features are two-way. (It's possible to move along any direction of the features.)
// Note 2. Any feature contains of one segment.
unique_ptr<IndexGraph> BuildFlagGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 0.0}}));
loader->AddRoad(1 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {0.0, 0.0}}));
loader->AddRoad(2 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {0.0, 1.0}}));
loader->AddRoad(3 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(4 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 1.0}, {0.5, 1.0}}));
loader->AddRoad(5 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.5, 1.0}, {1.0, 1.0}}));
loader->AddRoad(6 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.5, 1.0}, {0.5, 2.0}}));
vector<Joint> const joints = {
MakeJoint({{1 /* feature id */, 1 /* point id */}, {2, 0}}), /* joint at point (0, 0) */
MakeJoint({{2, 1}, {4, 0}}), /* joint at point (0, 1) */
MakeJoint({{4, 1}, {5, 0}, {6, 0}}), /* joint at point (0.5, 1) */
MakeJoint({{1, 0}, {3, 0}, {0, 1}}), /* joint at point (1, 0) */
MakeJoint({{3, 1}, {5, 1}}), /* joint at point (1, 1) */
MakeJoint({{6, 1}}), /* joint at point (0.5, 2) */
MakeJoint({{0, 0}}), /* joint at point (2, 0) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// Route through flag graph without any restrictions.
UNIT_TEST(FlagGraph)
{
unique_ptr<IndexGraph> graph = BuildFlagGraph();
IndexGraphStarter starter(
*graph, routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(0.5, 1)) /* finish */);
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 0}, {1, 1}, {0.5, 1}};
TestRouteGeometry(starter, AStarAlgorithm<IndexGraphStarter>::Result::OK, expectedGeom);
}
// Route through flag graph with one restriciton (type no) from F0 to F3.
UNIT_CLASS_TEST(RestrictionTest, FlagGraph_RestrictionF0F3No)
{
Init(BuildFlagGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {0 /* feature from */, 3 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{2 /* x */, 0 /* y */}, {1, 0}, {0, 0}, {0, 1}, {0.5, 1}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(0.5, 1)) /* finish */,
move(restrictions), *this);
}
// Route through flag graph with one restriciton (type only) from F0 to F1.
UNIT_CLASS_TEST(RestrictionTest, FlagGraph_RestrictionF0F1Only)
{
Init(BuildFlagGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {0 /* feature from */, 1 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 0}, {1, 1}, {0.5, 1}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(0.5, 1)) /* finish */,
move(restrictions), *this);
}
UNIT_CLASS_TEST(RestrictionTest, FlagGraph_PermutationsF1F3NoF7F8OnlyF8F4OnlyF4F6Only)
{
Init(BuildFlagGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {0 /* feature from */, 3 /* feature to */}},
{Restriction::Type::Only, {0 /* feature from */, 1 /* feature to */}},
{Restriction::Type::Only, {1 /* feature from */, 2 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{2 /* x */, 0 /* y */}, {1, 0}, {0, 0}, {0, 1}, {0.5, 1}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(0.5, 1)) /* finish */,
move(restrictions), *this);
}
// 1 *-F4-*-F5-*---F6---* Finish
// | |
// F2 F3
// | |
// 0 *---F1----*---F0---* Start
// 0 1 2
// Note 1. All features except for F7 are two-way.
// Note 2. Any feature contains of one segment.
unique_ptr<IndexGraph> BuildPosterGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 0.0}}));
loader->AddRoad(1 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {0.0, 0.0}}));
loader->AddRoad(2 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {0.0, 1.0}}));
loader->AddRoad(3 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(4 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 1.0}, {0.5, 1.0}}));
loader->AddRoad(5 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.5, 1.0}, {1.0, 1.0}}));
loader->AddRoad(6 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 1.0}, {2.0, 1.0}}));
vector<Joint> const joints = {
MakeJoint({{1 /* feature id */, 1 /* point id */}, {2, 0}}), /* joint at point (0, 0) */
MakeJoint({{2, 1}, {4, 0}}), /* joint at point (0, 1) */
MakeJoint({{4, 1}, {5, 0}}), /* joint at point (0.5, 1) */
MakeJoint({{1, 0}, {3, 0}, {0, 1}}), /* joint at point (1, 0) */
MakeJoint({{3, 1}, {5, 1}, {6, 0}}), /* joint at point (1, 1) */
MakeJoint({{0, 0}}), /* joint at point (2, 0) */
MakeJoint({{6, 1}}), /* joint at point (2, 1) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// Route through poster graph without any restrictions.
UNIT_TEST(PosterGraph)
{
unique_ptr<IndexGraph> graph = BuildPosterGraph();
IndexGraphStarter starter(
*graph, routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(2, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(2, 1)) /* finish */);
vector<m2::PointD> const expectedGeom = {{2 /* x */, 0 /* y */}, {1, 0}, {1, 1}, {2, 1}};
TestRouteGeometry(starter, AStarAlgorithm<IndexGraphStarter>::Result::OK, expectedGeom);
}
// Route through poster graph with restrictions F0-F3 (type no).
UNIT_CLASS_TEST(RestrictionTest, PosterGraph_RestrictionF0F3No)
{
Init(BuildPosterGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {0 /* feature from */, 3 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{2 /* x */, 0 /* y */}, {1, 0}, {0, 0}, {0, 1}, {0.5, 1}, {1, 1}, {2, 1}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(2, 0)), /* start */
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(2, 1)), /* finish */
move(restrictions), *this);
}
// Route through poster graph with restrictions F0-F1 (type only).
UNIT_CLASS_TEST(RestrictionTest, PosterGraph_RestrictionF0F1Only)
{
Init(BuildPosterGraph());
RestrictionVec restrictionsOnly = {
{Restriction::Type::Only, {0 /* feature from */, 1 /* feature to */}}};
RestrictionVec restrictionsNo;
ConvertRestrictionsOnlyToNoAndSort(*m_graph, restrictionsOnly, restrictionsNo);
vector<m2::PointD> const expectedGeom = {
{2 /* x */, 0 /* y */}, {1, 0}, {0, 0}, {0, 1}, {0.5, 1}, {1, 1}, {2, 1}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(2, 0)), /* start */
routing::IndexGraphStarter::FakeVertex(6, 0, m2::PointD(2, 1)), /* finish */
move(restrictionsNo), *this);
}
// 1 *--F1-->*
// ↗ ↘
// F1 F1
// ↗ ↘
// 0 Start *---F3--->*---F0--->-------F0----->*---F2-->* Finish
// -1 0 1 2 3 4
// Note. F0 is a two segments feature. F1 is a three segment one. F2 and F3 are one segment
// features.
unique_ptr<IndexGraph> BuildTwoWayGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {1.0, 0.0}, {3.0, 0}}));
loader->AddRoad(1 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {1.0, 1.0}, {2.0, 1.0}, {3.0, 0.0}}));
loader->AddRoad(2 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 0.0}, {4.0, 0.0}}));
loader->AddRoad(3 /* feature id */, true /* oneWay */, 1.0 /* speed */,
RoadGeometry::Points({{-1.0, 0.0}, {0.0, 0.0}}));
vector<Joint> const joints = {
MakeJoint(
{{0 /* feature id */, 0 /* point id */}, {1, 0}, {3, 1}}), /* joint at point (0, 0) */
MakeJoint(
{{0 /* feature id */, 2 /* point id */}, {1, 3}, {2, 0}}), /* joint at point (3, 0) */
MakeJoint({{3 /* feature id */, 0 /* point id */}}), /* joint at point (-1, 0) */
MakeJoint({{2 /* feature id */, 1 /* point id */}}), /* joint at point (4, 0) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
UNIT_TEST(TwoWayGraph)
{
unique_ptr<IndexGraph> graph = BuildTwoWayGraph();
IndexGraphStarter starter(
*graph, routing::IndexGraphStarter::FakeVertex(3, 0, m2::PointD(-1, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(2, 0, m2::PointD(4, 0)) /* finish */);
vector<m2::PointD> const expectedGeom = {{-1 /* x */, 0 /* y */}, {0, 0}, {1, 0}, {3, 0}, {4, 0}};
TestRouteGeometry(starter, AStarAlgorithm<IndexGraphStarter>::Result::OK, expectedGeom);
}
// 1 *---F4----*
// | |
// F2 F3
// | |
// 0 *<--F5---*<--F1----*<--F0---* Start
// Finish
// 0 1 2 3
// Note 1. F0, F1 and F5 are one-way features. F3, F2 and F4 are two-way features.
// Note 2. Any feature contains of one segment.
unique_ptr<IndexGraph> BuildSquaresGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{3.0, 0.0}, {2.0, 0.0}}));
loader->AddRoad(1 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {1.0, 0.0}}));
loader->AddRoad(2 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {1.0, 1.0}}));
loader->AddRoad(3 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 0.0}, {2.0, 1.0}}));
loader->AddRoad(4 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{2.0, 1.0}, {1.0, 1.0}}));
loader->AddRoad(5 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {0.0, 0.0}}));
vector<Joint> const joints = {
MakeJoint({{0 /* feature id */, 0 /* point id */}}), /* joint at point (3, 0) */
MakeJoint({{0, 1}, {3, 0}, {1, 0}}), /* joint at point (2, 0) */
MakeJoint({{3, 1}, {4, 0}}), /* joint at point (2, 1) */
MakeJoint({{2, 1}, {4, 1}}), /* joint at point (1, 1) */
MakeJoint({{1, 1}, {2, 0}, {5, 0}}), /* joint at point (1, 0) */
MakeJoint({{5, 1}}) /* joint at point (0, 0) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
UNIT_TEST(SquaresGraph)
{
unique_ptr<IndexGraph> graph = BuildSquaresGraph();
IndexGraphStarter starter(
*graph, routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(0, 0)) /* finish */);
vector<m2::PointD> const expectedGeom = {{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}};
TestRouteGeometry(starter, AStarAlgorithm<IndexGraphStarter>::Result::OK, expectedGeom);
}
// It's a test on correct working in case when because of adding restrictions
// start and finish could be match on blocked, moved or copied edges.
// See IndexGraphStarter constructor for a detailed description.
UNIT_CLASS_TEST(RestrictionTest, SquaresGraph_RestrictionF0F1OnlyF1F5Only)
{
Init(BuildSquaresGraph());
RestrictionVec restrictions = {
{Restriction::Type::Only, {0 /* feature from */, 1 /* feature to */}},
{Restriction::Type::Only, {1 /* feature from */, 5 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{3 /* x */, 0 /* y */}, {2, 0}, {1, 0}, {0, 0}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0, 0, m2::PointD(3, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(5, 0, m2::PointD(0, 0)) /* finish */,
move(restrictions), *this);
}
// 0 Start *--F0--->*---F1---*---F1---*---F1---*---F2-->* Finish
// 0 1 2 3 4 5
// Note. F0 and F2 are one segment one-way features. F1 is a 3 segment two-way feature.
unique_ptr<IndexGraph> BuildLineGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {0.0, 1.0}}));
loader->AddRoad(1 /* feature id */, false /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{1.0, 0.0}, {2.0, 0.0}, {3.0, 0.0}, {3.0, 0.0}}));
loader->AddRoad(2 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{4.0, 0.0}, {5.0, 0.0}}));
vector<Joint> const joints = {
MakeJoint({{0 /* feature id */, 0 /* point id */}}), /* joint at point (0, 0) */
MakeJoint({{0, 1}, {1, 0}}), /* joint at point (1, 0) */
MakeJoint({{1, 3}, {2, 0}}), /* joint at point (4, 0) */
MakeJoint({{2, 1}}), /* joint at point (5, 0) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// This test checks that despite the fact uturn on F1 is prohibited (moving from F1 to F1 is
// prohibited)
// it's still possible to move from F1 to F1 in straight direction.
UNIT_CLASS_TEST(RestrictionTest, LineGraph_RestrictionF1F1No)
{
Init(BuildLineGraph());
RestrictionVec restrictions = {
{Restriction::Type::No, {1 /* feature from */, 1 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {
{0 /* x */, 0 /* y */}, {1, 0}, {2, 0}, {3, 0}, {4, 0}, {5, 0}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0 /* feature id */, 0 /* seg id */,
m2::PointD(0, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(2, 0, m2::PointD(5, 0)) /* finish */,
move(restrictions), *this);
}
// 2 *---F2-->*
// ^
// F0
// |
// 1 *---F1-->* Finish
// ^
// F0
// |
// 0 *
// 0 1
// Start
unique_ptr<IndexGraph> BuildFGraph()
{
unique_ptr<TestGeometryLoader> loader = make_unique<TestGeometryLoader>();
loader->AddRoad(0 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 0.0}, {0.0, 1.0}, {0.0, 2.0}}));
loader->AddRoad(1 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 1.0}, {1.0, 1.0}}));
loader->AddRoad(2 /* feature id */, true /* one way */, 1.0 /* speed */,
RoadGeometry::Points({{0.0, 2.0}, {1.0, 2.0}}));
vector<Joint> const joints = {
MakeJoint({{0 /* feature id */, 0 /* point id */}}), /* joint at point (0, 0) */
MakeJoint({{0, 1}, {1, 0}}), /* joint at point (0, 1) */
MakeJoint({{0, 2}, {2, 0}}), /* joint at point (0, 2) */
MakeJoint({{1, 1}}), /* joint at point (1, 1) */
MakeJoint({{2, 1}}), /* joint at point (1, 2) */
};
traffic::TrafficCache const trafficCache;
unique_ptr<IndexGraph> graph =
make_unique<IndexGraph>(move(loader), CreateEstimator(trafficCache));
graph->Import(joints);
return graph;
}
// This test checks that having a Only restriction from F0 to F2 it's still possible move
// from F0 to F1.
UNIT_CLASS_TEST(RestrictionTest, FGraph_RestrictionF0F2Only)
{
Init(BuildFGraph());
RestrictionVec restrictions = {
{Restriction::Type::Only, {0 /* feature from */, 2 /* feature to */}}};
vector<m2::PointD> const expectedGeom = {{0 /* x */, 0 /* y */}, {0, 1}, {1, 1}};
TestRestrictions(expectedGeom, AStarAlgorithm<IndexGraphStarter>::Result::OK,
routing::IndexGraphStarter::FakeVertex(0 /* feature id */, 0 /* seg id */,
m2::PointD(0, 0)) /* start */,
routing::IndexGraphStarter::FakeVertex(1, 0, m2::PointD(1, 1)) /* finish */,
move(restrictions), *this);
}
} // namespace routing_test

View file

@ -27,12 +27,14 @@ SOURCES += \
astar_router_test.cpp \
async_router_test.cpp \
cross_routing_tests.cpp \
cumulative_restriction_test.cpp \
followed_polyline_test.cpp \
index_graph_test.cpp \
index_graph_tools.cpp \
nearest_edge_finder_tests.cpp \
online_cross_fetcher_test.cpp \
osrm_router_test.cpp \
restriction_test.cpp \
road_graph_builder.cpp \
road_graph_nearest_edges_test.cpp \
route_tests.cpp \

View file

@ -10,6 +10,7 @@
#include "routing/index_graph_starter.hpp"
#include "routing/index_road_graph.hpp"
#include "routing/pedestrian_model.hpp"
#include "routing/restriction_loader.hpp"
#include "routing/route.hpp"
#include "routing/routing_helpers.hpp"
#include "routing/turns_generator.hpp"
@ -190,6 +191,10 @@ bool SingleMwmRouter::LoadIndex(MwmSet::MwmId const & mwmId, string const & coun
FilesContainerR::TReader reader(mwmValue->m_cont.GetReader(ROUTING_FILE_TAG));
ReaderSource<FilesContainerR::TReader> src(reader);
IndexGraphSerializer::Deserialize(graph, src, kCarMask);
RestrictionLoader restrictionLoader(*mwmValue, graph);
if (restrictionLoader.HasRestrictions())
graph.SetRestrictions(restrictionLoader.StealRestrictions());
LOG(LINFO,
(ROUTING_FILE_TAG, "section for", country, "loaded in", timer.ElapsedSeconds(), "seconds"));
return true;
@ -203,8 +208,9 @@ bool SingleMwmRouter::LoadIndex(MwmSet::MwmId const & mwmId, string const & coun
}
bool SingleMwmRouter::RedressRoute(MwmSet::MwmId const & mwmId, IVehicleModel const & vehicleModel,
vector<Segment> const & segments, RouterDelegate const & delegate,
IndexGraphStarter & starter, Route & route) const
vector<Segment> const & segments,
RouterDelegate const & delegate, IndexGraphStarter & starter,
Route & route) const
{
vector<Junction> junctions;
size_t const numPoints = IndexGraphStarter::GetRouteNumPoints(segments);

View file

@ -50,6 +50,12 @@
56555E591D897D28009D786D /* testingmain.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 6742ACDE1C68A13F009CB89E /* testingmain.cpp */; };
56826BD01DB51C4E00807C62 /* car_router.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56826BCE1DB51C4E00807C62 /* car_router.cpp */; };
56826BD11DB51C4E00807C62 /* car_router.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56826BCF1DB51C4E00807C62 /* car_router.hpp */; };
56CA09E31E30E73B00D05C9A /* applying_traffic_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56CA09DE1E30E73B00D05C9A /* applying_traffic_test.cpp */; };
56CA09E41E30E73B00D05C9A /* cumulative_restriction_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56CA09DF1E30E73B00D05C9A /* cumulative_restriction_test.cpp */; };
56CA09E51E30E73B00D05C9A /* index_graph_tools.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56CA09E01E30E73B00D05C9A /* index_graph_tools.cpp */; };
56CA09E61E30E73B00D05C9A /* index_graph_tools.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56CA09E11E30E73B00D05C9A /* index_graph_tools.hpp */; };
56CA09E71E30E73B00D05C9A /* restriction_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56CA09E21E30E73B00D05C9A /* restriction_test.cpp */; };
56CA09E91E30F19800D05C9A /* libtraffic.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 56CA09E81E30F19800D05C9A /* libtraffic.a */; };
56EA2FD51D8FD8590083F01A /* routing_helpers.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56EA2FD41D8FD8590083F01A /* routing_helpers.hpp */; };
56F0D7341D896A5300045886 /* libmap.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 67BD35DE1C69F198003AA26F /* libmap.a */; };
56F0D7391D896A5300045886 /* libstorage.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 67BD35D41C69F155003AA26F /* libstorage.a */; };
@ -141,7 +147,6 @@
6742AD561C68C7D7009CB89E /* types.txt in Resources */ = {isa = PBXBuildFile; fileRef = 6742AD551C68C7C1009CB89E /* types.txt */; };
6742AD5C1C68C7FA009CB89E /* drules_proto_clear.bin in Resources */ = {isa = PBXBuildFile; fileRef = 6742AD571C68C7F6009CB89E /* drules_proto_clear.bin */; };
6742AD5D1C68C7FA009CB89E /* drules_proto_dark.bin in Resources */ = {isa = PBXBuildFile; fileRef = 6742AD581C68C7F6009CB89E /* drules_proto_dark.bin */; };
6742AD5E1C68C7FA009CB89E /* drules_proto_legacy.bin in Resources */ = {isa = PBXBuildFile; fileRef = 6742AD591C68C7F6009CB89E /* drules_proto_legacy.bin */; };
6742AD5F1C68C7FA009CB89E /* drules_proto-bw.bin in Resources */ = {isa = PBXBuildFile; fileRef = 6742AD5A1C68C7F6009CB89E /* drules_proto-bw.bin */; };
6742AD601C68C7FA009CB89E /* drules_proto.bin in Resources */ = {isa = PBXBuildFile; fileRef = 6742AD5B1C68C7F6009CB89E /* drules_proto.bin */; };
6742AD621C68F747009CB89E /* sound-strings in Resources */ = {isa = PBXBuildFile; fileRef = 6742AD611C68F747009CB89E /* sound-strings */; };
@ -223,8 +228,6 @@
67BD36051C69F51C003AA26F /* WorldCoasts.mwm in Resources */ = {isa = PBXBuildFile; fileRef = 67BD36021C69F513003AA26F /* WorldCoasts.mwm */; };
67C79BA11E2CEE1400C40034 /* restriction_loader.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 67C79B9F1E2CEE1400C40034 /* restriction_loader.cpp */; };
67C79BA21E2CEE1400C40034 /* restriction_loader.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 67C79BA01E2CEE1400C40034 /* restriction_loader.hpp */; };
67C79BA51E2CEE3100C40034 /* routing_serialization.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 67C79BA31E2CEE3100C40034 /* routing_serialization.cpp */; };
67C79BA61E2CEE3100C40034 /* routing_serialization.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 67C79BA41E2CEE3100C40034 /* routing_serialization.hpp */; };
67C7D4291B4EB48F00FE41AA /* car_model.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 67C7D4211B4EB48F00FE41AA /* car_model.cpp */; };
67C7D42A1B4EB48F00FE41AA /* car_model.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 67C7D4221B4EB48F00FE41AA /* car_model.hpp */; };
67C7D42B1B4EB48F00FE41AA /* pedestrian_model.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 67C7D4231B4EB48F00FE41AA /* pedestrian_model.cpp */; };
@ -292,6 +295,12 @@
563B91C41CC4F1DC00222BC1 /* bicycle_model.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = bicycle_model.hpp; sourceTree = "<group>"; };
56826BCE1DB51C4E00807C62 /* car_router.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = car_router.cpp; sourceTree = "<group>"; };
56826BCF1DB51C4E00807C62 /* car_router.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = car_router.hpp; sourceTree = "<group>"; };
56CA09DE1E30E73B00D05C9A /* applying_traffic_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = applying_traffic_test.cpp; sourceTree = "<group>"; };
56CA09DF1E30E73B00D05C9A /* cumulative_restriction_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = cumulative_restriction_test.cpp; sourceTree = "<group>"; };
56CA09E01E30E73B00D05C9A /* index_graph_tools.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = index_graph_tools.cpp; sourceTree = "<group>"; };
56CA09E11E30E73B00D05C9A /* index_graph_tools.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = index_graph_tools.hpp; sourceTree = "<group>"; };
56CA09E21E30E73B00D05C9A /* restriction_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = restriction_test.cpp; sourceTree = "<group>"; };
56CA09E81E30F19800D05C9A /* libtraffic.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libtraffic.a; path = "/Users/vladimirbykoyanko/src_github_master/omim/xcode/traffic/../../../omim-build/xcode/Debug/libtraffic.a"; sourceTree = "<absolute>"; };
56EA2FD41D8FD8590083F01A /* routing_helpers.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = routing_helpers.hpp; sourceTree = "<group>"; };
56F0D75F1D896A5300045886 /* routing_benchmarks.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = routing_benchmarks.app; sourceTree = BUILT_PRODUCTS_DIR; };
670B84BE1A9381D900CE4492 /* cross_routing_context.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = cross_routing_context.cpp; sourceTree = "<group>"; };
@ -415,8 +424,6 @@
67BD36021C69F513003AA26F /* WorldCoasts.mwm */ = {isa = PBXFileReference; lastKnownFileType = file; path = WorldCoasts.mwm; sourceTree = "<group>"; };
67C79B9F1E2CEE1400C40034 /* restriction_loader.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = restriction_loader.cpp; sourceTree = "<group>"; };
67C79BA01E2CEE1400C40034 /* restriction_loader.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = restriction_loader.hpp; sourceTree = "<group>"; };
67C79BA31E2CEE3100C40034 /* routing_serialization.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = routing_serialization.cpp; sourceTree = "<group>"; };
67C79BA41E2CEE3100C40034 /* routing_serialization.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = routing_serialization.hpp; sourceTree = "<group>"; };
67C7D4211B4EB48F00FE41AA /* car_model.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = car_model.cpp; sourceTree = "<group>"; };
67C7D4221B4EB48F00FE41AA /* car_model.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = car_model.hpp; sourceTree = "<group>"; };
67C7D4231B4EB48F00FE41AA /* pedestrian_model.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = pedestrian_model.cpp; sourceTree = "<group>"; };
@ -471,6 +478,7 @@
isa = PBXFrameworksBuildPhase;
buildActionMask = 2147483647;
files = (
56CA09E91E30F19800D05C9A /* libtraffic.a in Frameworks */,
3462FDAD1DC1E5BF00906FD7 /* libopening_hours.a in Frameworks */,
6742AD4D1C68AA59009CB89E /* librouting.a in Frameworks */,
6742AD401C68AA04009CB89E /* libosrm.a in Frameworks */,
@ -542,6 +550,7 @@
56F0D7611D896DAF00045886 /* Frameworks */ = {
isa = PBXGroup;
children = (
56CA09E81E30F19800D05C9A /* libtraffic.a */,
3462FDAC1DC1E5BF00906FD7 /* libopening_hours.a */,
);
name = Frameworks;
@ -561,6 +570,11 @@
6742ACA01C68A07C009CB89E /* routing_tests */ = {
isa = PBXGroup;
children = (
56CA09DE1E30E73B00D05C9A /* applying_traffic_test.cpp */,
56CA09DF1E30E73B00D05C9A /* cumulative_restriction_test.cpp */,
56CA09E01E30E73B00D05C9A /* index_graph_tools.cpp */,
56CA09E11E30E73B00D05C9A /* index_graph_tools.hpp */,
56CA09E21E30E73B00D05C9A /* restriction_test.cpp */,
6742ACDE1C68A13F009CB89E /* testingmain.cpp */,
6742ACA61C68A0B1009CB89E /* astar_algorithm_test.cpp */,
6742ACA71C68A0B1009CB89E /* astar_progress_test.cpp */,
@ -685,8 +699,6 @@
675343FA1A3F640D00A0A8C3 /* routing */ = {
isa = PBXGroup;
children = (
67C79BA31E2CEE3100C40034 /* routing_serialization.cpp */,
67C79BA41E2CEE3100C40034 /* routing_serialization.hpp */,
67C79B9F1E2CEE1400C40034 /* restriction_loader.cpp */,
67C79BA01E2CEE1400C40034 /* restriction_loader.hpp */,
0C470E6F1E0D4EB1005B824D /* segment.hpp */,
@ -844,11 +856,11 @@
A1616E2C1B6B60AB003F078E /* router_delegate.hpp in Headers */,
A17B42991BCFBD0E00A1EAE4 /* osrm_helpers.hpp in Headers */,
67C7D42E1B4EB48F00FE41AA /* turns_sound_settings.hpp in Headers */,
67C79BA61E2CEE3100C40034 /* routing_serialization.hpp in Headers */,
56099E341CC9247E00A7772A /* bicycle_directions.hpp in Headers */,
670EE55E1B6001E7001E8064 /* routing_session.hpp in Headers */,
56099E291CC7C97D00A7772A /* loaded_path_segment.hpp in Headers */,
670EE55F1B6001E7001E8064 /* routing_settings.hpp in Headers */,
56CA09E61E30E73B00D05C9A /* index_graph_tools.hpp in Headers */,
671F58BE1B874EC80032311E /* followed_polyline.hpp in Headers */,
67C7D42C1B4EB48F00FE41AA /* pedestrian_model.hpp in Headers */,
0C5FEC6A1DDE193F0017688C /* road_index.hpp in Headers */,
@ -1026,7 +1038,6 @@
6742AD601C68C7FA009CB89E /* drules_proto.bin in Resources */,
6742AD621C68F747009CB89E /* sound-strings in Resources */,
6742AD5D1C68C7FA009CB89E /* drules_proto_dark.bin in Resources */,
6742AD5E1C68C7FA009CB89E /* drules_proto_legacy.bin in Resources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
@ -1094,6 +1105,7 @@
buildActionMask = 2147483647;
files = (
0C5FEC641DDE192A0017688C /* joint.cpp in Sources */,
56CA09E71E30E73B00D05C9A /* restriction_test.cpp in Sources */,
0C5BC9D11E28FD4E0071BFDD /* index_road_graph.cpp in Sources */,
56826BD01DB51C4E00807C62 /* car_router.cpp in Sources */,
56099E2E1CC8FBDA00A7772A /* osrm_path_segment_factory.cpp in Sources */,
@ -1124,11 +1136,12 @@
674A28B11B1605D2001A525C /* osrm_engine.cpp in Sources */,
0C08AA341DF83223004195DD /* index_graph_serialization.cpp in Sources */,
674F9BD41B0A580E00704FFA /* road_graph.cpp in Sources */,
56CA09E51E30E73B00D05C9A /* index_graph_tools.cpp in Sources */,
0C0DF92A1DE898FF0055A22F /* routing_helpers.cpp in Sources */,
67C79BA51E2CEE3100C40034 /* routing_serialization.cpp in Sources */,
67AB92E61B7B3E6E00AB5194 /* turns_tts_text.cpp in Sources */,
0C5FEC601DDE192A0017688C /* index_graph.cpp in Sources */,
0C5FEC6D1DDE19A40017688C /* index_graph_test.cpp in Sources */,
56CA09E41E30E73B00D05C9A /* cumulative_restriction_test.cpp in Sources */,
6753441E1A3F644F00A0A8C3 /* turns.cpp in Sources */,
670B84C01A9381D900CE4492 /* cross_routing_context.cpp in Sources */,
A120B3501B4A7C0A002F3808 /* routing_algorithm.cpp in Sources */,
@ -1136,6 +1149,7 @@
0C5FEC5E1DDE192A0017688C /* geometry.cpp in Sources */,
674F9BD01B0A580E00704FFA /* online_cross_fetcher.cpp in Sources */,
670EE5751B664796001E8064 /* router.cpp in Sources */,
56CA09E31E30E73B00D05C9A /* applying_traffic_test.cpp in Sources */,
0C5FEC621DDE192A0017688C /* joint_index.cpp in Sources */,
670C62131AC5A15700C38A8C /* routing_mapping.cpp in Sources */,
67C79BA11E2CEE1400C40034 /* restriction_loader.cpp in Sources */,