由於以前的測試結果讓筆者對PHP使用GRPC很失望,若是使用HTTP的方式Guzzle還能夠提供併發能力來下降接口消耗,Grpc只能阻塞,可是若是使用HTTP的話對於調用限制和GO調用GO來講不太友好,有沒有一個一箭雙鵰的方式可使用Grpc的前提下又兼容HTTP調用,讓客戶端能夠更具自身狀況自由選擇,服務端工做只須要作一份呢?還別說真還有一個準備好的輪子那就是今天的主角《grpc-gateway》。git
附上:github
喵了個咪的博客:w-blog.cngolang
博文實例demo:https://github.com/sunmi-OS/grpc-gateway-demovim
grpc-gateway官網:https://github.com/grpc-ecosystem/grpc-gatewayapi
grpc-gateway是protoc的一個插件 。它讀取Grpc服務定義,並生成反向代理服務器,將RESTful JSON API請求轉換爲Grpc的方式調用。主要是根據 google.api.http定義中思想完成的,一下就是grpc-gateway結構圖:  服務器
grpc-gateway使用徹底的Go語言進行開發,因此安裝起來也很是簡單,首先須要獲取相關的依賴包併發
PS:須要先準備好準備好protoc的環境curl
go get -u github.com/grpc-ecosystem/grpc-gateway/protoc-gen-grpc-gateway go get -u github.com/grpc-ecosystem/grpc-gateway/protoc-gen-swagger go get -u github.com/golang/protobuf/protoc-gen-go
cd $GOPATH/src/ mkdir -p grpc-gateway-demo/gateway cd grpc-gateway-demo/gateway vim gateway.proto syntax = "proto3"; package gateway; # 新增如下引入 import "google/api/annotations.proto"; message StringMessage { string value = 1; } # 修改方法增長http定義 # service Gateway { # rpc SayHello Echo(StringMessage) returns (StringMessage) {} # } service Gateway { rpc Echo(StringMessage) returns (StringMessage) { option (google.api.http) = { post: "/v1/example/echo" body: "*" }; } }
生成grpc結構文件和gateway文件:tcp
protoc --proto_path=../ -I/usr/local/include -I. -I$GOPATH/src -I$GOPATH/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis --go_out=plugins=grpc:. gateway.proto
protoc --proto_path=../ -I/usr/local/include -I. -I$GOPATH/src -I$GOPATH/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis --grpc-gateway_out=logtostderr=true:. gateway.proto
最終能夠看到如下文件工具
服務端代碼:
cd .. vim grpc_service.go package main import ( "log" "net" pb "grpc-gateway-demo/gateway" "google.golang.org/grpc" "golang.org/x/net/context" ) const ( PORT = ":9192" ) type server struct {} func (s *server) Echo(ctx context.Context, in *pb.StringMessage) (*pb.StringMessage, error) { log.Println("request: ", in.Value) return &pb.StringMessage{Value: "Hello " + in.Value}, nil } func main() { lis, err := net.Listen("tcp", PORT) if err != nil { log.Fatalf("failed to listen: %v", err) } s := grpc.NewServer() pb.RegisterGatewayServer(s, &server{}) log.Println("rpc服務已經開啓") s.Serve(lis) }
運行grpc服務端:
go build grpc_service.go ./grpc_service
編寫gateway服務
vim grpc_gateway.go package main import ( "flag" "net/http" "log" "github.com/golang/glog" "golang.org/x/net/context" "github.com/grpc-ecosystem/grpc-gateway/runtime" "google.golang.org/grpc" gw "grpc-gateway-demo/gateway" ) var ( echoEndpoint = flag.String("echo_endpoint", "localhost:9192", "endpoint of Gateway") ) func run() error { ctx := context.Background() ctx, cancel := context.WithCancel(ctx) defer cancel() mux := runtime.NewServeMux() opts := []grpc.DialOption{grpc.WithInsecure()} err := gw.RegisterGatewayHandlerFromEndpoint(ctx, mux, *echoEndpoint, opts) if err != nil { return err } log.Println("服務開啓") return http.ListenAndServe(":8080", mux) } func main() { flag.Parse() defer glog.Flush() if err := run(); err != nil { glog.Fatal(err) } }
運行網關程序
go build grpc_gateway.go ./grpc_gateway
使用http的方式調用網關:
curl -X POST -k http://localhost:8080/v1/example/echo -d '{"value":" world"}' {"value":"Hello world"}
cd gateway protoc -I/usr/local/include -I. \ -I$GOPATH/src \ -I$GOPATH/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis \ --swagger_out=logtostderr=true:. \ gateway.proto
對比如下兩項:
全程使用ab 帶 -k進行壓測
在GO的場景下基本上4倍差距,可是考慮到自己Go在grpc和http上自己就有3.5倍的差距,自己在同等HTTP的狀況下通過grpc-gateway和不通過直接到API差距大概在20~30%左右,這樣的性能消耗帶來的是兼容HTTP而且還能夠自動生成swagger(還能夠做爲調試工具),何樂而不爲呢?