go Grpc 入门

前言

网上很多文章不是新版的,有问题,自己尝试了一把,在此记录。

grpc安装:

//protobuf的安装
 PB_REL="https://github.com/protocolbuffers/protobuf/releases" && curl -LO $PB_REL/download/v3.11.4/protoc-3.11.4-linux-x86_64.zip

//protoc-gen-go proto生成go的代码工具
 go get github.com/golang/protobuf/protoc-gen-go

//注意设置环境变量即可

参考文档:https://www.grpc.io/docs/quickstart/go/

proto书写:

syntax = "proto3"; //版本
option go_package = ".;hello"; //生成包名hello
service Greeter {  //定义service
    rpc SayHello(HelloRequest)returns(HelloReply){}
    rpc SayHelloAgain(HelloRequest) returns(HelloReply){}
}

message HelloRequest{ //定义请求
    string name = 1;
}

message HelloReply{ //定义返回
    string message = 1;
}

参考文档:
https://developers.google.com/protocol-buffers

proto文件生成go代码

//gprc使用
 protoc --proto_path=proto --go_out=plugins=grpc:hello --go_opt=paths=source_relative proto/hello.proto

//仅包含proto生成为go代码
 protoc --proto_path=proto --go_out=hello --go_opt=paths=source_relative proto/hello.proto

参考文档:https://www.grpc.io/docs/quickstart/go/#whats-next

代码书写

服务端:

package main

import (
    "context"
    "fmt"
    "google.golang.org/grpc"
    "grpc_demo/hello"
    "net"
)

type Server struct {
}

func main()  {
    g := grpc.NewServer()
    s := Server{}
    hello.RegisterGreeterServer(g,&s)
    lis, err := net.Listen("tcp", fmt.Sprintf(":8080"))
    if err != nil {
        panic("failed to listen: "+err.Error())
    }
    g.Serve(lis)

}

func (s *Server)  SayHello(ctx context.Context,request *hello.HelloRequest)(*hello.HelloReply,error){
    return &hello.HelloReply{Message:"Hello "+request.Name},nil
}

func (s *Server)  SayHelloAgain(ctx context.Context,request *hello.HelloRequest)(*hello.HelloReply,error){
    return &hello.HelloReply{Message:"Hello Again "+request.Name},nil
}

客户端:

package main

import (
    "context"
    "fmt"
    "google.golang.org/grpc"
    "grpc_demo/hello"
)

func main()  {
    conn,err := grpc.Dial("127.0.0.1:8080",grpc.WithInsecure())
    if err!=nil{
        panic(err)
    }
    defer conn.Close()
    c := hello.NewGreeterClient(conn)
    r,err := c.SayHello(context.Background(),&hello.HelloRequest{Name:"ucan"})
    if err!=nil{
        panic(err)
    }
    fmt.Println(r.Message)
}

demo地址:http://gitee.com/ucanme/grpc_demo

带证书版本代码
util.go

package util

import (
    "crypto/tls"
    "golang.org/x/net/http2"
    "io/ioutil"
    "log"
)

func GetTLSConfig(certPemPath, certKeyPath string) *tls.Config {
    var certKeyPair *tls.Certificate
    cert, _ := ioutil.ReadFile(certPemPath)
    key, _ := ioutil.ReadFile(certKeyPath)

    pair, err := tls.X509KeyPair(cert, key)
    if err != nil {
        log.Println("TLS KeyPair err: %v\n", err)
    }

    certKeyPair = &pair

    return &tls.Config{
        Certificates: []tls.Certificate{*certKeyPair},
        NextProtos:   []string{http2.NextProtoTLS},
    }
}

server.go

package main

import (
    "context"
    "crypto/tls"
    "fmt"
    "google.golang.org/grpc"
    "grpc_demo/hello"
    "grpc_demo/util"
    "net"
)

type Server struct {
}

