<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
正當有這個需求的時候,就看到了這個實現姿勢。源自coreos的一篇部落格,轉載到了grpc官方部落格gRPC with REST and Open APIs。
etcd3改用grpc後為了相容原來的api,同時要提供http/json方式的API,為了滿足這個需求,要麼開發兩套API,要麼實現一種轉換機制,他們選擇了後者,而我們選擇跟隨他們的腳步。
他們實現了一個協定轉換的閘道器,對應github上的專案grpc-gateway,這個閘道器負責接收使用者端請求,然後決定直接轉發給grpc服務還是轉給http服務,當然,http服務也需要請求grpc服務獲取響應,然後轉為json響應給使用者端。結構如圖:
下面我們就直接實戰吧。基於hello-tls專案擴充套件,使用者端改動不大,伺服器端和proto改動較大。
go get -u github.com/grpc-ecosystem/grpc-gateway/protoc-gen-grpc-gateway
$GOPATH/src/grpc-go-practice/ example/ |—— hello-http-2/ |—— client/ |—— main.go // 使用者端 |—— server/ |—— main.go // 伺服器端 |—— keys/ // 證書目錄 |—— server.key |—— server.pem |—— proto/ |—— google // googleApi http-proto定義 |—— api |—— annotations.proto |—— annotations.pb.go |—— http.proto |—— http.pb.go |—— hello_http.proto // proto描述檔案 |—— hello_http.pb.go // proto編譯後檔案 |—— hello_http_pb.gw.go // gateway編譯後檔案
這裡用到了google官方Api中的兩個proto描述檔案,直接拷貝不要做修改,裡面定義了protocol buffer擴充套件的HTTP option,為grpc的http轉換提供支援。
syntax = "proto3"; // 指定proto版本 package proto; // 指定包名 import "google/api/annotations.proto"; // 定義Hello服務 service HelloHttp { // 定義SayHello方法 rpc SayHello(HelloHttpRequest) returns (HelloHttpReply) { // http option option (google.api.http) = { post: "/example/echo" body: "*" }; } } // HelloRequest 請求結構 message HelloHttpRequest { string name = 1; } // HelloReply 響應結構 message HelloHttpReply { string message = 1; }
這裡在原來的SayHello
方法定義中增加了http option, POST方式,路由為"/example/echo"。
cd $GOPATH/src/grpc-go-practice/example/hello-http-2/proto # 編譯google.api protoc -I . --go_out=plugins=grpc,Mgoogle/protobuf/descriptor.proto=github.com/golang/protobuf/protoc-gen-go/descriptor:. google/api/*.proto # 編譯hello_http.proto protoc -I . --go_out=plugins=grpc,Mgoogle/api/annotations.proto=git.vodjk.com/go-grpc/example/proto/google/api:. ./*.proto # 編譯hello_http.proto gateway protoc --grpc-gateway_out=logtostderr=true:. ./hello_http.proto
注意這裡需要編譯google/api中的兩個proto檔案,同時在編譯hello_http.proto時指定引入包名,最後使用grpc-gateway編譯生成hello_http_pb.gw.go
檔案,這個檔案就是用來做協定轉換的,檢視檔案可以看到裡面生成的http handler,處理上面定義的路由"example/echo"接收POST引數,呼叫HelloHTTP服務的使用者端請求grpc服務並響應結果。
package main import ( "crypto/tls" "fmt" "io/ioutil" "log" "net" "net/http" "strings" "github.com/grpc-ecosystem/grpc-gateway/runtime" "golang.org/x/net/context" "google.golang.org/grpc" pb "git.vodjk.com/go-grpc/example/proto" "google.golang.org/grpc/credentials" "google.golang.org/grpc/grpclog" ) // 定義helloHttpService並實現約定的介面 type helloHttpService struct{} // HelloHttpService ... var HelloHttpService = helloHttpService{} func (h helloHttpService) SayHello(ctx context.Context, in *pb.HelloHttpRequest) (*pb.HelloHttpReply, error) { resp := new(pb.HelloHttpReply) resp.Message = "Hello " + in.Name + "." return resp, nil } // grpcHandlerFunc 檢查請求協定並返回http handler func grpcHandlerFunc(grpcServer *grpc.Server, otherHandler http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { // TODO(tamird): point to merged gRPC code rather than a PR. // This is a partial recreation of gRPC's internal checks https://github.com/grpc/grpc-go/pull/514/files#diff-95e9a25b738459a2d3030e1e6fa2a718R61 if r.ProtoMajor == 2 && strings.Contains(r.Header.Get("Content-Type"), "application/grpc") { grpcServer.ServeHTTP(w, r) } else { otherHandler.ServeHTTP(w, r) } }) } func main() { endpoint := "127.0.0.1:50052" // 範例化標準grpc server creds, err := credentials.NewServerTLSFromFile("../../keys/server.pem", "../../keys/server.key") if err != nil { grpclog.Fatalf("Failed to generate credentials %v", err) } conn, _ := net.Listen("tcp", endpoint) grpcServer := grpc.NewServer(grpc.Creds(creds)) pb.RegisterHelloHttpServer(grpcServer, HelloHttpService) // http-grpc gateway ctx := context.Background() ctx, cancel := context.WithCancel(ctx) defer cancel() dcreds, err := credentials.NewClientTLSFromFile("../../keys/server.pem", "server name") if err != nil { grpclog.Fatalf("Failed to create TLS credentials %v", err) } dopts := []grpc.DialOption{grpc.WithTransportCredentials(dcreds)} gwmux := runtime.NewServeMux() err = pb.RegisterHelloHttpHandlerFromEndpoint(ctx, gwmux, endpoint, dopts) if err != nil { fmt.Printf("serve: %vn", err) return } mux := http.NewServeMux() mux.Handle("/", gwmux) if err != nil { panic(err) } // 開啟HTTP服務 cert, _ := ioutil.ReadFile("../../keys/server.pem") key, _ := ioutil.ReadFile("../../keys/server.key") var demoKeyPair *tls.Certificate pair, err := tls.X509KeyPair(cert, key) if err != nil { panic(err) } demoKeyPair = &pair srv := &http.Server{ Addr: endpoint, Handler: grpcHandlerFunc(grpcServer, mux), TLSConfig: &tls.Config{ Certificates: []tls.Certificate{*demoKeyPair}, }, } fmt.Printf("grpc and https on port: %dn", 50052) err = srv.Serve(tls.NewListener(conn, srv.TLSConfig)) if err != nil { log.Fatal("ListenAndServe: ", err) } return }
好吧,這麼大一坨。核心就是開啟了一個http server,收到請求後檢查請求是grpc還是http,然後決定是由grpc服務直接處理還是交給gateway做轉發處理。其中grpcHandlerFunc
函數負責處理決定用哪個handler處理請求,這個方法是直接Copy過來用的,原文的註釋說他們也是從別處Copy的。感謝貢獻者。
基本流程:
注意:必須開啟HTTPS
開啟服務:
# hello-http-2/server go run main.go > grpc and https on port: 50052
呼叫grpc使用者端:
# hello-http-2/client go run main.go > Hello gRPC.
請求https:
curl -X POST -k https://localhost:50052/example/echo -d '{"name": "gRPC-HTTP is working!"}' > {"message":"Hello gRPC-HTTP is working!."}
為什麼是hello-http-2,因為1是個不完整的實現姿勢,可以不用https,但是需要分別開啟grpc服務和http服務,這裡不做說明了。
本系列範例程式碼 go-grpc-tutorial
以上就是Golang gRPC HTTP協定轉換範例的詳細內容,更多關於Golang gRPC HTTP協定轉換的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45