BRPC使用示例
BRPC使用示例
proto文件
syntax = "proto2";
package dfs;
option cc_generic_services = true;
service NameNodeService {
rpc Register(RegisterRequest) returns (RegisterResponse){}
rpc heartbeat(HeartbeatRequest) returns (HeartbeatResponse){}
rpc mkdir(MkDirRequest) returns (MkDirResponse){}
rpc shutdown(ShutdownRequest) returns (ShutdownResponse){}
rpc fetchEditsLog(FetchEditsLogRequest) returns (FetchEditsLogResponse){}
rpc updateCheckpointTxid(UpdateCheckpointTxidRequest) returns (UpdateCheckpointTxidResponse){}
}
message RegisterRequest{
required string ip = 1;
required string hostname = 2;
}
message RegisterResponse{
required int32 status = 1;
}
message HeartbeatRequest{
required string ip = 1;
required string hostname = 2;
}
message HeartbeatResponse{
required int32 status = 1;
}
message MkDirRequest{
required string path = 1;
}
message MkDirResponse{
required int32 status = 1;
}
message ShutdownRequest{
required int32 code = 1;
}
message ShutdownResponse{
required int32 status = 1;
}
message FetchEditsLogRequest{
required int64 syncedTxid = 1;
}
message FetchEditsLogResponse{
required string editsLog = 1;
}
message UpdateCheckpointTxidRequest{
required int64 txid = 1;
}
message UpdateCheckpointTxidResponse{
required int32 status = 1;
}
server
#include <brpc/server.h>
#include "dfs.pb.h"
#include <iostream>
namespace test {
class NameNodeServiceImpl : public dfs::NameNodeService {
public:
NameNodeServiceImpl() {};
virtual ~NameNodeServiceImpl() {};
virtual void Register(google::protobuf::RpcController* controller,
const dfs::RegisterRequest* request,
dfs::RegisterResponse* response,
google::protobuf::Closure* done) {
brpc::ClosureGuard done_guard(done);
brpc::Controller* cntl = static_cast<brpc::Controller*>(controller);
std::cout << "Server: " << std::endl;
std::cout << "ip: " << request->ip() << std::endl;
std::cout << "hostname: " << request->hostname() << std::endl;
response->set_status(0);
}
};
}
int main() {
brpc::Server server;
test::NameNodeServiceImpl namenode_service_impl;
if (server.AddService(&namenode_service_impl,
brpc::SERVER_DOESNT_OWN_SERVICE) != 0) {
std::cout << "Fail to add service" << std::endl;
return -1;
}
brpc::ServerOptions options;
options.idle_timeout_sec = -1;
if (server.Start(8000, &options) != 0) {
std::cout << "Fail to start EchoServer" << std::endl;
return -1;
}
// Wait until Ctrl-C is pressed, then Stop() and Join() the server.
server.RunUntilAskedToQuit();
return 0;
}
client
#include <brpc/channel.h>
#include "dfs.pb.h"
#include <iostream>
int main() {
brpc::Channel channel;
brpc::ChannelOptions options;
options.protocol = "baidu_std";
options.connection_type = "";
options.timeout_ms = 100;
options.max_retry = 3;
if (channel.Init("0.0.0.0:8000", "", &options) != 0) {
std::cout << "Fail to initialize channel" << std::endl;
return -1;
}
dfs::NameNodeService_Stub stub(&channel);
int log_id = 0;
while (!brpc::IsAskedToQuit()) {
dfs::RegisterRequest request;
dfs::RegisterResponse response;
brpc::Controller cntl;
request.set_ip("0.0.0.0");
request.set_hostname("localhost");
cntl.set_log_id(log_id++);
cntl.request_attachment().append("");
stub.Register(&cntl, &request, &response, NULL);
if (!cntl.Failed()) {
std::cout << "Received response from " << cntl.remote_side()
<< " to " << cntl.local_side()
<< ": " << response.status() << " (attached="
<< cntl.response_attachment() << ")"
<< " latency=" << cntl.latency_us() << "us" << std::endl;
} else {
std::cout << cntl.ErrorText() << std::endl;
}
usleep(1000 * 1000L);
}
std::cout << "TestClient is going to quit" << std::endl;
return 0;
}
CMakeList
cmake_minimum_required(VERSION 2.8.10)
project(echo_c++ C CXX)
option(LINK_SO "Whether examples are linked dynamically" OFF)
execute_process(
#COMMAND bash -c "find ${PROJECT_SOURCE_DIR}/../.. -type d -regex \".*output/include$\" | head -n1 | xargs dirname | tr -d '\n'"
COMMAND bash -c "find /root/git/BRPC -type d -regex \".*output/include$\" | head -n1 | xargs dirname | tr -d '\n'"
OUTPUT_VARIABLE OUTPUT_PATH
)
set(CMAKE_PREFIX_PATH ${OUTPUT_PATH})
include(FindThreads)
include(FindProtobuf)
protobuf_generate_cpp(PROTO_SRC PROTO_HEADER dfs.proto)
# include PROTO_HEADER
include_directories(${CMAKE_CURRENT_BINARY_DIR})
# Search for libthrift* by best effort. If it is not found and brpc is
# compiled with thrift protocol enabled, a link error would be reported.
find_library(THRIFT_LIB NAMES thrift)
if (NOT THRIFT_LIB)
set(THRIFT_LIB "")
endif()
find_library(THRIFTNB_LIB NAMES thriftnb)
if (NOT THRIFTNB_LIB)
set(THRIFTNB_LIB "")
endif()
find_path(BRPC_INCLUDE_PATH NAMES brpc/server.h)
if(LINK_SO)
find_library(BRPC_LIB NAMES brpc)
else()
find_library(BRPC_LIB NAMES libbrpc.a brpc)
endif()
if((NOT BRPC_INCLUDE_PATH) OR (NOT BRPC_LIB))
message(FATAL_ERROR "Fail to find brpc")
endif()
include_directories(${BRPC_INCLUDE_PATH})
find_path(GFLAGS_INCLUDE_PATH gflags/gflags.h)
find_library(GFLAGS_LIBRARY NAMES gflags libgflags)
if((NOT GFLAGS_INCLUDE_PATH) OR (NOT GFLAGS_LIBRARY))
message(FATAL_ERROR "Fail to find gflags")
endif()
include_directories(${GFLAGS_INCLUDE_PATH})
execute_process(
COMMAND bash -c "grep \"namespace [_A-Za-z0-9]\\+ {\" ${GFLAGS_INCLUDE_PATH}/gflags/gflags_declare.h | head -1 | awk '{print $2}' | tr -d '\n'"
OUTPUT_VARIABLE GFLAGS_NS
)
if(${GFLAGS_NS} STREQUAL "GFLAGS_NAMESPACE")
execute_process(
COMMAND bash -c "grep \"#define GFLAGS_NAMESPACE [_A-Za-z0-9]\\+\" ${GFLAGS_INCLUDE_PATH}/gflags/gflags_declare.h | head -1 | awk '{print $3}' | tr -d '\n'"
OUTPUT_VARIABLE GFLAGS_NS
)
endif()
if(CMAKE_SYSTEM_NAME STREQUAL "Darwin")
include(CheckFunctionExists)
CHECK_FUNCTION_EXISTS(clock_gettime HAVE_CLOCK_GETTIME)
if(NOT HAVE_CLOCK_GETTIME)
set(DEFINE_CLOCK_GETTIME "-DNO_CLOCK_GETTIME_IN_MAC")
endif()
endif()
set(CMAKE_CPP_FLAGS "${DEFINE_CLOCK_GETTIME} -DGFLAGS_NS=${GFLAGS_NS}")
set(CMAKE_CXX_FLAGS "${CMAKE_CPP_FLAGS} -DNDEBUG -O2 -D__const__= -pipe -W -Wall -Wno-unused-parameter -fPIC -fno-omit-frame-pointer")
if(CMAKE_VERSION VERSION_LESS "3.1.3")
if(CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")
endif()
if(CMAKE_CXX_COMPILER_ID STREQUAL "Clang")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")
endif()
else()
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
endif()
find_path(LEVELDB_INCLUDE_PATH NAMES leveldb/db.h)
find_library(LEVELDB_LIB NAMES leveldb)
if ((NOT LEVELDB_INCLUDE_PATH) OR (NOT LEVELDB_LIB))
message(FATAL_ERROR "Fail to find leveldb")
endif()
include_directories(${LEVELDB_INCLUDE_PATH})
if(CMAKE_SYSTEM_NAME STREQUAL "Darwin")
set(OPENSSL_ROOT_DIR
"/usr/local/opt/openssl" # Homebrew installed OpenSSL
)
endif()
find_package(OpenSSL)
include_directories(${OPENSSL_INCLUDE_DIR})
set(DYNAMIC_LIB
${CMAKE_THREAD_LIBS_INIT}
${GFLAGS_LIBRARY}
${PROTOBUF_LIBRARIES}
${LEVELDB_LIB}
${OPENSSL_CRYPTO_LIBRARY}
${OPENSSL_SSL_LIBRARY}
${THRIFT_LIB}
${THRIFTNB_LIB}
dl
)
if(CMAKE_SYSTEM_NAME STREQUAL "Darwin")
set(DYNAMIC_LIB ${DYNAMIC_LIB}
pthread
"-framework CoreFoundation"
"-framework CoreGraphics"
"-framework CoreData"
"-framework CoreText"
"-framework Security"
"-framework Foundation"
"-Wl,-U,_MallocExtension_ReleaseFreeMemory"
"-Wl,-U,_ProfilerStart"
"-Wl,-U,_ProfilerStop")
endif()
add_executable(test_client client_test.cc ${PROTO_SRC} ${PROTO_HEADER})
add_executable(test_server server_test.cc ${PROTO_SRC} ${PROTO_HEADER})
target_link_libraries(test_client ${BRPC_LIB} ${DYNAMIC_LIB})
target_link_libraries(test_server ${BRPC_LIB} ${DYNAMIC_LIB})
注意事项
- 使用前需要安装brpc
- 注意命名空间
转载请注明出处