From 1765590f3e3f8050bc9b6416c265a3d861a37879 Mon Sep 17 00:00:00 2001 From: Ilya Zverev Date: Sun, 14 Jan 2018 16:33:28 +0300 Subject: [PATCH] More work on c++ file --- conflate/conflate.py | 8 +- filter/filter_planet_by_cats.cpp | 171 +++++++++++++++++++++++++++---- 2 files changed, 154 insertions(+), 25 deletions(-) diff --git a/conflate/conflate.py b/conflate/conflate.py index 9fdaa50..14ef9fe 100755 --- a/conflate/conflate.py +++ b/conflate/conflate.py @@ -346,11 +346,12 @@ class OsmConflator: padding = self.profile.get('bbox_padding', BBOX_PADDING) return [get_bbox(b, padding) for b in boxes] - def check_against_profile_tags(self, tags): + def get_categories(self, tags): qualifies = self.profile.get('qualifies', args=tags) if qualifies is not None: return qualifies + categories = [''] query = self.profile.get('query', None) if query is not None and not isinstance(query, str): for tag in query: @@ -360,7 +361,7 @@ class OsmConflator: if len(tag) >= 2 and tag[1][0] != '~': if tag[1] != tags[tag[0]]: return False - return True + return categories def download_osm(self): """Constructs an Overpass API query and requests objects @@ -421,7 +422,8 @@ class OsmConflator: tags = {} for tag in el.findall('tag'): tags[tag.get('k')] = tag.get('v') - if not self.check_against_profile_tags(tags): + categories = self.get_categories(tags) + if categories is False or categories is None or len(categories) == 0: continue if el.tag == 'node': diff --git a/filter/filter_planet_by_cats.cpp b/filter/filter_planet_by_cats.cpp index 8eddec1..a65d33b 100644 --- a/filter/filter_planet_by_cats.cpp +++ b/filter/filter_planet_by_cats.cpp @@ -15,18 +15,20 @@ Written by Ilya Zverev for MAPS.ME. */ +#include #include #include #include +#include #include +#include -#include -#include #include #include #include #include #include +#include #include #include "RTree.h" @@ -35,16 +37,41 @@ using index_type = osmium::index::map::FlexMem; using location_handler_type = osmium::handler::NodeLocationsForWays; -constexpr double kSearchRadius = 0.01; // ~1 km +class AmenityRelationsManager : public osmium::relations::RelationsManager { +public: + + bool new_relation(osmium::Relation const & rel) noexcept { + const char *rel_type = rel.tags().get_value_by_key("type"); + return rel_type && !std::strcmp(rel_type, "multipolygon"); + } + + void complete_relation(osmium::Relation const & rel) { + this->buffer().add_item(rel); + this->buffer().commit(); + } +}; + +bool AppendToVector(uint16_t cat_id, void *vec) { + static_cast*>(vec)->push_back(cat_id); +} class AmenityHandler : public osmium::handler::Handler { + constexpr static double kSearchRadius = 0.001; // ~1 km TODO! revert to 0.01 + typedef RTree DatasetTree; + typedef std::vector> TQuery; + typedef std::vector TCategory; + DatasetTree m_tree; + osmium::io::Writer &m_writer; + std::map> m_categories; + std::map m_category_names; void print_object(const osmium::OSMObject &obj, const osmium::Location ¢er) { // TODO + std::cout << obj.type() << ' ' << obj.id() << std::endl; } // Calculate the center point of a NodeRefList. @@ -63,16 +90,82 @@ class AmenityHandler : public osmium::handler::Handler { return osmium::Location{x, y}; } - bool eligible(const osmium::Location &coord, osmium::TagList const &tags) { - // TODO: find all points in a certain radius around coord - // TODO: test tags for each of these points on tags - return true; // TODO: change default to false + bool TestTags(osmium::TagList const & tags, TQuery const & query) { + for (auto const & pair : query) { + // TODO + } + return true; } - osmium::io::Writer &m_writer; + bool IsEligible(const osmium::Location & loc, osmium::TagList const & tags) { + if (tags.empty()) + return false; + + int32_t radius = osmium::Location::double_to_fix(kSearchRadius); + int32_t min[] = {loc.x() - radius, loc.y() - radius}; + int32_t max[] = {loc.x() + radius, loc.y() + radius}; + std::vector found; + if (!m_tree.Search(min, max, &AppendToVector, &found)) + return false; + for (uint16_t cat_id : found) + for (TQuery query : m_categories[cat_id]) + if (TestTags(tags, query)) + return true; + return false; + } + + void SplitTrim(std::string const & s, char delimiter, std::size_t limit, std::vector & target) { + target.clear(); + std::size_t start = 0, end = 0; + while (start < s.length()) { + end = s.find(delimiter, start); + if (end == std::string::npos || target.size() == limit) + end = s.length(); + while (start < end && std::isspace(s[start])) + start++; + + std::size_t tmpend = end - 1; + while (tmpend > start && std::isspace(s[tmpend])) + tmpend++; + target.push_back(s.substr(start, tmpend - start + 1)); + start = end + 1; + } + } + + TQuery ParseQuery(std::string const & query) { + TQuery q; + std::vector parts, keys; + SplitTrim(query, '|', 100, parts); + for (std::string const & part : parts) { + SplitTrim(part, '=', 100, keys); + // TODO + } + return q; + } void LoadCategories(const char *filename) { - // TODO: read categories list and make an kd-tree of these. + std::ifstream infile(filename); + std::string line; + std::vector parts; + bool parsingPoints = false; + while (std::getline(infile, line)) { + if (!parsingPoints) { + if (!line.size()) + parsingPoints = true; + else { + SplitTrim(line, ',', 3, parts); // cat_id, name, query + uint16_t cat_id = std::stoi(parts[0]); + m_category_names[cat_id] = parts[1]; + m_categories[cat_id].push_back(ParseQuery(parts[2])); + } + } else { + SplitTrim(line, ',', 3, parts); // lon, lat, cat_id + const osmium::Location loc(std::stod(parts[0]), std::stod(parts[1])); + int32_t coords[] = {loc.x(), loc.y()}; + uint16_t cat_id = std::stoi(parts[2]); + m_tree.Insert(coords, coords, cat_id); + } + } } public: @@ -81,16 +174,53 @@ public: LoadCategories(categories); } - void node(const osmium::Node &node) { - if (eligible(node.location(), node.tags())) { + void node(osmium::Node const & node) { + if (IsEligible(node.location(), node.tags())) { print_object(node, node.location()); } } - void area(const osmium::Area &area) { - const auto center = calc_center(*area.cbegin()); - if (eligible(center, area.tags())) { - print_object(area, center); + void way(osmium::Way const & way) { + if (!way.is_closed()) + return; + + int64_t x = 0, y = 0, cnt = 0; + for (const auto& node_ref : way.nodes()) { + if (node_ref.location()) { + x += node_ref.x(); + y += node_ref.y(); + cnt++; + } + } + if (!cnt) + return; + + const osmium::Location center(x / cnt, y / cnt); + if (IsEligible(center, way.tags())) { + print_object(way, center); + } + } + + void relation(osmium::Relation const & rel) { + int64_t x = 0, y = 0, cnt = 0; + for (const auto& member : rel.members()) { + if (member.full_member() && member.type() == osmium::item_type::way) { + const osmium::Way *way = reinterpret_cast(&member.get_object()); + for (const auto& node_ref : way->nodes()) { + if (false && node_ref.location()) { + x += node_ref.x(); + y += node_ref.y(); + cnt++; + } + } + } + } + if (!cnt) + return; + + const osmium::Location center(x / cnt, y / cnt); + if (IsEligible(center, rel.tags())) { + print_object(rel, center); } } @@ -106,12 +236,9 @@ int main(int argc, char *argv[]) { const osmium::io::File input_file{argv[2]}; const osmium::io::File output_file{argc > 3 ? argv[3] : "", "osm"}; - std::cerr << "Pass 1/2: Assembling multipolygons...\n"; - osmium::area::Assembler::config_type assembler_config; - assembler_config.create_empty_areas = false; - osmium::area::MultipolygonManager mp_manager{ - assembler_config}; - osmium::relations::read_relations(input_file, mp_manager); + std::cerr << "Pass 1/2: Reading relations...\n"; + AmenityRelationsManager manager; + osmium::relations::read_relations(input_file, manager); osmium::io::Header header; header.set("generator", argv[0]); @@ -125,7 +252,7 @@ int main(int argc, char *argv[]) { osmium::io::Reader reader{input_file}; osmium::apply(reader, location_handler, data_handler, - mp_manager.handler( + manager.handler( [&data_handler](const osmium::memory::Buffer &area_buffer) { osmium::apply(area_buffer, data_handler); }));