[routing] Quick build maxspeed section implementation.

This commit is contained in:
Vladimir Byko-Ianko 2018-10-26 18:44:59 +03:00 committed by mpimenov
parent c596afea3b
commit 59e6a501ac
6 changed files with 179 additions and 17 deletions

View file

@ -13,17 +13,17 @@ UNIT_TEST(GeoObjectId)
GeoObjectId const node(GeoObjectId::Type::ObsoleteOsmNode, 12345);
TEST_EQUAL(node.GetSerialId(), 12345ULL, ());
TEST_EQUAL(node.GetType(), GeoObjectId::Type::ObsoleteOsmNode, ());
TEST_EQUAL(DebugPrint(node), "Osm Node 12345", ());
TEST_EQUAL(DebugPrint(node), "Obsolete Osm Node 12345", ());
GeoObjectId const way(GeoObjectId::Type::ObsoleteOsmWay, 93245123456332ULL);
TEST_EQUAL(way.GetSerialId(), 93245123456332ULL, ());
TEST_EQUAL(way.GetType(), GeoObjectId::Type::ObsoleteOsmWay, ());
TEST_EQUAL(DebugPrint(way), "Osm Way 93245123456332", ());
TEST_EQUAL(DebugPrint(way), "Obsolete Osm Way 93245123456332", ());
GeoObjectId const relation(GeoObjectId::Type::ObsoleteOsmRelation, 5);
TEST_EQUAL(relation.GetSerialId(), 5ULL, ());
TEST_EQUAL(relation.GetType(), GeoObjectId::Type::ObsoleteOsmRelation, ());
TEST_EQUAL(DebugPrint(relation), "Osm Relation 5", ());
TEST_EQUAL(DebugPrint(relation), "Obsolete Osm Relation 5", ());
// 2^48 - 1, maximal possible serial id.
GeoObjectId const surrogate(GeoObjectId::Type::OsmSurrogate, 281474976710655ULL);

View file

@ -84,9 +84,9 @@ std::string DebugPrint(GeoObjectId::Type const & t)
case GeoObjectId::Type::BookingComNode: return "Booking.com";
case GeoObjectId::Type::OsmSurrogate: return "Osm Surrogate";
case GeoObjectId::Type::Fias: return "FIAS";
case GeoObjectId::Type::ObsoleteOsmNode: return "Osm Node";
case GeoObjectId::Type::ObsoleteOsmWay: return "Osm Way";
case GeoObjectId::Type::ObsoleteOsmRelation: return "Osm Relation";
case GeoObjectId::Type::ObsoleteOsmNode: return "Obsolete Osm Node";
case GeoObjectId::Type::ObsoleteOsmWay: return "Obsolete Osm Way";
case GeoObjectId::Type::ObsoleteOsmRelation: return "Obsolete Osm Relation";
}
CHECK_SWITCH();
}

View file

