#ifndef OSMIUM_BUILDER_ATTR_HPP #define OSMIUM_BUILDER_ATTR_HPP /* This file is part of Osmium (http://osmcode.org/libosmium). Copyright 2013-2016 Jochen Topf 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace osmium { namespace builder { namespace detail { #ifdef _MSC_VER // workaround for bug in MSVC template struct is_handled_by; template struct is_handled_by { static constexpr bool value = false; }; template struct is_handled_by { static constexpr bool value = std::is_base_of::value || is_handled_by::value; }; template struct are_all_handled_by; template struct are_all_handled_by { static constexpr bool value = std::is_base_of::value; }; template struct are_all_handled_by { static constexpr bool value = std::is_base_of::value && are_all_handled_by::value; }; #else // True if Predicate matches for none of the types Ts template class Predicate, typename... Ts> struct static_none_of : std::is_same::type...>, std::tuple::type..., std::false_type>> {}; // True if Predicate matches for all of the types Ts template class Predicate, typename... Ts> struct static_all_of : std::is_same::type...>, std::tuple::type..., std::true_type>> {}; // True if THandler is derived from the handler for at least one of the types in TTypes template struct is_handled_by { template using HasHandler = std::is_base_of; static constexpr bool value = !static_none_of::value; }; // True if THandler is derived from the handlers of all the types in TTypes template struct are_all_handled_by { template using HasHandler = std::is_base_of; static constexpr bool value = static_all_of::value; }; #endif // Wraps any type, so that we can derive from it template struct type_wrapper { using type = TType; TType value; constexpr explicit type_wrapper(const TType& v) : value(v) { } }; // struct type_wrapper // Small wrapper for begin/end iterator template struct iterator_wrapper { using type = TType; TType first; TType last; constexpr iterator_wrapper(TType begin, TType end) : first(begin), last(end) {} constexpr TType begin() const { return first; } constexpr TType end() const { return last; } }; // struct iterator_wrapper struct entity_handler {}; struct object_handler; struct node_handler; struct tags_handler; struct nodes_handler; struct members_handler; struct changeset_handler; struct discussion_handler; struct ring_handler; } // namespace detail #define OSMIUM_ATTRIBUTE(_handler, _name, _type) \ struct _name : public osmium::builder::detail::type_wrapper<_type> { \ using handler = osmium::builder::detail::_handler; #define OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(_handler, _name, _type) \ OSMIUM_ATTRIBUTE(_handler, _name, _type) \ constexpr explicit _name(std::add_const<_type>::type& value) : \ type_wrapper(value) {} \ } #define OSMIUM_ATTRIBUTE_ITER(_handler, _name) \ template \ struct _name : public osmium::builder::detail::iterator_wrapper { \ using handler = osmium::builder::detail::_handler; \ constexpr _name(TIterator first, TIterator last) : \ osmium::builder::detail::iterator_wrapper(first, last) {} \ } namespace attr { OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(object_handler, _id, osmium::object_id_type); OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(object_handler, _version, osmium::object_version_type); OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(entity_handler, _uid, osmium::user_id_type); OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(entity_handler, _cid, osmium::changeset_id_type); OSMIUM_ATTRIBUTE(object_handler, _deleted, bool) constexpr explicit _deleted(bool value = true) noexcept : type_wrapper(value) {} }; OSMIUM_ATTRIBUTE(object_handler, _visible, bool) constexpr explicit _visible(bool value = true) noexcept : type_wrapper(value) {} }; OSMIUM_ATTRIBUTE(object_handler, _timestamp, osmium::Timestamp) constexpr explicit _timestamp(const osmium::Timestamp& value) noexcept : type_wrapper(value) {} constexpr explicit _timestamp(time_t value) noexcept : type_wrapper(osmium::Timestamp{value}) {} constexpr explicit _timestamp(uint32_t value) noexcept : type_wrapper(osmium::Timestamp{value}) {} explicit _timestamp(const char* value) : type_wrapper(osmium::Timestamp{value}) {} explicit _timestamp(const std::string& value) : type_wrapper(osmium::Timestamp{value}) {} }; OSMIUM_ATTRIBUTE(node_handler, _location, osmium::Location) constexpr explicit _location(const osmium::Location& value) noexcept : type_wrapper(value) {} explicit _location(double lat, double lon) : type_wrapper(osmium::Location{lat, lon}) {} }; OSMIUM_ATTRIBUTE(entity_handler, _user, const char*) constexpr explicit _user(const char* val) noexcept : type_wrapper(val) {} explicit _user(const std::string& val) noexcept : type_wrapper(val.c_str()) {} }; using pair_of_cstrings = std::pair; using pair_of_strings = std::pair; class member_type { osmium::item_type m_type; osmium::object_id_type m_ref; const char* m_role; public: constexpr member_type(osmium::item_type type, osmium::object_id_type ref, const char* role = "") noexcept : m_type(type), m_ref(ref), m_role(role) { } constexpr osmium::item_type type() const noexcept { return m_type; } constexpr osmium::object_id_type ref() const noexcept { return m_ref; } constexpr const char* role() const noexcept { return m_role; } }; // class member_type class member_type_string { osmium::item_type m_type; osmium::object_id_type m_ref; std::string m_role; public: member_type_string(osmium::item_type type, osmium::object_id_type ref, std::string&& role) : m_type(type), m_ref(ref), m_role(std::move(role)) { } osmium::item_type type() const noexcept { return m_type; } osmium::object_id_type ref() const noexcept { return m_ref; } const char* role() const noexcept { return m_role.c_str(); } }; // class member_type_string class comment_type { osmium::Timestamp m_date; osmium::user_id_type m_uid; const char* m_user; const char* m_text; public: constexpr comment_type(osmium::Timestamp date, osmium::user_id_type uid, const char* user, const char* text) noexcept : m_date(date), m_uid(uid), m_user(user), m_text(text) { } constexpr osmium::Timestamp date() const noexcept { return m_date; } constexpr osmium::user_id_type uid() const noexcept { return m_uid; } constexpr const char* user() const noexcept { return m_user; } constexpr const char* text() const noexcept { return m_text; } }; // class comment_type namespace detail { OSMIUM_ATTRIBUTE_ITER(tags_handler, tags_from_iterator_pair); OSMIUM_ATTRIBUTE_ITER(nodes_handler, nodes_from_iterator_pair); OSMIUM_ATTRIBUTE_ITER(members_handler, members_from_iterator_pair); OSMIUM_ATTRIBUTE_ITER(discussion_handler, comments_from_iterator_pair); OSMIUM_ATTRIBUTE_ITER(ring_handler, outer_ring_from_iterator_pair); OSMIUM_ATTRIBUTE_ITER(ring_handler, inner_ring_from_iterator_pair); } // namespace detail OSMIUM_ATTRIBUTE(tags_handler, _tag, pair_of_cstrings) explicit _tag(const pair_of_cstrings& value) noexcept : type_wrapper(value) {} explicit _tag(const std::pair& value) : type_wrapper(pair_of_cstrings{value.first, value.second}) {} explicit _tag(const std::pair& value) : type_wrapper(pair_of_cstrings{value.first, value.second}) {} explicit _tag(const std::pair& value) : type_wrapper(pair_of_cstrings{value.first, value.second}) {} explicit _tag(const pair_of_strings& value) : type_wrapper(std::make_pair(value.first.c_str(), value.second.c_str())) {} explicit _tag(const char* key, const char* val) : type_wrapper(std::make_pair(key, val)) {} explicit _tag(const std::string& key, const std::string& val) : type_wrapper(std::make_pair(key.c_str(), val.c_str())) {} }; template inline constexpr detail::tags_from_iterator_pair _tags(TTagIterator first, TTagIterator last) { return detail::tags_from_iterator_pair(first, last); } template inline detail::tags_from_iterator_pair _tags(const TContainer& container) { return detail::tags_from_iterator_pair(std::begin(container), std::end(container)); } using tag_ilist = std::initializer_list>; inline detail::tags_from_iterator_pair _tags(const tag_ilist& container) { return detail::tags_from_iterator_pair(std::begin(container), std::end(container)); } OSMIUM_ATTRIBUTE(nodes_handler, _node, osmium::NodeRef) constexpr explicit _node(osmium::object_id_type value) noexcept : type_wrapper(NodeRef{value}) {} constexpr explicit _node(const NodeRef& value) noexcept : type_wrapper(value) {} }; template inline constexpr detail::nodes_from_iterator_pair _nodes(TIdIterator first, TIdIterator last) { return detail::nodes_from_iterator_pair(first, last); } template inline detail::nodes_from_iterator_pair _nodes(const TContainer& container) { return detail::nodes_from_iterator_pair(std::begin(container), std::end(container)); } using object_id_ilist = std::initializer_list; inline detail::nodes_from_iterator_pair _nodes(const object_id_ilist& container) { return detail::nodes_from_iterator_pair(std::begin(container), std::end(container)); } using node_ref_ilist = std::initializer_list; inline detail::nodes_from_iterator_pair _nodes(const node_ref_ilist& container) { return detail::nodes_from_iterator_pair(std::begin(container), std::end(container)); } OSMIUM_ATTRIBUTE(members_handler, _member, member_type) constexpr explicit _member(const member_type& value) noexcept : type_wrapper(value) {} constexpr explicit _member(osmium::item_type type, osmium::object_id_type id) noexcept : type_wrapper({type, id}) {} constexpr explicit _member(osmium::item_type type, osmium::object_id_type id, const char* role) noexcept : type_wrapper({type, id, role}) {} explicit _member(osmium::item_type type, osmium::object_id_type id, const std::string& role) noexcept : type_wrapper({type, id, role.c_str()}) {} explicit _member(const osmium::RelationMember& member) noexcept : type_wrapper({member.type(), member.ref(), member.role()}) {} }; template inline constexpr detail::members_from_iterator_pair _members(TMemberIterator first, TMemberIterator last) { return detail::members_from_iterator_pair(first, last); } template inline detail::members_from_iterator_pair _members(const TContainer& container) { return detail::members_from_iterator_pair(std::begin(container), std::end(container)); } using member_ilist = std::initializer_list; inline detail::members_from_iterator_pair _members(const member_ilist& container) { return detail::members_from_iterator_pair(std::begin(container), std::end(container)); } OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(changeset_handler, _num_changes, osmium::num_changes_type); OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(changeset_handler, _num_comments, osmium::num_comments_type); OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(changeset_handler, _created_at, osmium::Timestamp); OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR(changeset_handler, _closed_at, osmium::Timestamp); OSMIUM_ATTRIBUTE(discussion_handler, _comment, comment_type) constexpr explicit _comment(const comment_type& value) noexcept : type_wrapper(value) {} explicit _comment(const osmium::ChangesetComment& comment) noexcept : type_wrapper({comment.date(), comment.uid(), comment.user(), comment.text()}) {} }; template inline constexpr detail::comments_from_iterator_pair _comments(TCommentIterator first, TCommentIterator last) { return detail::comments_from_iterator_pair(first, last); } template inline detail::comments_from_iterator_pair _comments(const TContainer& container) { return detail::comments_from_iterator_pair(std::begin(container), std::end(container)); } using comment_ilist = std::initializer_list; inline detail::comments_from_iterator_pair _comments(const comment_ilist& container) { return detail::comments_from_iterator_pair(std::begin(container), std::end(container)); } template inline constexpr detail::outer_ring_from_iterator_pair _outer_ring(TIdIterator first, TIdIterator last) { return detail::outer_ring_from_iterator_pair(first, last); } template inline detail::outer_ring_from_iterator_pair _outer_ring(const TContainer& container) { return detail::outer_ring_from_iterator_pair(std::begin(container), std::end(container)); } using object_id_ilist = std::initializer_list; inline detail::outer_ring_from_iterator_pair _outer_ring(const object_id_ilist& container) { return detail::outer_ring_from_iterator_pair(std::begin(container), std::end(container)); } using node_ref_ilist = std::initializer_list; inline detail::outer_ring_from_iterator_pair _outer_ring(const node_ref_ilist& container) { return detail::outer_ring_from_iterator_pair(std::begin(container), std::end(container)); } template inline constexpr detail::inner_ring_from_iterator_pair _inner_ring(TIdIterator first, TIdIterator last) { return detail::inner_ring_from_iterator_pair(first, last); } template inline detail::inner_ring_from_iterator_pair _inner_ring(const TContainer& container) { return detail::inner_ring_from_iterator_pair(std::begin(container), std::end(container)); } using object_id_ilist = std::initializer_list; inline detail::inner_ring_from_iterator_pair _inner_ring(const object_id_ilist& container) { return detail::inner_ring_from_iterator_pair(std::begin(container), std::end(container)); } using node_ref_ilist = std::initializer_list; inline detail::inner_ring_from_iterator_pair _inner_ring(const node_ref_ilist& container) { return detail::inner_ring_from_iterator_pair(std::begin(container), std::end(container)); } } // namespace attr #undef OSMIUM_ATTRIBUTE_ITER #undef OSMIUM_ATTRIBUTE_WITH_CONSTRUCTOR #undef OSMIUM_ATTRIBUTE namespace detail { struct changeset_handler : public entity_handler { template static void set_value(osmium::Changeset&, const TDummy&) noexcept { } static void set_value(osmium::Changeset& changeset, attr::_cid id) noexcept { changeset.set_id(id.value); } static void set_value(osmium::Changeset& changeset, attr::_num_changes num_changes) noexcept { changeset.set_num_changes(num_changes.value); } static void set_value(osmium::Changeset& changeset, attr::_num_comments num_comments) noexcept { changeset.set_num_comments(num_comments.value); } static void set_value(osmium::Changeset& changeset, attr::_created_at created_at) noexcept { changeset.set_created_at(created_at.value); } static void set_value(osmium::Changeset& changeset, attr::_closed_at closed_at) noexcept { changeset.set_closed_at(closed_at.value); } static void set_value(osmium::Changeset& changeset, attr::_uid uid) noexcept { changeset.set_uid(uid.value); } }; struct object_handler : public entity_handler { template static void set_value(osmium::OSMObject&, const TDummy&) noexcept { } static void set_value(osmium::OSMObject& object, attr::_id id) noexcept { object.set_id(id.value); } static void set_value(osmium::OSMObject& object, attr::_version version) noexcept { object.set_version(version.value); } static void set_value(osmium::OSMObject& object, attr::_visible visible) noexcept { object.set_visible(visible.value); } static void set_value(osmium::OSMObject& object, attr::_deleted deleted) noexcept { object.set_deleted(deleted.value); } static void set_value(osmium::OSMObject& object, attr::_timestamp timestamp) noexcept { object.set_timestamp(timestamp.value); } static void set_value(osmium::OSMObject& object, attr::_cid changeset) noexcept { object.set_changeset(changeset.value); } static void set_value(osmium::OSMObject& object, attr::_uid uid) noexcept { object.set_uid(uid.value); } }; // object_handler struct node_handler : public object_handler { using object_handler::set_value; static void set_value(osmium::Node& node, attr::_location location) noexcept { node.set_location(location.value); } }; // node_handler template inline void add_basic(TBuilder& builder, const TArgs&... args) noexcept { (void)std::initializer_list{ (THandler::set_value(builder.object(), args), 0)... }; } // ============================================================== template inline constexpr const char* get_user(const attr::_user& user, const TArgs&...) noexcept { return user.value; } inline constexpr const char* get_user() noexcept { return ""; } template inline constexpr typename std::enable_if::value, const char*>::type get_user(const TFirst&, const TRest&... args) noexcept { return get_user(args...); } template inline void add_user(TBuilder& builder, const TArgs&... args) { builder.add_user(get_user(args...)); } // ============================================================== struct tags_handler { template static void set_value(TagListBuilder&, const TDummy&) noexcept { } static void set_value(TagListBuilder& builder, const attr::_tag& tag) { builder.add_tag(tag.value); } template static void set_value(TagListBuilder& builder, const attr::detail::tags_from_iterator_pair& tags) { for (const auto& tag : tags) { builder.add_tag(tag); } } }; // struct tags_handler struct nodes_handler { template static void set_value(WayNodeListBuilder&, const TDummy&) noexcept { } static void set_value(WayNodeListBuilder& builder, const attr::_node& node_ref) { builder.add_node_ref(node_ref.value); } template static void set_value(WayNodeListBuilder& builder, const attr::detail::nodes_from_iterator_pair& nodes) { for (const auto& ref : nodes) { builder.add_node_ref(ref); } } }; // struct nodes_handler struct members_handler { template static void set_value(RelationMemberListBuilder&, const TDummy&) noexcept { } static void set_value(RelationMemberListBuilder& builder, const attr::_member& member) { builder.add_member(member.value.type(), member.value.ref(), member.value.role()); } template static void set_value(RelationMemberListBuilder& builder, const attr::detail::members_from_iterator_pair& members) { for (const auto& member : members) { builder.add_member(member.type(), member.ref(), member.role()); } } }; // struct members_handler struct discussion_handler { template static void set_value(ChangesetDiscussionBuilder&, const TDummy&) noexcept { } static void set_value(ChangesetDiscussionBuilder& builder, const attr::_comment& comment) { builder.add_comment(comment.value.date(), comment.value.uid(), comment.value.user()); builder.add_comment_text(comment.value.text()); } template static void set_value(ChangesetDiscussionBuilder& builder, const attr::detail::comments_from_iterator_pair& comments) { for (const auto& comment : comments) { builder.add_comment(comment.date(), comment.uid(), comment.user()); builder.add_comment_text(comment.text()); } } }; // struct discussion_handler struct ring_handler { template static void set_value(AreaBuilder&, const TDummy&) noexcept { } template static void set_value(AreaBuilder& parent, const attr::detail::outer_ring_from_iterator_pair& nodes) { OuterRingBuilder builder(parent.buffer(), &parent); for (const auto& ref : nodes) { builder.add_node_ref(ref); } } template static void set_value(AreaBuilder& parent, const attr::detail::inner_ring_from_iterator_pair& nodes) { InnerRingBuilder builder(parent.buffer(), &parent); for (const auto& ref : nodes) { builder.add_node_ref(ref); } } }; // struct ring_handler // ============================================================== template inline typename std::enable_if::value>::type add_list(osmium::builder::Builder&, const TArgs&...) noexcept { } template inline typename std::enable_if::value>::type add_list(osmium::builder::Builder& parent, const TArgs&... args) { TBuilder builder(parent.buffer(), &parent); (void)std::initializer_list{ (THandler::set_value(builder, args), 0)... }; } struct any_node_handlers : public node_handler, public tags_handler {}; struct any_way_handlers : public object_handler, public tags_handler, public nodes_handler {}; struct any_relation_handlers : public object_handler, public tags_handler, public members_handler {}; struct any_area_handlers : public object_handler, public tags_handler, public ring_handler {}; struct any_changeset_handlers : public changeset_handler, public tags_handler, public discussion_handler {}; } // namespace detail /** * Create a node using the given arguments and add it to the given buffer. * * @param buffer The buffer to which the node will be added. * @param args The attributes of the node. * @returns The position in the buffer where this node was added. */ template inline size_t add_node(osmium::memory::Buffer& buffer, const TArgs&... args) { static_assert(sizeof...(args) > 0, "add_node() must have buffer and at least one additional argument"); static_assert(detail::are_all_handled_by::value, "Attribute not allowed in add_node()"); NodeBuilder builder(buffer); detail::add_basic(builder, args...); detail::add_user(builder, args...); detail::add_list(builder, args...); return buffer.commit(); } /** * Create a way using the given arguments and add it to the given buffer. * * @param buffer The buffer to which the way will be added. * @param args The attributes of the way. * @returns The position in the buffer where this way was added. */ template inline size_t add_way(osmium::memory::Buffer& buffer, const TArgs&... args) { static_assert(sizeof...(args) > 0, "add_way() must have buffer and at least one additional argument"); static_assert(detail::are_all_handled_by::value, "Attribute not allowed in add_way()"); WayBuilder builder(buffer); detail::add_basic(builder, args...); detail::add_user(builder, args...); detail::add_list(builder, args...); detail::add_list(builder, args...); return buffer.commit(); } /** * Create a relation using the given arguments and add it to the given buffer. * * @param buffer The buffer to which the relation will be added. * @param args The attributes of the relation. * @returns The position in the buffer where this relation was added. */ template inline size_t add_relation(osmium::memory::Buffer& buffer, const TArgs&... args) { static_assert(sizeof...(args) > 0, "add_relation() must have buffer and at least one additional argument"); static_assert(detail::are_all_handled_by::value, "Attribute not allowed in add_relation()"); RelationBuilder builder(buffer); detail::add_basic(builder, args...); detail::add_user(builder, args...); detail::add_list(builder, args...); detail::add_list(builder, args...); return buffer.commit(); } /** * Create a changeset using the given arguments and add it to the given buffer. * * @param buffer The buffer to which the changeset will be added. * @param args The attributes of the changeset. * @returns The position in the buffer where this changeset was added. */ template inline size_t add_changeset(osmium::memory::Buffer& buffer, const TArgs&... args) { static_assert(sizeof...(args) > 0, "add_changeset() must have buffer and at least one additional argument"); static_assert(detail::are_all_handled_by::value, "Attribute not allowed in add_changeset()"); ChangesetBuilder builder(buffer); detail::add_basic(builder, args...); detail::add_user(builder, args...); detail::add_list(builder, args...); detail::add_list(builder, args...); return buffer.commit(); } /** * Create a area using the given arguments and add it to the given buffer. * * @param buffer The buffer to which the area will be added. * @param args The attributes of the area. * @returns The position in the buffer where this area was added. */ template inline size_t add_area(osmium::memory::Buffer& buffer, const TArgs&... args) { static_assert(sizeof...(args) > 0, "add_area() must have buffer and at least one additional argument"); static_assert(detail::are_all_handled_by::value, "Attribute not allowed in add_area()"); AreaBuilder builder(buffer); detail::add_basic(builder, args...); detail::add_user(builder, args...); detail::add_list(builder, args...); (void)std::initializer_list{ (detail::ring_handler::set_value(builder, args), 0)... }; return buffer.commit(); } /** * Create a WayNodeList using the given arguments and add it to the given buffer. * * @param buffer The buffer to which the list will be added. * @param args The contents of the list. * @returns The position in the buffer where this list was added. */ template inline size_t add_way_node_list(osmium::memory::Buffer& buffer, const TArgs&... args) { static_assert(sizeof...(args) > 0, "add_way_node_list() must have buffer and at least one additional argument"); static_assert(detail::are_all_handled_by::value, "Attribute not allowed in add_way_node_list()"); { WayNodeListBuilder builder(buffer); (void)std::initializer_list{ (detail::nodes_handler::set_value(builder, args), 0)... }; } return buffer.commit(); } /** * Create a TagList using the given arguments and add it to the given buffer. * * @param buffer The buffer to which the list will be added. * @param args The contents of the list. * @returns The position in the buffer where this list was added. */ template inline size_t add_tag_list(osmium::memory::Buffer& buffer, const TArgs&... args) { static_assert(sizeof...(args) > 0, "add_tag_list() must have buffer and at least one additional argument"); static_assert(detail::are_all_handled_by::value, "Attribute not allowed in add_tag_list()"); { TagListBuilder builder(buffer); (void)std::initializer_list{ (detail::tags_handler::set_value(builder, args), 0)... }; } return buffer.commit(); } } // namespace builder } // namespace osmium #endif // OSMIUM_BUILDER_ATTR_HPP