Files
osm2pgsql/tests/test-osm-file-parsing.cpp
2025-06-12 12:38:12 +02:00

260 lines
7.2 KiB
C++

/**
* SPDX-License-Identifier: GPL-2.0-or-later
*
* This file is part of osm2pgsql (https://osm2pgsql.org/).
*
* Copyright (C) 2006-2025 by the osm2pgsql developer community.
* For a full list of authors see the git log.
*/
#include <catch.hpp>
#include "middle.hpp"
#include "output-null.hpp"
#include "common-import.hpp"
#include "common-options.hpp"
struct type_stats_t
{
unsigned added = 0;
unsigned modified = 0;
unsigned deleted = 0;
};
struct counting_middle_t : public middle_t
{
explicit counting_middle_t(bool append)
: middle_t(nullptr), m_append(append)
{}
void start() override
{
assert(m_middle_state == middle_state::constructed);
#ifndef NDEBUG
m_middle_state = middle_state::node;
#endif
}
void stop() override { assert(m_middle_state == middle_state::done); }
void cleanup() {}
void node(osmium::Node const &node) override {
assert(m_middle_state == middle_state::node);
if (m_append) {
++node_count.deleted;
if (!node.deleted()) {
++node_count.added;
}
return;
}
++node_count.added;
}
void way(osmium::Way const &way) override {
assert(m_middle_state == middle_state::way);
if (m_append) {
++way_count.deleted;
if (!way.deleted()) {
++way_count.added;
}
return;
}
++way_count.added;
}
void relation(osmium::Relation const &relation) override {
assert(m_middle_state == middle_state::relation);
if (m_append) {
++relation_count.deleted;
if (!relation.deleted()) {
++relation_count.added;
}
return;
}
++relation_count.added;
}
std::shared_ptr<middle_query_t> get_query_instance() override
{
return std::shared_ptr<middle_query_t>{};
}
type_stats_t node_count, way_count, relation_count;
bool m_append;
};
struct counting_output_t : public output_null_t
{
explicit counting_output_t(options_t const &options)
: output_null_t(nullptr, nullptr, options)
{}
std::shared_ptr<output_t>
clone(std::shared_ptr<middle_query_t> const &,
std::shared_ptr<db_copy_thread_t> const &) const override
{
return std::make_shared<counting_output_t>(*get_options());
}
void node_add(osmium::Node const &n) override
{
++node.added;
sum_ids += n.id();
}
void way_add(osmium::Way *w) override
{
++way.added;
sum_ids += w->id();
sum_nds += w->nodes().size();
}
void relation_add(osmium::Relation const &r) override
{
++relation.added;
sum_ids += r.id();
sum_members += r.members().size();
}
void node_modify(osmium::Node const &) override { ++node.modified; }
void way_modify(osmium::Way *) override { ++way.modified; }
void relation_modify(osmium::Relation const &) override
{
++relation.modified;
}
void node_delete(osmium::Node const &) override { ++node.deleted; }
void way_delete(osmium::Way *) override { ++way.deleted; }
void relation_delete(osmium::Relation const &) override { ++relation.deleted; }
type_stats_t node, way, relation;
uint64_t sum_ids = 0;
std::size_t sum_nds = 0;
std::size_t sum_members = 0;
};
TEST_CASE("parse xml file")
{
options_t const options = testing::opt_t().slim();
auto const middle = std::make_shared<counting_middle_t>(false);
middle->start();
auto const output = std::make_shared<counting_output_t>(options);
testing::parse_file(options, middle, output, "test_multipolygon.osm",
false);
REQUIRE(output->sum_ids == 73514);
REQUIRE(output->sum_nds == 495);
REQUIRE(output->sum_members == 146);
REQUIRE(output->node.added == 353);
REQUIRE(output->node.modified == 0);
REQUIRE(output->node.deleted == 0);
REQUIRE(output->way.added == 140);
REQUIRE(output->way.modified == 0);
REQUIRE(output->way.deleted == 0);
REQUIRE(output->relation.added == 40);
REQUIRE(output->relation.modified == 0);
REQUIRE(output->relation.deleted == 0);
auto const *mid_test = middle.get();
REQUIRE(mid_test->node_count.added == 353);
REQUIRE(mid_test->node_count.deleted == 0);
REQUIRE(mid_test->way_count.added == 140);
REQUIRE(mid_test->way_count.deleted == 0);
REQUIRE(mid_test->relation_count.added == 40);
REQUIRE(mid_test->relation_count.deleted == 0);
}
TEST_CASE("parse diff file")
{
options_t const options = testing::opt_t().slim().append();
auto const middle = std::make_shared<counting_middle_t>(true);
middle->start();
auto const output = std::make_shared<counting_output_t>(options);
testing::parse_file(options, middle, output, "008-ch.osc.gz", false);
REQUIRE(output->node.added == 0);
REQUIRE(output->node.modified == 1176);
REQUIRE(output->node.deleted == 16773);
REQUIRE(output->way.added == 0);
REQUIRE(output->way.modified == 161);
REQUIRE(output->way.deleted == 4);
REQUIRE(output->relation.added == 0);
REQUIRE(output->relation.modified == 11);
REQUIRE(output->relation.deleted == 1);
auto *mid_test = middle.get();
REQUIRE(mid_test->node_count.added == 1176);
REQUIRE(mid_test->node_count.deleted == 17949);
REQUIRE(mid_test->way_count.added == 161);
REQUIRE(mid_test->way_count.deleted == 165);
REQUIRE(mid_test->relation_count.added == 11);
REQUIRE(mid_test->relation_count.deleted == 12);
}
TEST_CASE("parse xml file with extra args")
{
options_t options = testing::opt_t().slim().srs(PROJ_SPHERE_MERC);
options.extra_attributes = true;
auto const middle = std::make_shared<counting_middle_t>(false);
middle->start();
auto const output = std::make_shared<counting_output_t>(options);
testing::parse_file(options, middle, output, "test_multipolygon.osm",
false);
REQUIRE(output->sum_ids == 73514);
REQUIRE(output->sum_nds == 495);
REQUIRE(output->sum_members == 146);
REQUIRE(output->node.added == 353);
REQUIRE(output->node.modified == 0);
REQUIRE(output->node.deleted == 0);
REQUIRE(output->way.added == 140);
REQUIRE(output->way.modified == 0);
REQUIRE(output->way.deleted == 0);
REQUIRE(output->relation.added == 40);
REQUIRE(output->relation.modified == 0);
REQUIRE(output->relation.deleted == 0);
auto const *mid_test = middle.get();
REQUIRE(mid_test->node_count.added == 353);
REQUIRE(mid_test->node_count.deleted == 0);
REQUIRE(mid_test->way_count.added == 140);
REQUIRE(mid_test->way_count.deleted == 0);
REQUIRE(mid_test->relation_count.added == 40);
REQUIRE(mid_test->relation_count.deleted == 0);
}
TEST_CASE("invalid location")
{
options_t const options = testing::opt_t();
auto const middle = std::make_shared<counting_middle_t>(false);
middle->start();
auto const output = std::make_shared<counting_output_t>(options);
testing::parse_file(options, middle, output, "test_invalid_location.osm",
false);
REQUIRE(output->node.added == 0);
REQUIRE(output->way.added == 0);
REQUIRE(output->relation.added == 0);
}