forked from organicmaps/organicmaps
Integration tests implementation
This commit is contained in:
parent
5ebba22d26
commit
33e6349c12
9 changed files with 411 additions and 1 deletions
32
integration_tests/integration_tests.pro
Normal file
32
integration_tests/integration_tests.pro
Normal file
|
@ -0,0 +1,32 @@
|
|||
# This subproject implements integration tests.
|
||||
# This tests are launched on the whole world dataset.
|
||||
|
||||
# It is recommended to place tests here in the following cases:
|
||||
# - tests are written to be launch on the whole world dataset;
|
||||
# - tests covers significant number of subsystems;
|
||||
|
||||
TARGET = integration_tests
|
||||
CONFIG += console warn_on
|
||||
CONFIG -= app_bundle
|
||||
TEMPLATE = app
|
||||
|
||||
ROOT_DIR = ../
|
||||
DEPENDENCIES = map routing search storage indexer platform geometry coding base osrm jansson protobuf tomcrypt
|
||||
|
||||
macx-*: LIBS *= "-framework Foundation" "-framework IOKit"
|
||||
|
||||
include($$ROOT_DIR/common.pri)
|
||||
|
||||
QT *= core
|
||||
|
||||
win32* : LIBS *= -lShell32
|
||||
|
||||
|
||||
SOURCES += \
|
||||
../testing/testingmain.cpp \
|
||||
osrm_route_test.cpp \
|
||||
osrm_turn_test.cpp \
|
||||
osrm_test_tools.cpp \
|
||||
|
||||
HEADERS += \
|
||||
osrm_test_tools.hpp \
|
22
integration_tests/osrm_route_test.cpp
Normal file
22
integration_tests/osrm_route_test.cpp
Normal file
|
@ -0,0 +1,22 @@
|
|||
#include "../../testing/testing.hpp"
|
||||
|
||||
#include "osrm_test_tools.hpp"
|
||||
|
||||
using namespace routing;
|
||||
|
||||
namespace
|
||||
{
|
||||
UNIT_TEST(RussiaMoscowLenigradskiy39GerPanfilovtsev22RouteTest)
|
||||
{
|
||||
shared_ptr<integration::OsrmRouterComponents> routerComponents = integration::GetAllMaps();
|
||||
TEST(integration::CalculateRouteAndTestRouteLength(routerComponents, {37.53758809983519, 67.536162466434234},
|
||||
{0., 0.}, {37.40993977728661, 67.644784047393685}, 14296.), ());
|
||||
}
|
||||
|
||||
UNIT_TEST(RussiaMoscowGerPanfilovtsev22SolodshaPravdiRouteTest)
|
||||
{
|
||||
shared_ptr<integration::OsrmRouterComponents> routerComponents = integration::GetAllMaps();
|
||||
TEST(integration::CalculateRouteAndTestRouteLength(routerComponents, {37.409929478750627, 67.644798619710073},
|
||||
{0., 0.}, {39.836562407458047, 65.774372510437971}, 253275.), ());
|
||||
}
|
||||
}
|
221
integration_tests/osrm_test_tools.cpp
Normal file
221
integration_tests/osrm_test_tools.cpp
Normal file
|
@ -0,0 +1,221 @@
|
|||
#include "osrm_test_tools.hpp"
|
||||
|
||||
#include "../std/new.hpp"
|
||||
|
||||
#include "../indexer/index.hpp"
|
||||
|
||||
#include "../geometry/distance_on_sphere.hpp"
|
||||
|
||||
#include "../routing/route.hpp"
|
||||
|
||||
#include "../map/feature_vec_model.hpp"
|
||||
|
||||
#include "../platform/platform.hpp"
|
||||
#include "../platform/preferred_languages.hpp"
|
||||
|
||||
#include "../search/search_engine.hpp"
|
||||
|
||||
|
||||
using namespace routing;
|
||||
|
||||
namespace integration
|
||||
{
|
||||
class OsrmRouterWrapper : public OsrmRouter
|
||||
{
|
||||
public:
|
||||
OsrmRouterWrapper(Index const * index, CountryFileFnT const & fn) :
|
||||
OsrmRouter(index, fn) {}
|
||||
ResultCode SyncCalculateRoute(m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt, Route & route)
|
||||
{
|
||||
m_startPt = startPt;
|
||||
m_startDr = startDr;
|
||||
m_finalPt = finalPt;
|
||||
m_cachedFinalNodes.clear();
|
||||
m_isFinalChanged = false;
|
||||
m_requestCancel = false;
|
||||
|
||||
return OsrmRouter::CalculateRouteImpl(startPt, startDr, finalPt, route);
|
||||
}
|
||||
};
|
||||
|
||||
shared_ptr<search::Engine> CreateSearchEngine(search::Engine::IndexType const * pIndex)
|
||||
{
|
||||
ASSERT(pIndex, ());
|
||||
Platform const & pl = GetPlatform();
|
||||
try
|
||||
{
|
||||
shared_ptr<search::Engine> searchEngine(new search::Engine(
|
||||
pIndex,
|
||||
pl.GetReader(SEARCH_CATEGORIES_FILE_NAME),
|
||||
pl.GetReader(PACKED_POLYGONS_FILE),
|
||||
pl.GetReader(COUNTRIES_FILE),
|
||||
languages::GetCurrentOrig()));
|
||||
return searchEngine;
|
||||
}
|
||||
catch (RootException const &)
|
||||
{
|
||||
ASSERT(false, ());
|
||||
return nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
shared_ptr<model::FeaturesFetcher> CreateFeaturesFetcher(vector<string> const & mapNames)
|
||||
{
|
||||
shared_ptr<model::FeaturesFetcher> featuresFetcher(new model::FeaturesFetcher);
|
||||
featuresFetcher->InitClassificator();
|
||||
|
||||
for (auto const mapName : mapNames)
|
||||
{
|
||||
if (featuresFetcher->AddMap(mapName) == -1)
|
||||
{
|
||||
ASSERT(false, ());
|
||||
return nullptr;
|
||||
}
|
||||
}
|
||||
return featuresFetcher;
|
||||
}
|
||||
|
||||
shared_ptr<search::Engine> CreateSearchEngine(shared_ptr<model::FeaturesFetcher> featuresFetcher)
|
||||
{
|
||||
shared_ptr<search::Engine> searchEngine = CreateSearchEngine(&featuresFetcher->GetIndex());
|
||||
if (!searchEngine.get())
|
||||
ASSERT(false, ());
|
||||
return searchEngine;
|
||||
}
|
||||
|
||||
shared_ptr<OsrmRouterWrapper> CreateOsrmRouter(shared_ptr<model::FeaturesFetcher> featuresFetcher,
|
||||
shared_ptr<search::Engine> searchEngine)
|
||||
{
|
||||
shared_ptr<OsrmRouterWrapper> osrmRouter(new OsrmRouterWrapper(&featuresFetcher->GetIndex(),
|
||||
[searchEngine] (m2::PointD const & pt)
|
||||
{
|
||||
return searchEngine->GetCountryFile(pt);
|
||||
}));
|
||||
return osrmRouter;
|
||||
}
|
||||
|
||||
class OsrmRouterComponents
|
||||
{
|
||||
public:
|
||||
OsrmRouterComponents(vector<string> const & mapNames) :
|
||||
m_featuresFetcher(CreateFeaturesFetcher(mapNames)),
|
||||
m_searchEngine(CreateSearchEngine(m_featuresFetcher)),
|
||||
m_osrmRouter(CreateOsrmRouter(m_featuresFetcher, m_searchEngine)) {}
|
||||
OsrmRouterWrapper * GetOsrmRouter() const { return m_osrmRouter.get(); }
|
||||
private:
|
||||
shared_ptr<model::FeaturesFetcher> m_featuresFetcher;
|
||||
shared_ptr<search::Engine> m_searchEngine;
|
||||
shared_ptr<OsrmRouterWrapper> m_osrmRouter;
|
||||
};
|
||||
|
||||
void GetMapNames(vector<string> & maps)
|
||||
{
|
||||
Platform const & pl = GetPlatform();
|
||||
|
||||
pl.GetFilesByExt(pl.ResourcesDir(), DATA_FILE_EXTENSION, maps);
|
||||
pl.GetFilesByExt(pl.WritableDir(), DATA_FILE_EXTENSION, maps);
|
||||
|
||||
sort(maps.begin(), maps.end());
|
||||
maps.erase(unique(maps.begin(), maps.end()), maps.end());
|
||||
}
|
||||
|
||||
shared_ptr<OsrmRouterComponents> LoadMaps(vector<string> const & mapNames)
|
||||
{
|
||||
try{
|
||||
return shared_ptr<OsrmRouterComponents>(new OsrmRouterComponents(mapNames));
|
||||
}
|
||||
catch(bad_alloc &)
|
||||
{
|
||||
ASSERT(false, ());
|
||||
return nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
shared_ptr<OsrmRouterComponents> LoadAllMaps()
|
||||
{
|
||||
vector<string> maps;
|
||||
GetMapNames(maps);
|
||||
ASSERT(!maps.empty(), ());
|
||||
return LoadMaps(maps);
|
||||
}
|
||||
|
||||
shared_ptr<OsrmRouterComponents> GetAllMaps()
|
||||
{
|
||||
static shared_ptr<OsrmRouterComponents> inst = LoadAllMaps();
|
||||
ASSERT(inst.get(), ());
|
||||
return inst;
|
||||
}
|
||||
|
||||
RouteResultT CalculateRoute(shared_ptr<OsrmRouterComponents> routerComponents,
|
||||
m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt)
|
||||
{
|
||||
try
|
||||
{
|
||||
ASSERT(routerComponents.get(), ());
|
||||
OsrmRouterWrapper * osrmRouter = routerComponents->GetOsrmRouter();
|
||||
ASSERT(osrmRouter, ());
|
||||
shared_ptr<Route> route(new Route("mapsme"));
|
||||
OsrmRouter::ResultCode result = osrmRouter->SyncCalculateRoute(startPt, startDr, finalPt, *route.get());
|
||||
return RouteResultT(route, result);
|
||||
}
|
||||
catch(bad_alloc &)
|
||||
{
|
||||
ASSERT(false, ());
|
||||
return RouteResultT(nullptr, OsrmRouter::InternalError);
|
||||
}
|
||||
}
|
||||
|
||||
bool TestTurn(shared_ptr<Route> const route, uint32_t etalonTurnNumber, m2::PointD const & etalonTurnPnt,
|
||||
turns::TurnDirection etalonTurnDirection, uint32_t etalonRoundAboutExitNum)
|
||||
{
|
||||
ASSERT(route.get(), ());
|
||||
turns::TurnsGeomT const & turnsGeom = route->GetTurnsGeometry();
|
||||
if (etalonTurnNumber >= turnsGeom.size())
|
||||
return false;
|
||||
|
||||
Route::TurnsT const & turns = route->GetTurns();
|
||||
ASSERT_LESS(etalonTurnNumber, turns.size(), ());
|
||||
Route::TurnItem const & turn = turns[etalonTurnNumber];
|
||||
if (turn.m_turn != etalonTurnDirection)
|
||||
return false;
|
||||
if (turn.m_exitNum != etalonRoundAboutExitNum)
|
||||
return false;
|
||||
|
||||
turns::TurnGeom const & turnGeom = turnsGeom[etalonTurnNumber];
|
||||
ASSERT_LESS(turnGeom.m_turnIndex, turnGeom.m_points.size(), ());
|
||||
m2::PointD turnGeomPnt = turnGeom.m_points[turnGeom.m_turnIndex];
|
||||
|
||||
double const dist = ms::DistanceOnEarth(etalonTurnPnt.y, etalonTurnPnt.x, turnGeomPnt.y, turnGeomPnt.x);
|
||||
if (dist > turnInaccuracy)
|
||||
return false;
|
||||
return true;
|
||||
}
|
||||
|
||||
bool TestTurnCount(shared_ptr<routing::Route> const route, uint32_t etalonTurnCount)
|
||||
{
|
||||
ASSERT(route.get(), ());
|
||||
return route->GetTurnsGeometry().size() == etalonTurnCount;
|
||||
}
|
||||
|
||||
bool TestRouteLength(shared_ptr<Route> const route, double etalonRouteLength)
|
||||
{
|
||||
ASSERT(route.get(), ());
|
||||
double const delta = etalonRouteLength * routeLengthInaccurace;
|
||||
double const routeLength = route->GetDistance();
|
||||
if (routeLength - delta <= etalonRouteLength && routeLength + delta >= etalonRouteLength)
|
||||
return true;
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
bool CalculateRouteAndTestRouteLength(shared_ptr<OsrmRouterComponents> routerComponents, m2::PointD const & startPt,
|
||||
m2::PointD const & startDr, m2::PointD const & finalPt, double etalonRouteLength)
|
||||
{
|
||||
RouteResultT routeResult = CalculateRoute(routerComponents, startPt, startDr, finalPt);
|
||||
shared_ptr<Route> const route = routeResult.first;
|
||||
OsrmRouter::ResultCode const result = routeResult.second;
|
||||
if (result != OsrmRouter::NoError)
|
||||
return false;
|
||||
return TestRouteLength(route, etalonRouteLength);
|
||||
}
|
||||
}
|
58
integration_tests/osrm_test_tools.hpp
Normal file
58
integration_tests/osrm_test_tools.hpp
Normal file
|
@ -0,0 +1,58 @@
|
|||
#pragma once
|
||||
|
||||
#include "../std/shared_ptr.hpp"
|
||||
#include "../std/string.hpp"
|
||||
#include "../std/vector.hpp"
|
||||
|
||||
#include "../routing/osrm_router.hpp"
|
||||
|
||||
/*
|
||||
* These tests are developed to simplify routing integration tests writing.
|
||||
* You can use the interface bellow however you want but there are some hints.
|
||||
* 1. Most likely you want to use GetAllMaps() to get shared_ptr<OsrmRouterComponents>.
|
||||
* It loads all the maps from directories Platform::WritableDir()
|
||||
* and Platform::ResourcesDir() only once and then reuse it.
|
||||
* Use LoadMaps() only if you want to test something on a special map set.
|
||||
* 2. Loading maps and calculating routes is a time consumption process.
|
||||
* Do this only if you really need it.
|
||||
* 3. If you want to check that a turn is absent use TestTurnCount.
|
||||
* 4. The easiest way to gather all the information for writing an integration test is
|
||||
* - to put a break point in OsrmRouter::CalculateRouteImpl;
|
||||
* - to make a route with MapWithMe desktop application;
|
||||
* - to get all necessary parameters and result of route calculation;
|
||||
* - to place them into the test you're writing.
|
||||
* 5. The recommended way for naming tests for route from one place to another one is
|
||||
* <Country><City><Street1><House1><Street2><House2><Test time. TurnTest or RouteTest for the time being>
|
||||
*/
|
||||
|
||||
/// Inaccuracy of turn point in meters
|
||||
double const turnInaccuracy = 2.;
|
||||
/// Inaccurace of the route length. It is used for checking if routes have etalon(sample) length.
|
||||
/// So the a created route has etalon length if
|
||||
/// etalonLength - etalonRouteLength * routeLengthInaccurace <= route->GetDistance()
|
||||
/// && etalonRouteLength + etalonRouteLength * routeLengthInaccurace >= route->GetDistance()
|
||||
/// is equal to true.
|
||||
double const routeLengthInaccurace = .01;
|
||||
|
||||
typedef pair<shared_ptr<routing::Route>, routing::OsrmRouter::ResultCode> RouteResultT;
|
||||
|
||||
namespace integration
|
||||
{
|
||||
class OsrmRouterComponents;
|
||||
|
||||
shared_ptr<OsrmRouterComponents> GetAllMaps();
|
||||
shared_ptr<OsrmRouterComponents> LoadMaps(vector<string> const & mapNames);
|
||||
|
||||
RouteResultT CalculateRoute(shared_ptr<OsrmRouterComponents> routerComponents,
|
||||
m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt);
|
||||
|
||||
bool TestTurn(shared_ptr<routing::Route> const route, uint32_t etalonTurnNumber, m2::PointD const & etalonTurnPnt,
|
||||
routing::turns::TurnDirection etalonTurnDirection, uint32_t etalonRoundAboutExitNum = 0);
|
||||
|
||||
bool TestTurnCount(shared_ptr<routing::Route> const route, uint32_t etalonTurnCount);
|
||||
|
||||
bool TestRouteLength(shared_ptr<routing::Route> const route, double etalonRouteLength);
|
||||
|
||||
bool CalculateRouteAndTestRouteLength(shared_ptr<OsrmRouterComponents> routerComponents, m2::PointD const & startPt,
|
||||
m2::PointD const & startDr, m2::PointD const & finalPt, double etalonRouteLength);
|
||||
}
|
50
integration_tests/osrm_turn_test.cpp
Normal file
50
integration_tests/osrm_turn_test.cpp
Normal file
|
@ -0,0 +1,50 @@
|
|||
#include "../../testing/testing.hpp"
|
||||
|
||||
#include "osrm_test_tools.hpp"
|
||||
|
||||
#include "../routing/route.hpp"
|
||||
|
||||
using namespace routing;
|
||||
|
||||
UNIT_TEST(RussiaMoscowLenigradskiy39UturnTurnTest)
|
||||
{
|
||||
shared_ptr<integration::OsrmRouterComponents> routerComponents = integration::GetAllMaps();
|
||||
|
||||
RouteResultT const routeResult = integration::CalculateRoute(routerComponents, {37.537544383032568, 67.536216737893028},
|
||||
{0., 0.}, {37.538908531885973, 67.54544090660923});
|
||||
|
||||
shared_ptr<Route> const route = routeResult.first;
|
||||
OsrmRouter::ResultCode const result = routeResult.second;
|
||||
TEST_EQUAL(result, OsrmRouter::NoError, ());
|
||||
|
||||
TEST(integration::TestTurn(route, 0, {37.545981835916507, 67.530713137468041}, turns::TurnLeft), ());
|
||||
TEST(integration::TestTurn(route, 1, {37.546738218864334, 67.531659957257347}, turns::TurnLeft), ());
|
||||
TEST(integration::TestTurn(route, 2, {37.539925407915746, 67.537083383925875}, turns::TurnRight), ());
|
||||
TEST(!integration::TestTurn(route, 2, {37., 67.}, turns::TurnRight), ());
|
||||
|
||||
TEST(integration::TestRouteLength(route, 2033.), ());
|
||||
TEST(!integration::TestRouteLength(route, 2533.), ());
|
||||
TEST(!integration::TestRouteLength(route, 1533.), ());
|
||||
}
|
||||
|
||||
UNIT_TEST(RussiaMoscowSalameiNerisUturnTurnTest)
|
||||
{
|
||||
shared_ptr<integration::OsrmRouterComponents> routerComponents = integration::GetAllMaps();
|
||||
RouteResultT const routeResult = integration::CalculateRoute(routerComponents, {37.395332276656617, 67.633925439079519},
|
||||
{0., 0.}, {37.392503720352721, 67.61975260731343});
|
||||
shared_ptr<Route> const route = routeResult.first;
|
||||
OsrmRouter::ResultCode const result = routeResult.second;
|
||||
|
||||
TEST_EQUAL(result, OsrmRouter::NoError, ());
|
||||
|
||||
TEST(integration::TestTurn(route, 0, {37.388482521388539, 67.633382734905041}, turns::TurnSlightRight), ());
|
||||
TEST(integration::TestTurn(route, 1, {37.387117276989784, 67.633369323859881}, turns::TurnLeft), ());
|
||||
TEST(integration::TestTurn(route, 2, {37.387380133475205, 67.632781920081243}, turns::TurnLeft), ());
|
||||
TEST(integration::TestTurn(route, 3, {37.390526364673121, 67.633106467374461}, turns::TurnRight), ());
|
||||
|
||||
TEST(integration::TestTurnCount(route, 4), ());
|
||||
|
||||
TEST(integration::TestRouteLength(route, 1637.), ());
|
||||
|
||||
}
|
||||
|
3
omim.pro
3
omim.pro
|
@ -43,7 +43,8 @@ SUBDIRS = 3party \
|
|||
gui/gui_tests \
|
||||
qt \
|
||||
drape_head \
|
||||
map_server
|
||||
map_server \
|
||||
integration_tests \
|
||||
} else:drape_device {
|
||||
# libraries which are used on mobile devices with drape engine
|
||||
SUBDIRS = 3party \
|
||||
|
|
|
@ -22,6 +22,12 @@ struct PhantomNode;
|
|||
struct PathData;
|
||||
class FeatureType;
|
||||
struct RawRouteData;
|
||||
|
||||
namespace integration
|
||||
{
|
||||
class OsrmRouterWrapper;
|
||||
}
|
||||
|
||||
namespace routing
|
||||
{
|
||||
|
||||
|
@ -131,6 +137,10 @@ public:
|
|||
|
||||
RoutingMappingPtrT GetMappingByName(string const & fName, Index const * pIndex);
|
||||
|
||||
friend class integration::OsrmRouterWrapper;
|
||||
typedef function<string (m2::PointD const &)> CountryFileFnT;
|
||||
CountryFileFnT m_countryFn;
|
||||
|
||||
void Clear()
|
||||
{
|
||||
m_mapping.clear();
|
||||
|
|
|
@ -78,6 +78,7 @@ public:
|
|||
string const & GetRouterId() const { return m_router; }
|
||||
m2::PolylineD const & GetPoly() const { return m_poly; }
|
||||
turns::TurnsGeomT const & GetTurnsGeometry() const { return m_turnsGeom; }
|
||||
TurnsT const & GetTurns() const { return m_turns; }
|
||||
string const & GetName() const { return m_name; }
|
||||
bool IsValid() const { return (m_poly.GetSize() > 1); }
|
||||
|
||||
|
|
15
std/new.hpp
Normal file
15
std/new.hpp
Normal file
|
@ -0,0 +1,15 @@
|
|||
#pragma once
|
||||
#include "common_defines.hpp"
|
||||
|
||||
#ifdef new
|
||||
#undef new
|
||||
#endif
|
||||
|
||||
#include <new>
|
||||
|
||||
using std::bad_alloc;
|
||||
using std::bad_array_new_length;
|
||||
|
||||
#ifdef DEBUG_NEW
|
||||
#define new DEBUG_NEW
|
||||
#endif
|
Loading…
Add table
Reference in a new issue