Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
102 lines
2.8 KiB
102 lines
2.8 KiB
|
|
#include <iostream> |
|
|
|
#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 T> |
|
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<T*>(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<T*>(c); |
|
try { |
|
str->append(buf, n); |
|
return n; |
|
} catch (const std::exception&) { |
|
return 0; |
|
} |
|
} |
|
}; |
|
|
|
class StringSink { |
|
public: |
|
template <class T> |
|
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<T>::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; |
|
} |
|
}
|
|
|