func main() {
    g := grpc.NewServer()
    s := Server{}
    hello.RegisterGreeterServer(g, &s)
    conn, err := net.Listen("tcp", fmt.Sprintf(":8080"))
    if err != nil {
        panic("failed to listen: " + err.Error())
    }
    //tls
    g.Serve(tls.NewListener(conn, util.GetTLSConfig("./cert/server.pem", "./cert/server.key")))

}

func (s *Server) SayHello(ctx context.Context, request *hello.HelloRequest) (*hello.HelloReply, error) {
    return &hello.HelloReply{Message: "Hello " + request.Name}, nil
}

func (s *Server) SayHelloAgain(ctx context.Context, request *hello.HelloRequest) (*hello.HelloReply, error) {
    return &hello.HelloReply{Message: "Hello Again " + request.Name}, nil
}

client.go

/**
 * @author: wuji
 * @file:  main
 * @desc:
 * @date: 2020/4/23 7:31 下午
 */

package main

import (
    "context"
    "fmt"
    "google.golang.org/grpc"
    "google.golang.org/grpc/credentials"
    "grpc_demo/hello"
)

func main()  {
    //证书
    creds,err := credentials.NewClientTLSFromFile("../cert/server.pem","ucan")
    if err!=nil{
        panic(err)
    }

    conn,err := grpc.Dial("127.0.0.1:8080",grpc.WithTransportCredentials(creds))
    if err!=nil{
        panic(err)
    }
    defer conn.Close()
    c := hello.NewGreeterClient(conn)


    r,err := c.SayHello(context.Background(),&hello.HelloRequest{Name:"ucan"})
    if err!=nil{
        panic(err)
    }
    fmt.Println(r.Message)
}

proto编译参考

编译器使用
使用protoc命令编译.proto文件,不同语言支持需要指定输出参数,如:

protoc --proto_path=IMPORT_PATH --cpp_out=DST_DIR --java_out=DST_DIR --python_out=DST_DIR --go_out=DST_DIR --ruby_out=DST_DIR --javanano_out=DST_DIR --objc_out=DST_DIR --csharp_out=DST_DIR path/to/file.proto
这里详细介绍golang的编译姿势:

-I 参数:指定import路径,可以指定多个-I参数,编译时按顺序查找,不指定时默认查找当前目录

--go_out :golang编译支持,支持以下参数

plugins=plugin1+plugin2 - 指定插件,目前只支持grpc,即:plugins=grpc

M 参数 - 指定导入的.proto文件路径编译后对应的golang包名(不指定本参数默认就是.proto文件中import语句的路径)

import_prefix=xxx - 为所有import路径添加前缀,主要用于编译子目录内的多个proto文件,这个参数按理说很有用,尤其适用替代一些情况时的M参数,但是实际使用时有个蛋疼的问题导致并不能达到我们预想的效果,自己尝试看看吧

import_path=foo/bar - 用于指定未声明package或go_package的文件的包名,最右面的斜线前的字符会被忽略

末尾 :编译文件路径 .proto文件路径(支持通配符)

完整示例:

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

推荐阅读更多精彩内容

  • date: 2019-04-25 22:16:01title: tech| 再探 grpc 折腾 grpc 过几次...
    daydaygo阅读 13,577评论 2 17
  • 写在前面 首先声明这篇文章只是介绍gRPC在Go语言中的使用入门级的文章,不包含多少深入的内容,读者对象是g...
    foundwei阅读 2,618评论 0 5
  • 该篇文章介绍了golang的grpc编程。 通过下面的例子,你将会学到:1. 在一个.proto文件里define...
    晓_7611阅读 1,382评论 0 3
  • 1,关于grpc-go golang 可以可以做grpc的服务端和客户端。 官网的文档: http://www.g...
    ebayboy阅读 1,077评论 0 1
  • 原文出处:gRPC gRPC分享 概述 gRPC 一开始由 google 开发,是一款语言中立、平台中立、开源的远...
    小波同学阅读 7,196评论 0 18