/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2024 Artem Pavlenko * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * *****************************************************************************/ #include "catch.hpp" #include #include #include #include #include #include #include MAPNIK_DISABLE_WARNING_PUSH #include #include MAPNIK_DISABLE_WARNING_POP #include namespace { std::size_t count_shapefile_features(std::string const& filename) { #if defined(MAPNIK_MEMORY_MAPPED_FILE) mapnik::mapped_memory_cache::instance().clear(); #endif mapnik::parameters params; params["type"] = "shape"; params["file"] = filename; auto ds = mapnik::datasource_cache::instance().create(params); REQUIRE(ds != nullptr); CHECK(ds->type() == mapnik::datasource::datasource_t::Vector); auto fields = ds->get_descriptor().get_descriptors(); mapnik::query query(ds->envelope()); for (auto const& field : fields) { query.add_property_name(field.get_name()); } auto features = ds->features(query); REQUIRE(features != nullptr); std::size_t feature_count = 0; auto feature = features->next(); while (feature) { ++feature_count; feature = features->next(); } return feature_count; } int create_shapefile_index(std::string const& filename, bool index_parts, bool silent = true) { std::string cmd; if (std::getenv("DYLD_LIBRARY_PATH") != nullptr) { cmd += std::string("DYLD_LIBRARY_PATH=") + std::getenv("DYLD_LIBRARY_PATH") + " "; } cmd += "shapeindex"; #ifdef _WIN32 cmd += ".exe"; #endif cmd += " "; if (index_parts) cmd += "--index-parts "; cmd += filename; if (silent) { #ifndef _WIN32 cmd += " 2>/dev/null"; #else cmd += " 2> nul"; #endif } return std::system(cmd.c_str()); } } // namespace TEST_CASE("invalid shapeindex") { const bool have_shape_plugin = mapnik::datasource_cache::instance().plugin_registered("shape"); if (have_shape_plugin) { SECTION("Invalid index") { for (auto val : {std::make_tuple(true, std::string("mapnik-invalid-index.................")), // invalid header std::make_tuple(false, std::string("mapnik-index................."))}) // valid header + invalid index { std::string path = "test/data/shp/boundaries.shp"; std::string index_path = path.substr(0, path.rfind(".")) + ".index"; // remove *.index if present mapnik::util::mapped_memory_file::deleteFile(index_path); // count features std::size_t feature_count = count_shapefile_features(path); // create index std::ofstream index(index_path.c_str(), std::ios::binary); index.write(std::get<1>(val).c_str(), std::get<1>(val).size()); index.close(); // count features std::size_t feature_count_indexed = count_shapefile_features(path); if (std::get<0>(val)) // fallback to un-indexed access { // ensure number of features are the same CHECK(feature_count == feature_count_indexed); } else // the header is valid but index file itself is not - expect datasource to fail and return 0 // features. { CHECK(feature_count_indexed == 0); } // remove *.index if present mapnik::util::mapped_memory_file::deleteFile(index_path); } } } } TEST_CASE("shapeindex") { const bool have_shape_plugin = mapnik::datasource_cache::instance().plugin_registered("shape"); if (have_shape_plugin) { SECTION("Index") { for (auto const& path : mapnik::util::list_directory("test/data/shp/")) { if (boost::iends_with(path, ".shp")) { for (bool index_parts : {false, true}) { CAPTURE(path); CAPTURE(index_parts); std::string index_path = path.substr(0, path.rfind(".")) + ".index"; // remove *.index if present mapnik::util::mapped_memory_file::deleteFile(index_path); // count features std::size_t feature_count = count_shapefile_features(path); // create *.index if (feature_count > 0) { REQUIRE(create_shapefile_index(path, index_parts) == EXIT_SUCCESS); } else { REQUIRE(create_shapefile_index(path, index_parts) != EXIT_SUCCESS); REQUIRE(!mapnik::util::exists(index_path)); // index won't be created if there's no features } // count features std::size_t feature_count_indexed = count_shapefile_features(path); // ensure number of features are the same REQUIRE(feature_count == feature_count_indexed); // remove *.index if present mapnik::util::mapped_memory_file::deleteFile(index_path); } } } } } }