2016-03-01 11:56:55 -05:00
|
|
|
#ifndef OSMIUM_IO_DETAIL_INPUT_FORMAT_HPP
|
|
|
|
#define OSMIUM_IO_DETAIL_INPUT_FORMAT_HPP
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
This file is part of Osmium (http://osmcode.org/libosmium).
|
|
|
|
|
|
|
|
Copyright 2013-2016 Jochen Topf <jochen@topf.org> and others (see README).
|
|
|
|
|
|
|
|
Boost Software License - Version 1.0 - August 17th, 2003
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person or organization
|
|
|
|
obtaining a copy of the software and accompanying documentation covered by
|
|
|
|
this license (the "Software") to use, reproduce, display, distribute,
|
|
|
|
execute, and transmit the Software, and to prepare derivative works of the
|
|
|
|
Software, and to permit third-parties to whom the Software is furnished to
|
|
|
|
do so, all subject to the following:
|
|
|
|
|
|
|
|
The copyright notices in the Software and this entire statement, including
|
|
|
|
the above license grant, this restriction and the following disclaimer,
|
|
|
|
must be included in all copies of the Software, in whole or in part, and
|
|
|
|
all derivative works of the Software, unless such copies or derivative
|
|
|
|
works are solely in the form of machine-executable object code generated by
|
|
|
|
a source language processor.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
|
|
|
SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
|
|
|
FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
|
|
|
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
DEALINGS IN THE SOFTWARE.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <exception>
|
|
|
|
#include <functional>
|
|
|
|
#include <future>
|
|
|
|
#include <map>
|
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
#include <osmium/io/detail/queue_util.hpp>
|
2016-10-03 13:08:59 -04:00
|
|
|
#include <osmium/io/error.hpp>
|
2016-03-01 11:56:55 -05:00
|
|
|
#include <osmium/io/file.hpp>
|
|
|
|
#include <osmium/io/file_format.hpp>
|
|
|
|
#include <osmium/io/header.hpp>
|
|
|
|
#include <osmium/memory/buffer.hpp>
|
|
|
|
#include <osmium/osm/entity_bits.hpp>
|
|
|
|
|
|
|
|
namespace osmium {
|
|
|
|
|
|
|
|
namespace io {
|
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
|
2016-11-11 09:50:02 -05:00
|
|
|
struct reader_options {
|
|
|
|
osmium::osm_entity_bits::type read_which_entities = osm_entity_bits::all;
|
|
|
|
osmium::io::read_meta read_metadata = read_meta::yes;
|
|
|
|
};
|
|
|
|
|
2016-03-01 11:56:55 -05:00
|
|
|
class Parser {
|
|
|
|
|
|
|
|
future_buffer_queue_type& m_output_queue;
|
|
|
|
std::promise<osmium::io::Header>& m_header_promise;
|
|
|
|
queue_wrapper<std::string> m_input_queue;
|
2016-11-11 09:50:02 -05:00
|
|
|
reader_options m_options;
|
2016-03-01 11:56:55 -05:00
|
|
|
bool m_header_is_done;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
std::string get_input() {
|
|
|
|
return m_input_queue.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool input_done() const {
|
|
|
|
return m_input_queue.has_reached_end_of_data();
|
|
|
|
}
|
|
|
|
|
2016-11-11 09:50:02 -05:00
|
|
|
osmium::osm_entity_bits::type read_types() const noexcept {
|
|
|
|
return m_options.read_which_entities;
|
|
|
|
}
|
|
|
|
|
|
|
|
osmium::io::read_meta read_metadata() const noexcept {
|
|
|
|
return m_options.read_metadata;
|
2016-03-01 11:56:55 -05:00
|
|
|
}
|
|
|
|
|
2016-11-11 09:50:02 -05:00
|
|
|
bool header_is_done() const noexcept {
|
2016-03-01 11:56:55 -05:00
|
|
|
return m_header_is_done;
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_header_value(const osmium::io::Header& header) {
|
|
|
|
if (!m_header_is_done) {
|
|
|
|
m_header_is_done = true;
|
|
|
|
m_header_promise.set_value(header);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_header_exception(const std::exception_ptr& exception) {
|
|
|
|
if (!m_header_is_done) {
|
|
|
|
m_header_is_done = true;
|
|
|
|
m_header_promise.set_exception(exception);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Wrap the buffer into a future and add it to the output queue.
|
|
|
|
*/
|
|
|
|
void send_to_output_queue(osmium::memory::Buffer&& buffer) {
|
|
|
|
add_to_queue(m_output_queue, std::move(buffer));
|
|
|
|
}
|
|
|
|
|
|
|
|
void send_to_output_queue(std::future<osmium::memory::Buffer>&& future) {
|
|
|
|
m_output_queue.push(std::move(future));
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
Parser(future_string_queue_type& input_queue,
|
|
|
|
future_buffer_queue_type& output_queue,
|
|
|
|
std::promise<osmium::io::Header>& header_promise,
|
2016-11-11 09:50:02 -05:00
|
|
|
osmium::io::detail::reader_options options) :
|
2016-03-01 11:56:55 -05:00
|
|
|
m_output_queue(output_queue),
|
|
|
|
m_header_promise(header_promise),
|
|
|
|
m_input_queue(input_queue),
|
2016-11-11 09:50:02 -05:00
|
|
|
m_options(options),
|
2016-03-01 11:56:55 -05:00
|
|
|
m_header_is_done(false) {
|
|
|
|
}
|
|
|
|
|
|
|
|
Parser(const Parser&) = delete;
|
|
|
|
Parser& operator=(const Parser&) = delete;
|
|
|
|
|
|
|
|
Parser(Parser&&) = delete;
|
|
|
|
Parser& operator=(Parser&&) = delete;
|
|
|
|
|
|
|
|
virtual ~Parser() noexcept = default;
|
|
|
|
|
|
|
|
virtual void run() = 0;
|
|
|
|
|
|
|
|
void parse() {
|
|
|
|
try {
|
|
|
|
run();
|
|
|
|
} catch (...) {
|
|
|
|
std::exception_ptr exception = std::current_exception();
|
|
|
|
set_header_exception(exception);
|
|
|
|
add_to_queue(m_output_queue, std::move(exception));
|
|
|
|
}
|
|
|
|
|
|
|
|
add_end_of_data_to_queue(m_output_queue);
|
|
|
|
}
|
|
|
|
|
|
|
|
}; // class Parser
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This factory class is used to create objects that decode OSM
|
|
|
|
* data written in a specified format.
|
|
|
|
*
|
|
|
|
* Do not use this class directly. Use the osmium::io::Reader
|
|
|
|
* class instead.
|
|
|
|
*/
|
|
|
|
class ParserFactory {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2016-10-03 13:08:59 -04:00
|
|
|
using create_parser_type = std::function<std::unique_ptr<Parser>(future_string_queue_type&,
|
|
|
|
future_buffer_queue_type&,
|
|
|
|
std::promise<osmium::io::Header>& header_promise,
|
2016-11-11 09:50:02 -05:00
|
|
|
osmium::io::detail::reader_options options)>;
|
2016-03-01 11:56:55 -05:00
|
|
|
|
|
|
|
private:
|
|
|
|
|
2016-10-03 13:08:59 -04:00
|
|
|
using map_type = std::map<osmium::io::file_format, create_parser_type>;
|
2016-03-01 11:56:55 -05:00
|
|
|
|
|
|
|
map_type m_callbacks;
|
|
|
|
|
|
|
|
ParserFactory() :
|
|
|
|
m_callbacks() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
static ParserFactory& instance() {
|
|
|
|
static ParserFactory factory;
|
|
|
|
return factory;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool register_parser(osmium::io::file_format format, create_parser_type create_function) {
|
|
|
|
if (! m_callbacks.insert(map_type::value_type(format, create_function)).second) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
create_parser_type get_creator_function(const osmium::io::File& file) {
|
|
|
|
auto it = m_callbacks.find(file.format());
|
|
|
|
if (it == m_callbacks.end()) {
|
|
|
|
throw unsupported_file_format_error(
|
|
|
|
std::string("Can not open file '") +
|
|
|
|
file.filename() +
|
|
|
|
"' with type '" +
|
|
|
|
as_string(file.format()) +
|
|
|
|
"'. No support for reading this format in this program.");
|
|
|
|
}
|
|
|
|
return it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
}; // class ParserFactory
|
|
|
|
|
|
|
|
} // namespace detail
|
|
|
|
|
|
|
|
} // namespace io
|
|
|
|
|
|
|
|
} // namespace osmium
|
|
|
|
|
|
|
|
#endif // OSMIUM_IO_DETAIL_INPUT_FORMAT_HPP
|