REST协议改造gRPC实战

gRPC 是由 Google 主导开发的 RPC 框架,使用 HTTP/2 协议并用 ProtoBuf 作为序列化工具。

gRPC官方对REST的声音是:

  • 和REST一样遵循HTTP协议(明确的说是HTTP/2),但是gRPC提供了全双工流
  • 和传统的REST不同的是gRPC使用了静态路径,从而提高性能
  • 用一些格式化的错误码代替了HTTP的状态码更好的标示错误

背景

regist 是在微服务架构上再抽出的一层,用于服务注册,心跳机制,服务注销。

regist 和 微服务架构 之间使用的是 REST 协议进行通信,现需要改造成 gRPC

  • RPC 框架: gRPC

  • 协议:HTTP/2

  • 序列化工具: ProtoBuf

  • 服务端: regist (golang)

  • 客户端 :微服务架构(java)

一、服务端改造(golang)

1. protocal buffer安装

去官网下载 protoc 压缩包,解压后,将 protoc.exe放在 GOPATH\bin目录下面

2. 安装GoLang protoc 插件

gRPC-go可以通过golang 的get命令直接安装,非常方便。
go get -a github.com/golang/protobuf/protoc-gen-go

3. 定义register.proto文件
syntax = "proto3";
package proto;

service Regist {
  rpc Register (ResponseService) returns (RegisterReply) {}
  rpc Deregister (ResponseService) returns (DeregisterReply) {}
}

message RegisterReply {
  string message = 1;
}

message DeregisterReply {
  string message = 1;
}

message ResponseService {
  string id = 1;
  string name = 2;
  string version = 3;
  string address = 4;
  int32 port = 5;
  map<string, string> metadata = 6; 
}
  • syntax = "proto3",声明protobuf版本为3,默认为2。
  • 定义了一个服务 Regist,其中有两个API RegisterRegister
  • 定义3个message为接受和返回的参数。
4. 生成 register.pb.go 文件

register.proto文件所在根目录下,执行:
protoc --go_out=plugins=grpc:. register.proto

则会在同目录下生成对应的 register.pb.go 文件,相应的服务器端和客户端的GoLang代码。生成的代码中包含了客户端能够进行RPC的方法以及服务器端需要进行实现的接口。

5.服务端代码改造
  • register.pb.go 文件中提供的服务接口
type RegistClient interface {
    Register(ctx context.Context, in *ResponseService, opts ...grpc.CallOption) (*RegisterReply, error)
    Deregister(ctx context.Context, in *ResponseService, opts ...grpc.CallOption) (*DeregisterReply, error)
} 
  • 服务端代码实现上面的接口
func (s *server) Register(ctx context.Context, in *pb.ResponseService) (*pb.RegisterReply, error) {
    // 定义注册的服务,组装 gRPC 传过来的参数
    service := &rp.Service{
        Name:     in.GetName(),
        Version:  in.GetVersion(),
        Metadata: in.GetMetadata(),
        NodeName: nodeName,
        Nodes: []*rp.Node{
            &rp.Node{
                ID:      in.GetId(),
                Address: in.GetAddress(),
                Port:    int(in.GetPort()),
            },
        },
        Endpoints: []*rp.Endpoint{
            &rp.Endpoint{
                Name:     in.GetName(),
                Request:  &rp.Value{},
                Response: &rp.Value{},
                Metadata: in.GetMetadata(),
            },
        },
    }
    // 调用 consul 注册接口
    return &pb.RegisterReply{}, registry.Register(service, rp.RegisterTTL(time.Duration(ttl)*time.Second))
}

二、客户端改造(java)

1. pom.xml文件中添加 maven 依赖
<dependency>
    <groupId>io.grpc</groupId>
    <artifactId>grpc-netty</artifactId>
    <version>1.7.0</version>
</dependency>
<dependency>
    <groupId>io.grpc</groupId>
    <artifactId>grpc-protobuf</artifactId>
    <version>1.7.0</version>
</dependency>
<dependency>
    <groupId>io.grpc</groupId>
    <artifactId>grpc-stub</artifactId>
    <version>1.7.0</version>
</dependency>
2. pom.xml文件中 配置protobuf maven插件
    <build>
        <extensions>
            <extension>
                <groupId>kr.motd.maven</groupId>
                <artifactId>os-maven-plugin</artifactId>
                <version>1.5.0.Final</version>
            </extension>
        </extensions>
        <plugins>
            <plugin>
                <groupId>org.xolstice.maven.plugins</groupId>
                <artifactId>protobuf-maven-plugin</artifactId>
                <version>0.5.0</version>
                <configuration>
                    <protocArtifact>com.google.protobuf:protoc:3.4.0:exe:${os.detected.classifier}</protocArtifact>
                    <pluginId>grpc-java</pluginId>
                    <pluginArtifact>io.grpc:protoc-gen-grpc-java:1.7.0:exe:${os.detected.classifier}</pluginArtifact>
                </configuration>
                <executions>
                    <execution> 
                        <goals>
                            <goal>compile</goal>
                            <goal>compile-custom</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
3. 定义regist.proto文件
syntax = "proto3";
package proto;
 
option java_multiple_files = true;
option java_package = "com.fiberhome.smartms.grpc";
option java_outer_classname = "RegistProto";

service Regist {
  rpc Register (ResponseService) returns (RegisterReply) {}
  rpc Deregister (ResponseService) returns (DeregisterReply) {}
}

message RegisterReply {
  string message = 1;
}

message DeregisterReply {
  string message = 1;
}

message ResponseService {
  string id = 1;
  string name = 2;
  string version = 3;
  string address = 4;
  int32 port = 5;
  map<string, string> metadata = 6; 
}
4. 自动生成java接口代码

在pom.xml文件根目录中,执行mvn compile(也可试下mvn install),根据.proto文件自动生成一系列的接口文件。

- 若生成不成功,多试几次

5.客户端代码改造
public class ServiceRegistryGrpc implements ServiceRegistry<Registration> {

    private static final Logger logger = Logger.getLogger(ServiceRegistryGrpc.class.getName());

    private final ManagedChannel channel;
    private final RegistGrpc.RegistBlockingStub blockingStub;
        
    public ServiceRegistryGrpc(String host, int port) {
        channel = ManagedChannelBuilder.forAddress(host, port).usePlaintext(true).build();
        blockingStub = RegistGrpc.newBlockingStub(channel);
    }

    public void shutdown() throws InterruptedException {
        channel.shutdown().awaitTermination(5, TimeUnit.SECONDS);
    }

    @Override
    public void register(Registration reg) {
        Service service = reg.getService();
        ServiceRegistryGrpc client = new ServiceRegistryGrpc("mos", 9999);
        ResponseService request = ResponseService.newBuilder().setId(service.getId()).setName(service.getName())
                .setVersion(service.getVersion()).setAddress(service.getAddress()).setPort(service.getPort()).build();
        RegisterReply response;
        try {
            response = blockingStub.register(request);
        } catch (StatusRuntimeException e) {
            logger.log(Level.WARNING, "RPC failed: {0}", e.getStatus());
            return;
        }
        logger.info("Registering service success ");
    }
}
  • public ServiceRegistryGrpc(String host, int port) {},这个函数是构造客户端 连接 服务

  • response = blockingStub.register(request);,调用服务接口

三、验证

  1. 启动服务端


    服务端
  2. 启动客户端


    客户端
  3. 数据通信成功


    服务端获取客户端 传来的数据
  1. consul注册成功


    attachment服务
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容