From f9934cb4749bc0b0e941ff1ef3f2782e7b0bef23 Mon Sep 17 00:00:00 2001 From: Olga Khlopkova Date: Wed, 8 Jul 2020 14:40:11 +0300 Subject: [PATCH] [transit] If edge weight is zero, calculate it based on route. --- transit/world_feed/feed_helpers.cpp | 12 +++++- transit/world_feed/world_feed.cpp | 58 +++++++++++++++++++++++++++++ transit/world_feed/world_feed.hpp | 2 + 3 files changed, 71 insertions(+), 1 deletion(-) diff --git a/transit/world_feed/feed_helpers.cpp b/transit/world_feed/feed_helpers.cpp index aebbb5d668..b9f9d33f27 100644 --- a/transit/world_feed/feed_helpers.cpp +++ b/transit/world_feed/feed_helpers.cpp @@ -117,7 +117,7 @@ std::pair PrepareNearestPointOnTrack(m2::PointD const & point, siz // We find the most fitting projection of the stop to the polyline. For two different projections // with approximately equal distances to the stop the most preferable is the one that is closer // to the beginning of the polyline segment. - auto const proj = + auto proj = std::min_element(projections.begin(), projections.end(), [](ProjectionData const & p1, ProjectionData const & p2) { if (CloserToStartAndOnSimilarDistToLine(p1, p2)) @@ -132,6 +132,16 @@ std::pair PrepareNearestPointOnTrack(m2::PointD const & point, siz return p1.m_distFromPoint < p2.m_distFromPoint; }); + // This case is possible not only for the first stop on the shape. We try to resolve situation + // when two stops are projected to the same point on the shape. + if (proj->m_indexOnShape == startIndex) + { + proj = std::min_element(projections.begin(), projections.end(), + [](ProjectionData const & p1, ProjectionData const & p2) { + return p1.m_distFromPoint < p2.m_distFromPoint; + }); + } + if (proj->m_needsInsertion) polyline.insert(polyline.begin() + proj->m_indexOnShape, proj->m_proj); diff --git a/transit/world_feed/world_feed.cpp b/transit/world_feed/world_feed.cpp index ee3885f615..242d4953ba 100644 --- a/transit/world_feed/world_feed.cpp +++ b/transit/world_feed/world_feed.cpp @@ -17,6 +17,7 @@ #include "base/newtype.hpp" #include +#include #include #include #include @@ -29,6 +30,10 @@ namespace { +// TODO(o.khlopkova) Set average speed for each type of transit separately - trains, buses, etc. +// Average transit speed. Approximately 40 km/h. +static double constexpr kAvgTransitSpeedMpS = 11.1; + template void AddToRegions(C & container, ID const & id, transit::Regions const & regions) { @@ -602,6 +607,14 @@ bool WorldFeed::FillStopsEdges() static_cast(stopTime2.arrival_time.get_total_seconds() - stopTime1.departure_time.get_total_seconds()); + if (data.m_weight == 0) + { + double const distBetweenStopsM = + stopTime2.shape_dist_traveled - stopTime1.shape_dist_traveled; + if (distBetweenStopsM > 0) + data.m_weight = std::ceil(distBetweenStopsM / kAvgTransitSpeedMpS); + } + auto [itEdge, insertedEdge] = m_edges.m_data.emplace(EdgeId(stop1Id, stop2Id, lineId), data); // There can be two identical pairs of stops on the same trip. @@ -1104,6 +1117,40 @@ void WorldFeed::FillGates() } } +bool WorldFeed::UpdateEdgeWeights() +{ + for (auto & [edgeId, edgeData] : m_edges.m_data) + { + if (edgeData.m_weight == 0) + { + auto const polyLine = m_shapes.m_data.at(edgeData.m_shapeLink.m_shapeId).m_points; + + bool const isInverted = edgeData.m_shapeLink.m_startIndex > edgeData.m_shapeLink.m_endIndex; + + size_t const startIndex = + isInverted ? edgeData.m_shapeLink.m_endIndex : edgeData.m_shapeLink.m_startIndex; + size_t const endIndex = + isInverted ? edgeData.m_shapeLink.m_startIndex : edgeData.m_shapeLink.m_endIndex; + + auto const edgePolyLine = + std::vector(polyLine.begin() + startIndex, polyLine.begin() + endIndex + 1); + CHECK_GREATER(edgePolyLine.size(), 1, ()); + + double edgeLengthM = 0.0; + + for (size_t i = 0; i < edgePolyLine.size() - 1; ++i) + edgeLengthM += mercator::DistanceOnEarth(edgePolyLine[i], edgePolyLine[i + 1]); + + if (edgeLengthM == 0.0) + return false; + + edgeData.m_weight = std::ceil(edgeLengthM / kAvgTransitSpeedMpS); + } + } + + return true; +} + bool WorldFeed::SetFeed(gtfs::Feed && feed) { m_feed = std::move(feed); @@ -1160,6 +1207,14 @@ bool WorldFeed::SetFeed(gtfs::Feed && feed) FillGates(); LOG(LINFO, ("Filled gates.")); + + if (!UpdateEdgeWeights()) + { + LOG(LWARNING, ("Found inconsistencies while updating edge weights.")); + return false; + } + + LOG(LINFO, ("Updated edges weights.")); return true; } @@ -1280,7 +1335,10 @@ void Edges::Write(IdEdgeSet const & ids, std::ofstream & stream) const ToJSONObject(*node, "line_id", edgeId.m_lineId); ToJSONObject(*node, "stop_id_from", edgeId.m_fromStopId); ToJSONObject(*node, "stop_id_to", edgeId.m_toStopId); + + CHECK_GREATER(edge.m_weight, 0, (edgeId.m_fromStopId, edgeId.m_toStopId, edgeId.m_lineId)); ToJSONObject(*node, "weight", edge.m_weight); + json_object_set_new(node.get(), "shape", ShapeLinkToJson(edge.m_shapeLink).release()); WriteJson(node.get(), stream); diff --git a/transit/world_feed/world_feed.hpp b/transit/world_feed/world_feed.hpp index bb5ef846df..c2f1091726 100644 --- a/transit/world_feed/world_feed.hpp +++ b/transit/world_feed/world_feed.hpp @@ -327,6 +327,8 @@ private: void FillTransfers(); // Fills gates based on GTFS stops. void FillGates(); + // Recalculates 0-weights of edges based on the shape length. + bool UpdateEdgeWeights(); bool ProjectStopsToShape(TransitId shapeId, std::vector & shape, IdList const & stopIds,