#include #include "google/protobuf/descriptor.upb.h" #include "google/protobuf/descriptor.upbdefs.h" #include "tests/test_util.h" #include "tests/upb_test.h" #include "upb/pb/decoder.h" #include "upb/pb/encoder.h" #include "upb/port_def.inc" #include "upb/upb.hpp" template class FillStringHandler { public: static void SetHandler(upb_byteshandler* handler) { upb_byteshandler_setstartstr(handler, &FillStringHandler::StartString, NULL); upb_byteshandler_setstring(handler, &FillStringHandler::StringBuf, NULL); } private: // TODO(haberman): add UpbBind/UpbMakeHandler support to BytesHandler so these // can be prettier callbacks. static void* StartString(void *c, const void *hd, size_t size) { UPB_UNUSED(hd); UPB_UNUSED(size); T* str = static_cast(c); str->clear(); return c; } static size_t StringBuf(void* c, const void* hd, const char* buf, size_t n, const upb_bufhandle* h) { UPB_UNUSED(hd); UPB_UNUSED(h); T* str = static_cast(c); try { str->append(buf, n); return n; } catch (const std::exception&) { return 0; } } }; class StringSink { public: template explicit StringSink(T* target) { // TODO(haberman): we need to avoid rebuilding a new handler every time, // but with class globals disallowed for google3 C++ this is tricky. upb_byteshandler_init(&handler_); FillStringHandler::SetHandler(&handler_); input_.Reset(&handler_, target); } upb::BytesSink input() { return input_; } private: upb_byteshandler handler_; upb::BytesSink input_; }; void test_pb_roundtrip() { std::string input( google_protobuf_descriptor_proto_upbdefinit.descriptor.data, google_protobuf_descriptor_proto_upbdefinit.descriptor.size); std::cout << input.size() << "\n"; upb::SymbolTable symtab; upb::HandlerCache encoder_cache(upb::pb::EncoderPtr::NewCache()); upb::pb::CodeCache decoder_cache(&encoder_cache); upb::Arena arena; upb::Status status; upb::MessageDefPtr md( google_protobuf_FileDescriptorProto_getmsgdef(symtab.ptr())); ASSERT(md); const upb::Handlers *encoder_handlers = encoder_cache.Get(md); ASSERT(encoder_handlers); const upb::pb::DecoderMethodPtr method = decoder_cache.Get(md); std::string output; StringSink string_sink(&output); upb::pb::EncoderPtr encoder = upb::pb::EncoderPtr::Create(&arena, encoder_handlers, string_sink.input()); upb::pb::DecoderPtr decoder = upb::pb::DecoderPtr::Create(&arena, method, encoder.input(), &status); bool ok = upb::PutBuffer(input, decoder.input()); ASSERT(ok); ASSERT(input == output); } extern "C" { int run_tests(int argc, char *argv[]) { UPB_UNUSED(argc); UPB_UNUSED(argv); test_pb_roundtrip(); return 0; } }