Libosmium  2.17.0
Fast and flexible C++ library for working with OpenStreetMap data
Loading...
Searching...
No Matches
problem_reporter_ogr.hpp
Go to the documentation of this file.
1#ifndef OSMIUM_AREA_PROBLEM_REPORTER_OGR_HPP
2#define OSMIUM_AREA_PROBLEM_REPORTER_OGR_HPP
3
4/*
5
6This file is part of Osmium (https://osmcode.org/libosmium).
7
8Copyright 2013-2021 Jochen Topf <jochen@topf.org> and others (see README).
9
10Boost Software License - Version 1.0 - August 17th, 2003
11
12Permission is hereby granted, free of charge, to any person or organization
13obtaining a copy of the software and accompanying documentation covered by
14this license (the "Software") to use, reproduce, display, distribute,
15execute, and transmit the Software, and to prepare derivative works of the
16Software, and to permit third-parties to whom the Software is furnished to
17do so, all subject to the following:
18
19The copyright notices in the Software and this entire statement, including
20the above license grant, this restriction and the following disclaimer,
21must be included in all copies of the Software, in whole or in part, and
22all derivative works of the Software, unless such copies or derivative
23works are solely in the form of machine-executable object code generated by
24a source language processor.
25
26THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
27IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
28FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
29SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
30FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
31ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
32DEALINGS IN THE SOFTWARE.
33
34*/
35
47#include <osmium/geom/ogr.hpp>
52#include <osmium/osm/types.hpp>
53#include <osmium/osm/way.hpp>
54
55#include <gdalcpp.hpp>
56
57#include <memory>
58
59namespace osmium {
60
61 namespace area {
62
68
70
71 gdalcpp::Layer m_layer_perror;
72 gdalcpp::Layer m_layer_lerror;
73 gdalcpp::Layer m_layer_ways;
74
75 void set_object(gdalcpp::Feature& feature) {
76 const char t[2] = {osmium::item_type_to_char(m_object_type), '\0'};
77 feature.set_field("obj_type", t);
78 feature.set_field("obj_id", int32_t(m_object_id));
79 feature.set_field("nodes", int32_t(m_nodes));
80 }
81
82 void write_point(const char* problem_type, osmium::object_id_type id1, osmium::object_id_type id2, osmium::Location location) {
83 gdalcpp::Feature feature{m_layer_perror, m_ogr_factory.create_point(location)};
84 set_object(feature);
85 feature.set_field("id1", double(id1));
86 feature.set_field("id2", double(id2));
87 feature.set_field("problem", problem_type);
88 feature.add_to_layer();
89 }
90
91 void write_line(const char* problem_type, osmium::object_id_type id1, osmium::object_id_type id2, osmium::Location loc1, osmium::Location loc2) {
92 auto ogr_linestring = std::unique_ptr<OGRLineString>{new OGRLineString{}};
93 ogr_linestring->addPoint(loc1.lon(), loc1.lat());
94 ogr_linestring->addPoint(loc2.lon(), loc2.lat());
95
96 gdalcpp::Feature feature{m_layer_lerror, std::move(ogr_linestring)};
97 set_object(feature);
98 feature.set_field("id1", static_cast<double>(id1));
99 feature.set_field("id2", static_cast<double>(id2));
100 feature.set_field("problem", problem_type);
101 feature.add_to_layer();
102 }
103
104 public:
105
106 explicit ProblemReporterOGR(gdalcpp::Dataset& dataset) :
107 m_layer_perror(dataset, "perrors", wkbPoint),
108 m_layer_lerror(dataset, "lerrors", wkbLineString),
109 m_layer_ways(dataset, "ways", wkbLineString) {
110
111 // 64bit integers are not supported in GDAL < 2, so we
112 // are using a workaround here in fields where we expect
113 // node IDs, we use real numbers.
115 .add_field("obj_type", OFTString, 1)
116 .add_field("obj_id", OFTInteger, 10)
117 .add_field("nodes", OFTInteger, 8)
118 .add_field("id1", OFTReal, 12, 1)
119 .add_field("id2", OFTReal, 12, 1)
120 .add_field("problem", OFTString, 30)
121 ;
122
124 .add_field("obj_type", OFTString, 1)
125 .add_field("obj_id", OFTInteger, 10)
126 .add_field("nodes", OFTInteger, 8)
127 .add_field("id1", OFTReal, 12, 1)
128 .add_field("id2", OFTReal, 12, 1)
129 .add_field("problem", OFTString, 30)
130 ;
131
133 .add_field("obj_type", OFTString, 1)
134 .add_field("obj_id", OFTInteger, 10)
135 .add_field("way_id", OFTInteger, 10)
136 .add_field("nodes", OFTInteger, 8)
137 ;
138 }
139
141 write_point("duplicate_node", node_id1, node_id2, location);
142 }
143
145 write_point("touching_ring", node_id, 0, location);
146 }
147
149 osmium::object_id_type way2_id, osmium::Location way2_seg_start, osmium::Location way2_seg_end, osmium::Location intersection) override {
150 write_point("intersection", way1_id, way2_id, intersection);
151 write_line("intersection", way1_id, way2_id, way1_seg_start, way1_seg_end);
152 write_line("intersection", way2_id, way1_id, way2_seg_start, way2_seg_end);
153 }
154
155 void report_duplicate_segment(const osmium::NodeRef& nr1, const osmium::NodeRef& nr2) override {
156 write_line("duplicate_segment", nr1.ref(), nr2.ref(), nr1.location(), nr2.location());
157 }
158
159 void report_overlapping_segment(const osmium::NodeRef& nr1, const osmium::NodeRef& nr2) override {
160 write_line("overlapping_segment", nr1.ref(), nr2.ref(), nr1.location(), nr2.location());
161 }
162
163 void report_ring_not_closed(const osmium::NodeRef& nr, const osmium::Way* way) override {
164 write_point("ring_not_closed", nr.ref(), way ? way->id() : 0, nr.location());
165 }
166
168 write_line("role_should_be_outer", way_id, 0, seg_start, seg_end);
169 }
170
172 write_line("role_should_be_inner", way_id, 0, seg_start, seg_end);
173 }
174
176 if (way.nodes().size() < 2) {
177 return;
178 }
179 try {
180 gdalcpp::Feature feature{m_layer_lerror, m_ogr_factory.create_linestring(way)};
181 set_object(feature);
182 feature.set_field("id1", int32_t(way.id()));
183 feature.set_field("id2", 0);
184 feature.set_field("problem", "way_in_multiple_rings");
185 feature.add_to_layer();
186 } catch (const osmium::geometry_error&) {
187 // XXX
188 }
189 }
190
192 if (way.nodes().size() < 2) {
193 return;
194 }
195 try {
196 gdalcpp::Feature feature{m_layer_lerror, m_ogr_factory.create_linestring(way)};
197 set_object(feature);
198 feature.set_field("id1", int32_t(way.id()));
199 feature.set_field("id2", 0);
200 feature.set_field("problem", "inner_with_same_tags");
201 feature.add_to_layer();
202 } catch (const osmium::geometry_error&) {
203 // XXX
204 }
205 }
206
207 void report_duplicate_way(const osmium::Way& way) override {
208 if (way.nodes().size() < 2) {
209 return;
210 }
211 try {
212 gdalcpp::Feature feature{m_layer_lerror, m_ogr_factory.create_linestring(way)};
213 set_object(feature);
214 feature.set_field("id1", int32_t(way.id()));
215 feature.set_field("id2", 0);
216 feature.set_field("problem", "duplicate_way");
217 feature.add_to_layer();
218 } catch (const osmium::geometry_error&) {
219 // XXX
220 }
221 }
222
223 void report_way(const osmium::Way& way) override {
224 if (way.nodes().empty()) {
225 return;
226 }
227 if (way.nodes().size() == 1) {
228 const auto& first_nr = way.nodes()[0];
229 write_point("single_node_in_way", way.id(), first_nr.ref(), first_nr.location());
230 return;
231 }
232 try {
233 gdalcpp::Feature feature{m_layer_ways, m_ogr_factory.create_linestring(way)};
234 set_object(feature);
235 feature.set_field("way_id", int32_t(way.id()));
236 feature.add_to_layer();
237 } catch (const osmium::geometry_error&) {
238 // XXX
239 }
240 }
241
242 }; // class ProblemReporterOGR
243
244 } // namespace area
245
246} // namespace osmium
247
248#endif // OSMIUM_AREA_PROBLEM_REPORTER_OGR_HPP
Definition: location.hpp:271
double lon() const
Definition: location.hpp:396
double lat() const
Definition: location.hpp:415
Definition: node_ref.hpp:50
constexpr osmium::object_id_type ref() const noexcept
Definition: node_ref.hpp:71
osmium::Location & location() noexcept
Definition: node_ref.hpp:85
Definition: way.hpp:72
Definition: problem_reporter_ogr.hpp:67
void report_ring_not_closed(const osmium::NodeRef &nr, const osmium::Way *way) override
Definition: problem_reporter_ogr.hpp:163
void report_duplicate_segment(const osmium::NodeRef &nr1, const osmium::NodeRef &nr2) override
Definition: problem_reporter_ogr.hpp:155
osmium::geom::OGRFactory m_ogr_factory
Definition: problem_reporter_ogr.hpp:69
void report_inner_with_same_tags(const osmium::Way &way) override
Definition: problem_reporter_ogr.hpp:191
gdalcpp::Layer m_layer_ways
Definition: problem_reporter_ogr.hpp:73
void report_overlapping_segment(const osmium::NodeRef &nr1, const osmium::NodeRef &nr2) override
Definition: problem_reporter_ogr.hpp:159
void report_role_should_be_outer(osmium::object_id_type way_id, osmium::Location seg_start, osmium::Location seg_end) override
Definition: problem_reporter_ogr.hpp:167
void write_point(const char *problem_type, osmium::object_id_type id1, osmium::object_id_type id2, osmium::Location location)
Definition: problem_reporter_ogr.hpp:82
gdalcpp::Layer m_layer_lerror
Definition: problem_reporter_ogr.hpp:72
void report_duplicate_way(const osmium::Way &way) override
Definition: problem_reporter_ogr.hpp:207
ProblemReporterOGR(gdalcpp::Dataset &dataset)
Definition: problem_reporter_ogr.hpp:106
void report_way_in_multiple_rings(const osmium::Way &way) override
Definition: problem_reporter_ogr.hpp:175
void write_line(const char *problem_type, osmium::object_id_type id1, osmium::object_id_type id2, osmium::Location loc1, osmium::Location loc2)
Definition: problem_reporter_ogr.hpp:91
void report_touching_ring(osmium::object_id_type node_id, osmium::Location location) override
Definition: problem_reporter_ogr.hpp:144
gdalcpp::Layer m_layer_perror
Definition: problem_reporter_ogr.hpp:71
void report_way(const osmium::Way &way) override
Definition: problem_reporter_ogr.hpp:223
void report_role_should_be_inner(osmium::object_id_type way_id, osmium::Location seg_start, osmium::Location seg_end) override
Definition: problem_reporter_ogr.hpp:171
void set_object(gdalcpp::Feature &feature)
Definition: problem_reporter_ogr.hpp:75
void report_intersection(osmium::object_id_type way1_id, osmium::Location way1_seg_start, osmium::Location way1_seg_end, osmium::object_id_type way2_id, osmium::Location way2_seg_start, osmium::Location way2_seg_end, osmium::Location intersection) override
Definition: problem_reporter_ogr.hpp:148
void report_duplicate_node(osmium::object_id_type node_id1, osmium::object_id_type node_id2, osmium::Location location) override
Definition: problem_reporter_ogr.hpp:140
Definition: problem_reporter.hpp:60
osmium::object_id_type m_object_id
Definition: problem_reporter.hpp:68
osmium::item_type m_object_type
Definition: problem_reporter.hpp:65
size_t m_nodes
Definition: problem_reporter.hpp:71
Definition: factory.hpp:148
point_type create_point(const osmium::Location &location) const
Definition: factory.hpp:210
linestring_type create_linestring(const osmium::WayNodeList &wnl, use_nodes un=use_nodes::unique, direction dir=direction::forward)
Definition: factory.hpp:265
Definition: factory.hpp:59
@ area
Definition: entity_bits.hpp:72
Namespace for everything in the Osmium library.
Definition: assembler.hpp:53
int64_t object_id_type
Type for OSM object (node, way, or relation) IDs.
Definition: types.hpp:45
char item_type_to_char(const item_type type) noexcept
Definition: item_type.hpp:120