#include #include #include "coordinates.hpp" #include "equal_json.hpp" #include "fixture.hpp" #include "engine/api/flatbuffers/fbresult_generated.h" #include "osrm/coordinate.hpp" #include "osrm/engine_config.hpp" #include "osrm/exception.hpp" #include "osrm/json_container.hpp" #include "osrm/osrm.hpp" #include "osrm/route_parameters.hpp" #include "osrm/status.hpp" #include #include #include #include #include #include osrm::Status run_route_json(const osrm::OSRM &osrm, const osrm::RouteParameters ¶ms, osrm::json::Object &json_result, bool use_json_only_api) { if (use_json_only_api) { return osrm.Route(params, json_result); } osrm::engine::api::ResultT result = osrm::json::Object(); auto rc = osrm.Route(params, result); json_result = result.get(); return rc; } BOOST_AUTO_TEST_SUITE(route) void test_route_same_coordinates_fixture(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params; params.steps = true; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); // unset snapping dependent hint for (auto &itr : json_result.values["waypoints"].get().values) { // Hint values aren't stable, so blank it out itr.get().values["hint"] = ""; // Round value to 6 decimal places for double comparison later itr.get().values["distance"] = round(itr.get().values["distance"].get().value * 1000000); } const auto location = json::Array{{{7.437070}, {43.749248}}}; json::Object reference{ {{"code", "Ok"}, {"waypoints", json::Array{{json::Object{{{"name", "Boulevard du Larvotto"}, {"location", location}, {"distance", round(0.137249 * 1000000)}, {"hint", ""}}}, json::Object{{{"name", "Boulevard du Larvotto"}, {"location", location}, {"distance", round(0.137249 * 1000000)}, {"hint", ""}}}}}}, {"routes", json::Array{{json::Object{ {{"distance", 0.}, {"duration", 0.}, {"weight", 0.}, {"weight_name", "routability"}, {"geometry", "yw_jGupkl@??"}, {"legs", json::Array{{json::Object{ {{"distance", 0.}, {"duration", 0.}, {"weight", 0.}, {"summary", "Boulevard du Larvotto"}, {"steps", json::Array{{{json::Object{{{"duration", 0.}, {"distance", 0.}, {"weight", 0.}, {"geometry", "yw_jGupkl@??"}, {"name", "Boulevard du Larvotto"}, {"mode", "driving"}, {"driving_side", "right"}, {"maneuver", json::Object{{ {"location", location}, {"bearing_before", 0}, {"bearing_after", 238}, {"type", "depart"}, }}}, {"intersections", json::Array{{json::Object{ {{"location", location}, {"bearings", json::Array{{238}}}, {"entry", json::Array{{json::True()}}}, {"out", 0}}}}}}}}}, json::Object{{{"duration", 0.}, {"distance", 0.}, {"weight", 0.}, {"geometry", "yw_jGupkl@"}, {"name", "Boulevard du Larvotto"}, {"mode", "driving"}, {"driving_side", "right"}, {"maneuver", json::Object{{{"location", location}, {"bearing_before", 238}, {"bearing_after", 0}, {"type", "arrive"}}}}, {"intersections", json::Array{{json::Object{ {{"location", location}, {"bearings", json::Array{{58}}}, {"entry", json::Array{{json::True()}}}, {"in", 0}}}}}} }}}}}}}}}}}}}}}}}; CHECK_EQUAL_JSON(reference, json_result); } BOOST_AUTO_TEST_CASE(test_route_same_coordinates_fixture_old_api) { test_route_same_coordinates_fixture(true); } BOOST_AUTO_TEST_CASE(test_route_same_coordinates_fixture_new_api) { test_route_same_coordinates_fixture(false); } void test_route_same_coordinates(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params; params.steps = true; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); const auto code = json_result.values.at("code").get().value; BOOST_CHECK_EQUAL(code, "Ok"); const auto &waypoints = json_result.values.at("waypoints").get().values; BOOST_CHECK(waypoints.size() == params.coordinates.size()); for (const auto &waypoint : waypoints) { const auto &waypoint_object = waypoint.get(); // nothing can be said about name, empty or contains name of the street const auto name = waypoint_object.values.at("name").get().value; BOOST_CHECK(((void)name, true)); const auto location = waypoint_object.values.at("location").get().values; const auto longitude = location[0].get().value; const auto latitude = location[1].get().value; BOOST_CHECK(longitude >= -180. && longitude <= 180.); BOOST_CHECK(latitude >= -90. && latitude <= 90.); const auto hint = waypoint_object.values.at("hint").get().value; BOOST_CHECK(!hint.empty()); } const auto &routes = json_result.values.at("routes").get().values; BOOST_REQUIRE_GT(routes.size(), 0); for (const auto &route : routes) { const auto &route_object = route.get(); const auto distance = route_object.values.at("distance").get().value; BOOST_CHECK_EQUAL(distance, 0); const auto duration = route_object.values.at("duration").get().value; BOOST_CHECK_EQUAL(duration, 0); // geometries=polyline by default const auto geometry = route_object.values.at("geometry").get().value; BOOST_CHECK(!geometry.empty()); const auto &legs = route_object.values.at("legs").get().values; BOOST_CHECK(!legs.empty()); for (const auto &leg : legs) { const auto &leg_object = leg.get(); const auto distance = leg_object.values.at("distance").get().value; BOOST_CHECK_EQUAL(distance, 0); const auto duration = leg_object.values.at("duration").get().value; BOOST_CHECK_EQUAL(duration, 0); // nothing can be said about summary, empty or contains human readable summary const auto summary = leg_object.values.at("summary").get().value; BOOST_CHECK(((void)summary, true)); const auto &steps = leg_object.values.at("steps").get().values; BOOST_CHECK(!steps.empty()); std::size_t step_count = 0; for (const auto &step : steps) { const auto &step_object = step.get(); const auto distance = step_object.values.at("distance").get().value; BOOST_CHECK_EQUAL(distance, 0); const auto duration = step_object.values.at("duration").get().value; BOOST_CHECK_EQUAL(duration, 0); // geometries=polyline by default const auto geometry = step_object.values.at("geometry").get().value; BOOST_CHECK(!geometry.empty()); // nothing can be said about name, empty or contains way name const auto name = step_object.values.at("name").get().value; BOOST_CHECK(((void)name, true)); // nothing can be said about mode, contains mode of transportation const auto mode = step_object.values.at("mode").get().value; BOOST_CHECK(!name.empty()); const auto &maneuver = step_object.values.at("maneuver").get().values; const auto type = maneuver.at("type").get().value; BOOST_CHECK(!type.empty()); const auto &intersections = step_object.values.at("intersections").get().values; for (auto &intersection : intersections) { const auto &intersection_object = intersection.get().values; const auto location = intersection_object.at("location").get().values; const auto longitude = location[0].get().value; const auto latitude = location[1].get().value; BOOST_CHECK(longitude >= -180. && longitude <= 180.); BOOST_CHECK(latitude >= -90. && latitude <= 90.); const auto &bearings = intersection_object.at("bearings").get().values; BOOST_CHECK(!bearings.empty()); const auto &entries = intersection_object.at("entry").get().values; BOOST_CHECK(bearings.size() == entries.size()); for (const auto bearing : bearings) BOOST_CHECK(0. <= bearing.get().value && bearing.get().value <= 360.); if (step_count > 0) { const auto in = intersection_object.at("in").get().value; BOOST_CHECK(in < bearings.size()); } if (step_count + 1 < steps.size()) { const auto out = intersection_object.at("out").get().value; BOOST_CHECK(out < bearings.size()); } } // modifier is optional // TODO(daniel-j-h): // exit is optional // TODO(daniel-j-h): ++step_count; } } } } BOOST_AUTO_TEST_CASE(test_route_same_coordinates_old_api) { test_route_same_coordinates(true); } BOOST_AUTO_TEST_CASE(test_route_same_coordinates_new_api) { test_route_same_coordinates(false); } void test_route_same_coordinates_no_waypoints(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params; params.skip_waypoints = true; params.steps = true; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); const auto code = json_result.values.at("code").get().value; BOOST_CHECK_EQUAL(code, "Ok"); BOOST_CHECK(json_result.values.find("waypoints") == json_result.values.end()); const auto &routes = json_result.values.at("routes").get().values; BOOST_REQUIRE_GT(routes.size(), 0); for (const auto &route : routes) { const auto &route_object = route.get(); const auto distance = route_object.values.at("distance").get().value; BOOST_CHECK_EQUAL(distance, 0); const auto duration = route_object.values.at("duration").get().value; BOOST_CHECK_EQUAL(duration, 0); // geometries=polyline by default const auto geometry = route_object.values.at("geometry").get().value; BOOST_CHECK(!geometry.empty()); const auto &legs = route_object.values.at("legs").get().values; BOOST_CHECK(!legs.empty()); // The rest of legs contents is verified by test_route_same_coordinates } } BOOST_AUTO_TEST_CASE(test_route_same_coordinates_no_waypoints_old_api) { test_route_same_coordinates_no_waypoints(true); } BOOST_AUTO_TEST_CASE(test_route_same_coordinates_no_waypoints_new_api) { test_route_same_coordinates_no_waypoints(false); } void test_route_response_for_locations_in_small_component(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; const auto locations = get_locations_in_small_component(); RouteParameters params; params.coordinates.push_back(locations.at(0)); params.coordinates.push_back(locations.at(1)); params.coordinates.push_back(locations.at(2)); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); const auto code = json_result.values.at("code").get().value; BOOST_CHECK_EQUAL(code, "Ok"); const auto &waypoints = json_result.values.at("waypoints").get().values; BOOST_CHECK_EQUAL(waypoints.size(), params.coordinates.size()); for (const auto &waypoint : waypoints) { const auto &waypoint_object = waypoint.get(); const auto location = waypoint_object.values.at("location").get().values; const auto longitude = location[0].get().value; const auto latitude = location[1].get().value; BOOST_CHECK(longitude >= -180. && longitude <= 180.); BOOST_CHECK(latitude >= -90. && latitude <= 90.); } } BOOST_AUTO_TEST_CASE(test_route_response_for_locations_in_small_component_old_api) { test_route_response_for_locations_in_small_component(true); } BOOST_AUTO_TEST_CASE(test_route_response_for_locations_in_small_component_new_api) { test_route_response_for_locations_in_small_component(false); } void test_route_response_for_locations_in_big_component(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; const auto locations = get_locations_in_big_component(); RouteParameters params; params.coordinates.push_back(locations.at(0)); params.coordinates.push_back(locations.at(1)); params.coordinates.push_back(locations.at(2)); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); const auto code = json_result.values.at("code").get().value; BOOST_CHECK_EQUAL(code, "Ok"); const auto &waypoints = json_result.values.at("waypoints").get().values; BOOST_CHECK_EQUAL(waypoints.size(), params.coordinates.size()); for (const auto &waypoint : waypoints) { const auto &waypoint_object = waypoint.get(); const auto location = waypoint_object.values.at("location").get().values; const auto longitude = location[0].get().value; const auto latitude = location[1].get().value; BOOST_CHECK(longitude >= -180. && longitude <= 180.); BOOST_CHECK(latitude >= -90. && latitude <= 90.); } } BOOST_AUTO_TEST_CASE(test_route_response_for_locations_in_big_component_old_api) { test_route_response_for_locations_in_big_component(true); } BOOST_AUTO_TEST_CASE(test_route_response_for_locations_in_big_component_new_api) { test_route_response_for_locations_in_big_component(false); } void test_route_response_for_locations_across_components(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; const auto big_component = get_locations_in_big_component(); const auto small_component = get_locations_in_small_component(); RouteParameters params; params.coordinates.push_back(small_component.at(0)); params.coordinates.push_back(big_component.at(0)); params.coordinates.push_back(small_component.at(1)); params.coordinates.push_back(big_component.at(1)); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); const auto code = json_result.values.at("code").get().value; BOOST_CHECK_EQUAL(code, "Ok"); const auto &waypoints = json_result.values.at("waypoints").get().values; BOOST_CHECK_EQUAL(waypoints.size(), params.coordinates.size()); for (const auto &waypoint : waypoints) { const auto &waypoint_object = waypoint.get(); const auto location = waypoint_object.values.at("location").get().values; const auto longitude = location[0].get().value; const auto latitude = location[1].get().value; BOOST_CHECK(longitude >= -180. && longitude <= 180.); BOOST_CHECK(latitude >= -90. && latitude <= 90.); } } BOOST_AUTO_TEST_CASE(test_route_response_for_locations_across_components_old_api) { test_route_response_for_locations_across_components(true); } BOOST_AUTO_TEST_CASE(test_route_response_for_locations_across_components_new_api) { test_route_response_for_locations_across_components(false); } void test_route_user_disables_generating_hints(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params; params.steps = true; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); params.generate_hints = false; json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); for (auto waypoint : json_result.values["waypoints"].get().values) BOOST_CHECK_EQUAL(waypoint.get().values.count("hint"), 0); } BOOST_AUTO_TEST_CASE(test_route_user_disables_generating_hints_old_api) { test_route_user_disables_generating_hints(true); } BOOST_AUTO_TEST_CASE(test_route_user_disables_generating_hints_new_api) { test_route_user_disables_generating_hints(false); } void speed_annotation_matches_duration_and_distance(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params; params.annotations_type = RouteParameters::AnnotationsType::Duration | RouteParameters::AnnotationsType::Distance | RouteParameters::AnnotationsType::Speed; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); const auto &routes = json_result.values["routes"].get().values; const auto &legs = routes[0].get().values.at("legs").get().values; const auto &annotation = legs[0].get().values.at("annotation").get(); const auto &speeds = annotation.values.at("speed").get().values; const auto &durations = annotation.values.at("duration").get().values; const auto &distances = annotation.values.at("distance").get().values; int length = speeds.size(); BOOST_CHECK_EQUAL(length, 1); for (int i = 0; i < length; i++) { auto speed = speeds[i].get().value; auto duration = durations[i].get().value; auto distance = distances[i].get().value; auto calc = std::round(distance / duration * 10.) / 10.; BOOST_CHECK_EQUAL(speed, std::isnan(calc) ? 0 : calc); // Because we route from/to the same location, all annotations should be 0; BOOST_CHECK_EQUAL(speed, 0); BOOST_CHECK_EQUAL(distance, 0); BOOST_CHECK_EQUAL(duration, 0); } } BOOST_AUTO_TEST_CASE(speed_annotation_matches_duration_and_distance_old_api) { speed_annotation_matches_duration_and_distance(true); } BOOST_AUTO_TEST_CASE(speed_annotation_matches_duration_and_distance_new_api) { speed_annotation_matches_duration_and_distance(false); } void test_manual_setting_of_annotations_property(bool use_json_only_api) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params{}; params.annotations = true; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, use_json_only_api); BOOST_CHECK(rc == Status::Ok); const auto code = json_result.values.at("code").get().value; BOOST_CHECK_EQUAL(code, "Ok"); auto annotations = json_result.values["routes"] .get() .values[0] .get() .values["legs"] .get() .values[0] .get() .values["annotation"] .get() .values; BOOST_CHECK_EQUAL(annotations.size(), 7); } BOOST_AUTO_TEST_CASE(test_manual_setting_of_annotations_property_old_api) { test_manual_setting_of_annotations_property(true); } BOOST_AUTO_TEST_CASE(test_manual_setting_of_annotations_property_new_api) { test_manual_setting_of_annotations_property(false); } using NodePair = std::pair; using NodePairToWayIDMap = std::map; NodePairToWayIDMap read_node_pair_to_way_id_map(osmium::io::Reader &osm) { struct H : public osmium::handler::Handler { NodePairToWayIDMap ret; void way(const osmium::Way &way) { osmium::unsigned_object_id_type first = 0; for (const auto &n : way.nodes()) { const auto second = n.positive_ref(); if (first != 0) { ret[{first, second}] = way.id(); } first = second; } } } handler; osmium::apply(osm, handler); return std::move(handler.ret); } using LonLat = std::pair; using LonLatVector = std::vector; LonLatVector check_route_annotated_ways(std::vector &coordinates, osrm::OSRM &osrm, NodePairToWayIDMap &node_pair_to_way_id_map, bool use_steps) { LonLatVector ret; using namespace osrm; (void)node_pair_to_way_id_map; RouteParameters params{}; params.annotations_type = RouteParameters::AnnotationsType::Nodes | RouteParameters::AnnotationsType::Ways; params.steps = use_steps; params.geometries = engine::api::RouteParameters::GeometriesType::GeoJSON; params.coordinates = coordinates; json::Object json_result; const auto rc = run_route_json(osrm, params, json_result, true); BOOST_CHECK(rc == Status::Ok); const auto code = json_result.values.at("code").get().value; BOOST_CHECK_EQUAL(code, "Ok"); auto routes = json_result.values["routes"].get().values; BOOST_CHECK_EQUAL(routes.size(), 1); auto route = routes[0]; auto geom = route.get() .values["geometry"] .get() .values["coordinates"] .get() .values; auto legs = route.get().values["legs"].get().values; for (auto leg : legs) { if (use_steps) { auto steps = leg.get().values["steps"].get().values; for (auto step : steps) { auto geom = step.get() .values["geometry"] .get() .values["coordinates"] .get() .values; for (auto gleg : geom) { auto p = gleg.get().values; auto lon = p[0].get().value; auto lat = p[1].get().value; ret.push_back(LonLat{lon, lat}); } } } auto annotations = leg.get().values["annotation"].get().values; BOOST_CHECK_EQUAL(annotations.size(), 2); auto nodes = annotations["nodes"].get().values; auto ways = annotations["ways"].get().values; BOOST_CHECK_GT(nodes.size(), 1); BOOST_CHECK_EQUAL(nodes.size() - 1, ways.size()); auto nodes_it = nodes.cbegin(); auto ways_it = ways.cbegin(); osmium::unsigned_object_id_type first = nodes_it->get().value; for (nodes_it++; nodes_it != nodes.cend(); nodes_it++, ways_it++) { osmium::unsigned_object_id_type second = nodes_it->get().value; int64_t way_id = ways_it->get().value; auto found = node_pair_to_way_id_map.find(NodePair(first, second)); auto reverse = false; if (found == node_pair_to_way_id_map.end()) { reverse = true; found = node_pair_to_way_id_map.find(NodePair(second, first)); } BOOST_CHECK_MESSAGE(found != node_pair_to_way_id_map.end(), "The node pair not found: " << first << "<->" << second); int64_t found_way_id = reverse ? -found->second : found->second; BOOST_CHECK_MESSAGE(found_way_id == way_id, "The node pair way doesn't correspond: " << first << "<->" << second << "=" << found_way_id << "=?=" << way_id); first = second; } } return ret; } BOOST_AUTO_TEST_CASE(test_route_annotated_ways) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); osmium::io::Reader osm(OSRM_TEST_DATA_DIR "/monaco.osm.pbf"); NodePairToWayIDMap node_pair_to_way_id_map = read_node_pair_to_way_id_map(osm); auto coordinates = get_split_trace_locations(); BOOST_TEST_MESSAGE("split_trace_locations without steps"); check_route_annotated_ways(coordinates, osrm, node_pair_to_way_id_map, false); BOOST_TEST_MESSAGE("split_trace_locations with steps"); check_route_annotated_ways(coordinates, osrm, node_pair_to_way_id_map, true); coordinates = get_locations_in_big_component(); // auto coords = BOOST_TEST_MESSAGE("locations_in_big_component without steps"); check_route_annotated_ways(coordinates, osrm, node_pair_to_way_id_map, false); BOOST_TEST_MESSAGE("locations_in_big_component with steps"); auto coords = check_route_annotated_ways(coordinates, osrm, node_pair_to_way_id_map, true); auto c1 = coords.cbegin(), c2 = coords.cend(); for (c1++, c2--; c1 != coords.cend() && c2 != coords.cbegin(); c1++, c2--) { if (c1 == c2) continue; coordinates = Locations{{Longitude{c1->first}, Latitude{c1->second}}, {Longitude{c2->first}, Latitude{c2->second}}}; BOOST_TEST_MESSAGE("Checking: <" << osrm::util::toFloating(coordinates[0].lat) << ":" << osrm::util::toFloating(coordinates[0].lon) << "> -> <" << osrm::util::toFloating(coordinates[1].lat) << ":" << osrm::util::toFloating(coordinates[1].lon) << ">"); check_route_annotated_ways(coordinates, osrm, node_pair_to_way_id_map, false); check_route_annotated_ways(coordinates, osrm, node_pair_to_way_id_map, true); } } BOOST_AUTO_TEST_CASE(test_route_serialize_fb) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params; params.steps = true; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); engine::api::ResultT result = flatbuffers::FlatBufferBuilder(); const auto rc = osrm.Route(params, result); BOOST_CHECK(rc == Status::Ok); auto &fb_result = result.get(); auto fb = engine::api::fbresult::GetFBResult(fb_result.GetBufferPointer()); BOOST_CHECK(!fb->error()); BOOST_CHECK(fb->waypoints() != nullptr); const auto waypoints = fb->waypoints(); BOOST_CHECK(waypoints->size() == params.coordinates.size()); for (const auto &waypoint : *waypoints) { const auto longitude = waypoint->location()->longitude(); const auto latitude = waypoint->location()->latitude(); BOOST_CHECK(longitude >= -180. && longitude <= 180.); BOOST_CHECK(latitude >= -90. && latitude <= 90.); BOOST_CHECK(!waypoint->hint()->str().empty()); } BOOST_CHECK(fb->routes() != nullptr); const auto routes = fb->routes(); BOOST_REQUIRE_GT(routes->size(), 0); for (const auto &route : *routes) { BOOST_CHECK_EQUAL(route->distance(), 0); BOOST_CHECK_EQUAL(route->duration(), 0); const auto &legs = route->legs(); BOOST_CHECK(legs->size() > 0); for (const auto &leg : *legs) { BOOST_CHECK_EQUAL(leg->distance(), 0); BOOST_CHECK_EQUAL(leg->duration(), 0); BOOST_CHECK(leg->steps() != nullptr); const auto steps = leg->steps(); BOOST_CHECK(steps->size() > 0); std::size_t step_count = 0; for (const auto step : *steps) { BOOST_CHECK_EQUAL(step->distance(), 0); BOOST_CHECK_EQUAL(step->duration(), 0); BOOST_CHECK(step->maneuver() != nullptr); BOOST_CHECK(step->intersections() != nullptr); const auto intersections = step->intersections(); for (auto intersection : *intersections) { const auto longitude = intersection->location()->longitude(); const auto latitude = intersection->location()->latitude(); BOOST_CHECK(longitude >= -180. && longitude <= 180.); BOOST_CHECK(latitude >= -90. && latitude <= 90.); BOOST_CHECK(intersection->bearings() != nullptr); const auto bearings = intersection->bearings(); BOOST_CHECK(bearings->size() > 0); for (const auto bearing : *bearings) BOOST_CHECK(0. <= bearing && bearing <= 360.); if (step_count > 0) { BOOST_CHECK(intersection->in_bearing() < bearings->size()); } if (step_count + 1 < steps->size()) { BOOST_CHECK(intersection->out_bearing() < bearings->size()); } } ++step_count; } } } } BOOST_AUTO_TEST_CASE(test_route_serialize_fb_skip_waypoints) { auto osrm = getOSRM(OSRM_TEST_DATA_DIR "/ch/monaco.osrm"); using namespace osrm; RouteParameters params; params.skip_waypoints = true; params.steps = true; params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); params.coordinates.push_back(get_dummy_location()); engine::api::ResultT result = flatbuffers::FlatBufferBuilder(); const auto rc = osrm.Route(params, result); BOOST_CHECK(rc == Status::Ok); auto &fb_result = result.get(); auto fb = engine::api::fbresult::GetFBResult(fb_result.GetBufferPointer()); BOOST_CHECK(!fb->error()); BOOST_CHECK(fb->waypoints() == nullptr); BOOST_CHECK(fb->routes() != nullptr); const auto routes = fb->routes(); BOOST_REQUIRE_GT(routes->size(), 0); for (const auto &route : *routes) { BOOST_CHECK_EQUAL(route->distance(), 0); BOOST_CHECK_EQUAL(route->duration(), 0); const auto &legs = route->legs(); BOOST_CHECK(legs->size() > 0); // Rest of the content is verified by test_route_serialize_fb } } BOOST_AUTO_TEST_SUITE_END()