#include #include #include #include #include #include #include #include #include #include #include #include #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 styleObserver; StubRenderSourceObserver renderSourceObserver; Transform transform; TransformState transformState; ThreadPool threadPool { 1 }; AnnotationManager annotationManager; ImageManager imageManager; GlyphManager glyphManager { fileSource }; TileParameters tileParameters { 1.0, MapDebugOptions(), transformState, threadPool, fileSource, MapMode::Continuous, annotationManager, imageManager, glyphManager }; 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.styleObserver.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.setObserver(&test.styleObserver); source.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.styleObserver.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.setObserver(&test.styleObserver); source.loadDescription(test.fileSource); test.run(); } TEST(Source, RasterTileEmpty) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.noContent = true; return response; }; RasterLayer layer("id", "source"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.loadDescription(test.fileSource); test.renderSourceObserver.tileChanged = [&] (RenderSource& source_, const OverscaledTileID&) { EXPECT_EQ("source", source_.baseImpl->id); test.end(); }; test.renderSourceObserver.tileError = [&] (RenderSource&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); test.run(); } TEST(Source, VectorTileEmpty) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.noContent = true; return response; }; LineLayer layer("id", "source"); layer.setSourceLayer("water"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.loadDescription(test.fileSource); test.renderSourceObserver.tileChanged = [&] (RenderSource& source_, const OverscaledTileID&) { EXPECT_EQ("source", source_.baseImpl->id); test.end(); }; test.renderSourceObserver.tileError = [&] (RenderSource&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); 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; }; RasterLayer layer("id", "source"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.loadDescription(test.fileSource); test.renderSourceObserver.tileError = [&] (RenderSource& 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(); }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); 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; }; LineLayer layer("id", "source"); layer.setSourceLayer("water"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.loadDescription(test.fileSource); test.renderSourceObserver.tileError = [&] (RenderSource& 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(); }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); test.run(); } TEST(Source, RasterTileCorrupt) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.data = std::make_unique("CORRUPTED"); return response; }; RasterLayer layer("id", "source"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.loadDescription(test.fileSource); test.renderSourceObserver.tileError = [&] (RenderSource& 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(); }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); test.run(); } TEST(Source, VectorTileCorrupt) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { Response response; response.data = std::make_unique("CORRUPTED"); return response; }; LineLayer layer("id", "source"); layer.setSourceLayer("water"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.loadDescription(test.fileSource); test.renderSourceObserver.tileError = [&] (RenderSource& 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(); }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); test.run(); } TEST(Source, RasterTileCancel) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { test.end(); return optional(); }; RasterLayer layer("id", "source"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; RasterSource source("source", tileset, 512); source.loadDescription(test.fileSource); test.renderSourceObserver.tileChanged = [&] (RenderSource&, const OverscaledTileID&) { FAIL() << "Should never be called"; }; test.renderSourceObserver.tileError = [&] (RenderSource&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); test.run(); } TEST(Source, VectorTileCancel) { SourceTest test; test.fileSource.tileResponse = [&] (const Resource&) { test.end(); return optional(); }; LineLayer layer("id", "source"); layer.setSourceLayer("water"); std::vector> layers {{ layer.baseImpl }}; Tileset tileset; tileset.tiles = { "tiles" }; VectorSource source("source", tileset); source.loadDescription(test.fileSource); test.renderSourceObserver.tileChanged = [&] (RenderSource&, const OverscaledTileID&) { FAIL() << "Should never be called"; }; test.renderSourceObserver.tileError = [&] (RenderSource&, const OverscaledTileID&, std::exception_ptr) { FAIL() << "Should never be called"; }; auto renderSource = RenderSource::create(source.baseImpl); renderSource->setObserver(&test.renderSourceObserver); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); test.run(); } TEST(Source, RasterTileAttribution) { SourceTest test; RasterLayer layer("id", "source"); std::vector> layers {{ layer.baseImpl }}; 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.styleObserver.sourceChanged = [&] (Source& source) { EXPECT_EQ(mapboxOSM, source.getAttribution()); EXPECT_FALSE(mapboxOSM.find("©️ OpenStreetMap") == std::string::npos); test.end(); }; RasterSource source("source", "url", 512); source.setObserver(&test.styleObserver); source.loadDescription(test.fileSource); auto renderSource = RenderSource::create(source.baseImpl); renderSource->update(source.baseImpl, layers, true, true, test.tileParameters); 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(R"({"geometry": {"type": "Point", "coordinates": [1.1, 1.1]}, "type": "Feature", "properties": {}})"); return response; }; test.styleObserver.sourceDescriptionChanged = [&] (Source&) { // Should be called (test will hang if it doesn't) test.end(); }; GeoJSONSource source("source"); source.setObserver(&test.styleObserver); // Load initial, so the source state will be loaded=true source.loadDescription(test.fileSource); // Schedule an update test.loop.invoke([&] () { // Update the url source.setURL(std::string("http://source-url.ext")); }); test.run(); } TEST(Source, ImageSourceImageUpdate) { SourceTest test; test.fileSource.response = [&] (const Resource& resource) { EXPECT_EQ("http://url", resource.url); Response response; response.data = std::make_unique(util::read_file("test/fixtures/image/no_profile.png")); return response; }; test.styleObserver.sourceChanged = [&] (Source&) { // Should be called (test will hang if it doesn't) test.end(); }; std::array coords; ImageSource source("source", coords); source.setURL("http://url"); source.setObserver(&test.styleObserver); // Load initial, so the source state will be loaded=true source.loadDescription(test.fileSource); UnassociatedImage rgba({ 1, 1 }); rgba.data[0] = 255; rgba.data[1] = 254; rgba.data[2] = 253; rgba.data[3] = 128; // Schedule an update test.loop.invoke([&] () { // Update the url source.setImage(std::move(rgba)); }); test.run(); }