@ -532,7 +532,7 @@ int main(int argc, char ** argv)
{
LOG(LINFO, ("Generating maxspeed section for", datFile));
string const maxspeedFilename = genInfo.GetIntermediateFileName(MAXSPEED_FILENAME);
CHECK(BuildMaxspeed(datFile, osmToFeatureFilename, maxspeedFilename),
CHECK(routing::BuildMaxspeed(datFile, osmToFeatureFilename, maxspeedFilename),
("Generating maxspeed section error."));
}

View file

@ -1,12 +1,152 @@
#include "generator/maxspeed_builder.hpp"
#include "routing/maxspeed_conversion.hpp"
#include "generator/maxspeed_parser.hpp"
#include "generator/routing_helpers.hpp"
namespace generator
#include "routing/maxspeed_conversion.hpp"
#include "routing/maxspeed_serialization.hpp"
#include "indexer/feature.hpp"
#include "indexer/feature_data.hpp"
#include "indexer/feature_processor.hpp"
#include "coding/file_container.hpp"
#include "coding/file_writer.hpp"
#include "base/assert.hpp"
#include "base/logging.hpp"
#include "base/string_utils.hpp"
#include <algorithm>
#include <cstdint>
#include <fstream>
#include <utility>
#include "defines.hpp"
// Important note. The code below in this file will be rewritten and covered by tests within the PR.
// Now it's written quickly to make a test map build this weekend.
namespace
{
bool BuildMaxspeed(std::string const & dataPath, std::string const & osmToFeaturePath,
std::string const & maxspeedFilename)
char const kDelim[] = ", \t\r\n";
bool ParseOneSpeedValue(strings::SimpleTokenizer & iter, uint16_t & value)
{
return false;
uint64_t parsedSpeed = 0;
if (!strings::to_uint64(*iter, parsedSpeed))
return false;
if (parsedSpeed > std::numeric_limits<uint16_t>::max())
return false;
value = static_cast<uint16_t>(parsedSpeed);
++iter;
return true;
}
} // namespace generator
} // namespace
namespace routing
{
using namespace feature;
using namespace generator;
using namespace std;
// @TODO(bykoianko) Consider implement a maxspeed collector class instead of ParseMaxspeeds() and
// part of BuildMaxspeed() methods.
bool ParseMaxspeeds(string const & maxspeedFilename, OsmIdToMaxspeed & osmIdToMaxspeed)
{
osmIdToMaxspeed.clear();
ifstream stream(maxspeedFilename);
if (stream.fail())
return false;
string line;
while (std::getline(stream, line))
{
strings::SimpleTokenizer iter(line, kDelim);
if (!iter) // the line is empty
return false;
uint64_t osmId = 0;
if (!strings::to_uint64(*iter, osmId))
return false;
++iter;
ParsedMaxspeed speed;
speed.m_units = StringToUnits(*iter);
++iter;
if (!ParseOneSpeedValue(iter, speed.m_forward))
return false;
if (iter)
{
// There's backward maxspeed limit.
if (!ParseOneSpeedValue(iter, speed.m_backward))
return false;
if (iter)
return false;
}
auto const res = osmIdToMaxspeed.insert(make_pair(base::MakeOsmWay(osmId), speed));
if (!res.second)
return false;
}
return true;
}
void SerializeMaxspeed(string const & dataPath, vector<FeatureMaxspeed> && speeds)
{
LOG(LINFO, ("SerializeMaxspeed(", dataPath, ", ...) speeds size:", speeds.size()));
if (speeds.empty())
return;
LOG(LINFO, ("SerializeMaxspeed() The fisrt speed", speeds[0]));
FilesContainerW cont(dataPath, FileWriter::OP_WRITE_EXISTING);
FileWriter writer = cont.GetWriter(MAXSPEED_FILE_TAG);
MaxspeedSerializer::Serialize(speeds, writer);
}
bool BuildMaxspeed(string const & dataPath, string const & osmToFeaturePath,
string const & maxspeedFilename)
{
LOG(LINFO, ("BuildMaxspeed(", dataPath, ",", osmToFeaturePath, ",", maxspeedFilename, ")"));
OsmIdToMaxspeed osmIdToMaxspeed;
CHECK(ParseMaxspeeds(maxspeedFilename, osmIdToMaxspeed), ());
LOG(LINFO, ("BuildMaxspeed() osmIdToMaxspeed size:", osmIdToMaxspeed.size()));
map<uint32_t, base::GeoObjectId> featureIdToOsmId;
CHECK(ParseFeatureIdToOsmIdMapping(osmToFeaturePath, featureIdToOsmId), ());
LOG(LINFO, ("BuildMaxspeed() featureIdToOsmId size:", featureIdToOsmId.size()));
vector<FeatureMaxspeed> speeds;
ForEachFromDat(dataPath, [&](FeatureType & ft, uint32_t fid) {
if (!routing::IsCarRoad(TypesHolder(ft)))
return;
auto const osmIdIt = featureIdToOsmId.find(fid);
if (osmIdIt == featureIdToOsmId.cend())
return;
// @TODO Consider adding check here. |fid| should be in |featureIdToOsmId| anyway.
auto const maxspeedIt = osmIdToMaxspeed.find(osmIdIt->second);
if (maxspeedIt == osmIdToMaxspeed.cend())
return;
auto const & parsedMaxspeed = maxspeedIt->second;
// Note. It's wrong that by default if there's no maxspeed backward SpeedInUnits::m_units
// is Metric and according to this code it's be saved as Imperial for country
// with imperial metrics. Anyway this code should be rewritten before merge.
speeds.push_back(
FeatureMaxspeed(fid, SpeedInUnits(parsedMaxspeed.m_forward, parsedMaxspeed.m_units),
SpeedInUnits(parsedMaxspeed.m_backward, parsedMaxspeed.m_units)));
});
SerializeMaxspeed(dataPath, move(speeds));
return true;
}
} // namespace routing

View file

@ -1,10 +1,32 @@
#pragma once
#include "routing/maxspeed_conversion.hpp"
#include "platform/measurement_utils.hpp"
#include "base/geo_object_id.hpp"
#include <map>
#include <string>
#include <vector>
namespace generator
namespace routing
{
/// \brief Builds maxspeed section.
struct ParsedMaxspeed
{
measurement_utils::Units m_units = measurement_utils::Units::Metric;
uint16_t m_forward = routing::kInvalidSpeed;
uint16_t m_backward = routing::kInvalidSpeed;
};
using OsmIdToMaxspeed = std::map<base::GeoObjectId, ParsedMaxspeed>;
bool ParseMaxspeeds(std::string const & maxspeedFilename, OsmIdToMaxspeed & osmIdToMaxspeed);
/// \brief Write |speeds| to maxspeed section to mwm with |dataPath|.
void SerializeMaxspeed(std::string const & dataPath, std::vector<FeatureMaxspeed> && speeds);
/// \brief Builds maxspeed section in mwm with |dataPath|. This section contains max speed limits
/// if it's available.
/// \param maxspeedFilename file name to csv file with maxspeed tag values.
/// \note To start building the section, the following data must be ready:
/// 1. GenerateIntermediateData(). Saves to a file data about maxspeed tags value of road features
@ -12,4 +34,4 @@ namespace generator
/// 3. Generates geometry
bool BuildMaxspeed(std::string const & dataPath, std::string const & osmToFeaturePath,
std::string const & maxspeedFilename);
} // namespace generator
} // namespace routing

View file

@ -512,7 +512,7 @@ if [ "$MODE" == "mwm" ]; then
fi
if [ -z "$NO_REGIONS" ]; then
PARAMS_WITH_SEARCH="$PARAMS --generate_search_index --cities_boundaries_data=$CITIES_BOUNDARIES_DATA --make_city_roads"
PARAMS_WITH_SEARCH="$PARAMS --generate_search_index --cities_boundaries_data=$CITIES_BOUNDARIES_DATA --make_city_roads --generate_maxspeed"
[ -n "${SRTM_PATH-}" -a -d "${SRTM_PATH-}" ] && PARAMS_WITH_SEARCH="$PARAMS_WITH_SEARCH --srtm_path=$SRTM_PATH"
[ -f "$UGC_FILE" ] && PARAMS_WITH_SEARCH="$PARAMS_WITH_SEARCH --ugc_data=$UGC_FILE"
[ -f "$POPULAR_PLACES_FILE" ] && PARAMS_WITH_SEARCH="$PARAMS_WITH_SEARCH --popular_places_data=$POPULAR_PLACES_FILE --generate_popular_places"