BRPC使用示例

BRPC使用示例

BRPC: https://github.com/apache/incubator-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
  • 注意命名空间
posted @ 2022-09-14 11:11  荒唐了年少  阅读(986)  评论(0编辑  收藏  举报