mirror of
https://github.com/google/flatbuffers.git
synced 2026-06-02 12:05:50 +00:00
* Rework flatbuffers + gRPC integration - Introduce `flatbuffers::grpc::Message<T>`, a `grpc_slice`-backed message buffer that handles refcounting and allows flatbuffers to transfer ownership to gRPC efficiently. This replaces `flatbuffers::BufferRef<T>`, which required a copy call and was also unsafe w.r.t. buffer lifetime. - Introduce `flatbuffers::grpc::MessageBuilder`, a gRPC-specific builder that forces a `grpc_slice`-backed allocator and also adds some helpful `Message<T>`-related methods. - Update serializers accordingly (now zero-copy between flatbuffers and gRPC). * gRPC: verify messages by default, but allow user to override * gRPC: fix some formatting issues * Disable verification by default, but add helper method * Make FlatBufferBuilder fields protected + remove vec accessor * Use bool add_ref parameter to toggle refcount incr * Remove unnecessary inline specifiers * Fix formatting * Use auto * Remove empty lines * Use grpc_slice helper macros * Simplify reset code * Disable Message copy ctor and assignment by default * Remove unused member * Enable gRPC verification by default * Use auto * Bake in message verification (remove template specialization) * Add RoundUp func * Consolidate gRPC message copy flag * Make vector_downward allocations fully lazy * Test message verification failure code/message * Add grpctest verification test comments * Simplify reallocate implementation * Make initial_size a size_t * Use ternary op for growth_policy * Use truthiness rather than dont explicit nullptr check * Indent preprocessor directives * Remove grpc message copy/assignment * Fix a few bugs * Add gRPC example * Add basic gRPC docs * Use doxygen EXAMPLE_PATH + @include * Reference example fbs in grpc docs * Move gRPC examples into grpc/samples * Fix pointer/reference formatting * Use std::function rather than templated callback func * Create fresh message builder for each request * Use Clear() in Reset() impl * Use FLATBUFFERS_CONSTEXPR
81 lines
2.7 KiB
C++
81 lines
2.7 KiB
C++
#include "greeter.grpc.fb.h"
|
|
#include "greeter_generated.h"
|
|
|
|
#include <grpc++/grpc++.h>
|
|
|
|
#include <iostream>
|
|
#include <memory>
|
|
#include <string>
|
|
|
|
class GreeterServiceImpl final : public Greeter::Service {
|
|
virtual grpc::Status SayHello(
|
|
grpc::ServerContext *context,
|
|
const flatbuffers::grpc::Message<HelloRequest> *request_msg,
|
|
flatbuffers::grpc::Message<HelloReply> *response_msg) override {
|
|
// flatbuffers::grpc::MessageBuilder mb_;
|
|
// We call GetRoot to "parse" the message. Verification is already
|
|
// performed by default. See the notes below for more details.
|
|
const HelloRequest *request = request_msg->GetRoot();
|
|
|
|
// Fields are retrieved as usual with FlatBuffers
|
|
const std::string &name = request->name()->str();
|
|
|
|
// `flatbuffers::grpc::MessageBuilder` is a `FlatBufferBuilder` with a
|
|
// special allocator for efficient gRPC buffer transfer, but otherwise
|
|
// usage is the same as usual.
|
|
auto msg_offset = mb_.CreateString("Hello, " + name);
|
|
auto hello_offset = CreateHelloReply(mb_, msg_offset);
|
|
mb_.Finish(hello_offset);
|
|
|
|
// The `ReleaseMessage<T>()` function detaches the message from the
|
|
// builder, so we can transfer the resopnse to gRPC while simultaneously
|
|
// detaching that memory buffer from the builer.
|
|
*response_msg = mb_.ReleaseMessage<HelloReply>();
|
|
assert(response_msg->Verify());
|
|
|
|
// Return an OK status.
|
|
return grpc::Status::OK;
|
|
}
|
|
|
|
virtual grpc::Status SayManyHellos(
|
|
grpc::ServerContext *context,
|
|
const flatbuffers::grpc::Message<ManyHellosRequest> *request_msg,
|
|
grpc::ServerWriter<flatbuffers::grpc::Message<HelloReply>> *writer)
|
|
override {
|
|
// The streaming usage below is simply a combination of standard gRPC
|
|
// streaming with the FlatBuffers usage shown above.
|
|
const ManyHellosRequest *request = request_msg->GetRoot();
|
|
const std::string &name = request->name()->str();
|
|
int num_greetings = request->num_greetings();
|
|
|
|
for (int i = 0; i < num_greetings; i++) {
|
|
auto msg_offset = mb_.CreateString("Many hellos, " + name);
|
|
auto hello_offset = CreateHelloReply(mb_, msg_offset);
|
|
mb_.Finish(hello_offset);
|
|
writer->Write(mb_.ReleaseMessage<HelloReply>());
|
|
}
|
|
|
|
return grpc::Status::OK;
|
|
}
|
|
|
|
flatbuffers::grpc::MessageBuilder mb_;
|
|
};
|
|
|
|
void RunServer() {
|
|
std::string server_address("0.0.0.0:50051");
|
|
GreeterServiceImpl service;
|
|
|
|
grpc::ServerBuilder builder;
|
|
builder.AddListeningPort(server_address, grpc::InsecureServerCredentials());
|
|
builder.RegisterService(&service);
|
|
std::unique_ptr<grpc::Server> server(builder.BuildAndStart());
|
|
std::cerr << "Server listening on " << server_address << std::endl;
|
|
|
|
server->Wait();
|
|
}
|
|
|
|
int main(int argc, const char *argv[]) {
|
|
RunServer();
|
|
return 0;
|
|
}
|