#ifndef PROTOZERO_PBF_MESSAGE_HPP #define PROTOZERO_PBF_MESSAGE_HPP /***************************************************************************** protozero - Minimalistic protocol buffer decoder and encoder in C++. This file is from https://github.com/mapbox/protozero where you can find more documentation. *****************************************************************************/ /** * @file pbf_message.hpp * * @brief Contains the pbf_message template class. */ #include #include #include namespace protozero { /** * This class represents a protobuf message. Either a top-level message or * a nested sub-message. Top-level messages can be created from any buffer * with a pointer and length: * * @code * enum class Message : protozero::pbf_tag_type { * ... * }; * * std::string buffer; * // fill buffer... * pbf_message message(buffer.data(), buffer.size()); * @endcode * * Sub-messages are created using get_message(): * * @code * enum class SubMessage : protozero::pbf_tag_type { * ... * }; * * pbf_message message(...); * message.next(); * pbf_message submessage = message.get_message(); * @endcode * * All methods of the pbf_message class except get_bytes() and get_string() * provide the strong exception guarantee, ie they either succeed or do not * change the pbf_message object they are called on. Use the get_data() method * instead of get_bytes() or get_string(), if you need this guarantee. * * This template class is based on the pbf_reader class and has all the same * methods. The difference is that whereever the pbf_reader class takes an * integer tag, this template class takes a tag of the template type T. * * Read the tutorial to understand how this class is used. */ template class pbf_message : public pbf_reader { static_assert(std::is_same::type>::value, "T must be enum with underlying type protozero::pbf_tag_type"); public: using enum_type = T; template pbf_message(Args&&... args) noexcept : pbf_reader(std::forward(args)...) { } bool next() { return pbf_reader::next(); } bool next(T next_tag) { return pbf_reader::next(pbf_tag_type(next_tag)); } bool next(T next_tag, pbf_wire_type type) { return pbf_reader::next(pbf_tag_type(next_tag), type); } T tag() const noexcept { return T(pbf_reader::tag()); } }; // class pbf_message } // end namespace protozero #endif // PROTOZERO_PBF_MESSAGE_HPP