#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace mbgl; class SourceTest { public: util::RunLoop loop; StubFileSource fileSource; StubStyleObserver observer; Transform transform; TransformState transformState; ThreadPool threadPool { 1 }; AnnotationManager annotationManager { 1.0 }; style::Style style { fileSource, 1.0 }; style::UpdateParameters updateParameters { 1.0, MapDebugOptions(), transformState, threadPool, fileSource, MapMode::Continuous, annotationManager, style }; SourceTest() { // Squelch logging. Log::setObserver(std::make_unique()); transform.resize({{ 512, 512 }}); transform.setLatLngZoom({0, 0}, 0); transformState = transform.getState(); } void run() { loop.run(); } void end() { loop.stop(); } }; TEST(Source, LoadingFail) { SourceTest test; test.fileSource.sourceResponse = [&] (const Resource& resource) { EXPECT_EQ("url", resource.url); Response response; response.error = std::make_unique( Response::Error::Reason::Other, "Failed by the test case"); return response; }; test.observer.sourceError = [&] (Source& source, std::exception_ptr error) { EXPECT_EQ("source", source.getID()); EXPECT_EQ("Failed by the test case", util::toString(error)); test.end(); }; VectorSource source("source", "url"); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); test.run(); } TEST(Source, LoadingCorrupt) { SourceTest test; test.fileSource.sourceResponse = [&] (const Resource& resource) { EXPECT_EQ("url", resource.url); Response response; response.data = std::make_unique("CORRUPTED"); return response; }; test.observer.sourceError = [&] (Source& source, std::exception_ptr error) { EXPECT_EQ("source", source.getID()); EXPECT_EQ("0 - Invalid value.", util::toString(error)); test.end(); }; VectorSource source("source", "url"); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); test.run(); } TEST(Source, RasterTileEmpty) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.noContent = true; return response; }; test.observer.tileChanged = [&] (Source& source, const OverscaledTileID&) { EXPECT_EQ("source", source.getID()); test.end(); }; test.observer.tileError = [&] (Source&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, VectorTileEmpty) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.noContent = true; return response; }; test.observer.tileChanged = [&] (Source& source, const OverscaledTileID&) { EXPECT_EQ("source", source.getID()); test.end(); }; test.observer.tileError = [&] (Source&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, RasterTileFail) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.error = std::make_unique( Response::Error::Reason::Other, "Failed by the test case"); return response; }; test.observer.tileError = [&] (Source& source, const OverscaledTileID& tileID, std::exception_ptr error) { EXPECT_EQ(SourceType::Raster, source.baseImpl->type); EXPECT_EQ(OverscaledTileID(0, 0, 0), tileID); EXPECT_EQ("Failed by the test case", util::toString(error)); test.end(); }; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, VectorTileFail) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.error = std::make_unique( Response::Error::Reason::Other, "Failed by the test case"); return response; }; test.observer.tileError = [&] (Source& source, const OverscaledTileID& tileID, std::exception_ptr error) { EXPECT_EQ(SourceType::Vector, source.baseImpl->type); EXPECT_EQ(OverscaledTileID(0, 0, 0), tileID); EXPECT_EQ("Failed by the test case", util::toString(error)); test.end(); }; Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, RasterTileCorrupt) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.data = std::make_unique("CORRUPTED"); return response; }; test.observer.tileError = [&] (Source& source, const OverscaledTileID& tileID, std::exception_ptr error) { EXPECT_EQ(source.baseImpl->type, SourceType::Raster); EXPECT_EQ(OverscaledTileID(0, 0, 0), tileID); EXPECT_TRUE(bool(error)); // Not asserting on platform-specific error text. test.end(); }; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, VectorTileCorrupt) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.data = std::make_unique("CORRUPTED"); return response; }; test.observer.tileError = [&] (Source& source, const OverscaledTileID& tileID, std::exception_ptr error) { EXPECT_EQ(source.baseImpl->type, SourceType::Vector); EXPECT_EQ(OverscaledTileID(0, 0, 0), tileID); EXPECT_EQ(util::toString(error), "unknown pbf field type exception"); test.end(); }; // Need to have at least one layer that uses the source. auto layer = std::make_unique("id", "source"); layer->setSourceLayer("water"); test.style.addLayer(std::move(layer)); Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, RasterTileCancel) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { test.end(); return optional(); }; test.observer.tileChanged = [&] (Source&, const OverscaledTileID&) { FAIL() << "Should never be called"; }; test.observer.tileError = [&] (Source&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, VectorTileCancel) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { test.end(); return optional(); }; test.observer.tileChanged = [&] (Source&, const OverscaledTileID&) { FAIL() << "Should never be called"; }; test.observer.tileError = [&] (Source&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, RasterTileAttribution) { SourceTest test; std::string mapboxOSM = ("© Mapbox " "©️ OpenStreetMap"); test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.noContent = true; return response; }; test.fileSource.sourceResponse = [&] (const Resource& resource) { EXPECT_EQ("url", resource.url); Response response; response.data = std::make_unique(R"TILEJSON({ "tilejson": "2.1.0", "attribution": ")TILEJSON" + mapboxOSM + R"TILEJSON(", "tiles": [ "tiles" ] })TILEJSON"); return response; }; test.observer.sourceAttributionChanged = [&] (Source&, std::string attribution) { EXPECT_EQ(mapboxOSM, attribution); EXPECT_FALSE(mapboxOSM.find("©️ OpenStreetMap") == std::string::npos); test.end(); }; test.observer.tileError = [&] (Source&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; RasterSource source("source", "url", 512); source.baseImpl->setObserver(&test.observer); source.baseImpl->loadDescription(test.fileSource); source.baseImpl->updateTiles(test.updateParameters); test.run(); } TEST(Source, GeoJSonSourceUrlUpdate) { SourceTest test; test.fileSource.sourceResponse = [&] (const Resource& resource) { EXPECT_EQ("url", resource.url); Response response; response.data = std::make_unique("{\"geometry\": {\"type\": \"Point\", \"coordinates\": [1.1, 1.1]}, \"type\": \"Feature\", \"properties\": {}}"); return response; }; test.observer.sourceDescriptionChanged = [&] (Source&) { //Should be called (test will hang if it doesn't) test.end(); }; test.observer.tileError = [&] (Source&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; GeoJSONSource source("source"); source.baseImpl->setObserver(&test.observer); //Load initial, so the source state will be loaded=true source.baseImpl->loadDescription(test.fileSource); //Schedule an update test.loop.invoke([&] () { //Update the url source.setURL(std::string("http://source-url.ext")); }); test.run(); }