質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.37%
Protocol Buffers

Protocol Buffersは、Googleが開発しているシリアライズフォーマットです。インタフェース定義言語 (IDL) で構造を定義する通信・永続化を目的としています。元々のGoogleの実装にはC++/Java/Pythonを使用。オープンソースのフリーソフトウェアです。

Go

Go(golang)は、Googleで開発されたオープンソースのプログラミング言語です。

Q&A

解決済

2回答

1689閲覧

【gRPC】gRPCサーバーを利用してAPIのうまい構成を実現したい

tanamochi

総合スコア83

Protocol Buffers

Protocol Buffersは、Googleが開発しているシリアライズフォーマットです。インタフェース定義言語 (IDL) で構造を定義する通信・永続化を目的としています。元々のGoogleの実装にはC++/Java/Pythonを使用。オープンソースのフリーソフトウェアです。

Go

Go(golang)は、Googleで開発されたオープンソースのプログラミング言語です。

0グッド

0クリップ

投稿2018/12/20 02:42

編集2018/12/21 06:09

いつもお世話になっております。
毎度、新しいことを初めては壁にぶち当たっております。
お知恵をお借りできればと思いますのでよろしくお願いします。

前提・実現したいこと

go言語でgRPCサーバーを使ってAPIを作成しているのですが、サンプルを試してみて動いたのですが複数作った場合の実装で本当にこれでいいのか?となっているのでご教授いただければと

発生している問題・試したこと

サンプルにあったprotoファイルを複製、改変し2つめのAPIを作成
1.importにて読み込み
2.関数を追加
3.サービスを起動

以下サンプル改変後のソースです

Go

1package main 2 3import ( 4 "context" 5 "log" 6 "net" 7 8 "google.golang.org/grpc" 9 pb "google.golang.org/grpc/examples/helloworld/helloworld" 10 pb2 "google.golang.org/grpc/examples/helloworld/helloworldTest" 11 "google.golang.org/grpc/reflection" 12) 13 14const ( 15 port = ":50051" 16) 17 18// server is used to implement helloworld.GreeterServer. 19type server struct{} 20 21// SayHello implements helloworld.GreeterServer 22func (s *server) SayHello(ctx context.Context, in *pb.HelloRequest) (*pb.HelloReply, error) { 23 return &pb.HelloReply{Message: "Hello " + in.Name}, nil 24} 25 26// SayHello implements helloworld.GreeterServer 27func (s *server) SayHelloTest(ctx context.Context, in *pb2.HelloRequest) (*pb2.HelloReply, error) { 28 return &pb2.HelloReply{Message: "Sum is ", Sum: in.V + in.V1}, nil 29} 30 31func main() { 32 lis, err := net.Listen("tcp", port) 33 if err != nil { 34 log.Fatalf("failed to listen: %v", err) 35 } 36 s := grpc.NewServer() 37 pb.RegisterGreeterServer(s, &server{}) 38 39 pb2.RegisterGreeterServer(s, &server{}) 40 // Register reflection service on gRPC server. 41 reflection.Register(s) 42 if err := s.Serve(lis); err != nil { 43 log.Fatalf("failed to serve: %v", err) 44 } 45}

protoファイルを作成、関数を追加するのは当然として1,2をAPIごとに行うのはどうかと思っています。
毎回、それぞれ別の名前でimportをしてサービスを立てる必要があるのでしょうか?
gRPC、Goともに初心者でして良い方法があるのかこうするしかないのか判断できていない状況です。
また、必要だったとしてももう少し人為的ミスの少なくなる記述の仕方があればご教授いただければと

何卒、よろしくお願いします。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

冒頭に宣言しているのはサーバーではなくサービスの振る舞いです。サーバーと呼べるものはひとつしかなくて、複数のサービスがぶら下がっているわけなのでなにも無駄な事はありません。
もちろんimportしておくだけでサービスをデフォルトサーバーにぶら下げるという事はできなくはありませんが、サーバーインスタンスを隠蔽してやるべき事を別のところに書くだけです。
僕は今の書き方の方が良いという考えです。
なぜならサーバーは運用のどこかでカスタマイズがしたくなるからです。タイムアウト値やロギングなどなど。
メインパッケージがいろんな依存を取りまとめる方がより明示的で追いやすくて良いと思います。

投稿2018/12/20 23:35

nobonobo

総合スコア3367

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

tanamochi

2018/12/21 06:16 編集

回答ありがとうございます! 確かにサービスですね、修正しましたm(_ _)m せっかくお詳しい方が回答してくださったのに質問のしかたが悪かったです、すみません。 import、サービス追加をapiごとに追記すると人為ミスが起きそうだと思ったのでimportをまとめる、サービス起動をで外部で配列追加してまとめてforeachで行うなどができるといいなと思った次第です。 protoファイルにある程度まとめてapiを記述できるみたいなので自己解決の形にいたしました。
guest

0

自己解決

結果的に記述量はやや増えましたが機能ごとある程度ファイルを分割することができました!
APIごとにimportするimport pb123 "../../proto/***"が増えるのかと思っていましたが
protoファイルの中に複数のAPIを作れるというのがわかったので
importが数百とか書かなくていいということがわかりこの形になりました
もっとこうしたらいい、ここが間違っているなどあればツッコミお願いしますm(_ _)m

フォルダ構成

|--cmd | |--main.go |--conf | |--config.toml |--proto | |--hoge | | |--hoge.pb.go | | |--hoge.proto | |--foo | | |--foo.pb.go | | |--foo.proto |--service | |--hoge.go | |--foo.go |--sys | |--gRPC | | |--gRPC.go

ソース
service/hoge.go

go

1package service 2 3import ( 4 "context" 5 6 pb "../proto/hoge" 7) 8 9type HogeServer struct{} 10 11func (s *HogeServer) Hoge(ctx context.Context, in *pb.HogeRequest) (*pb.HogeReply, error) { 12 return &pb.HogeReply{Message: "hoge"}, nil 13}

service/foo.go

go

1package service 2 3import ( 4 "context" 5 6 pb "../proto/foo" 7) 8 9type FooServer struct{} 10 11func (s *FooServer) Foo(ctx context.Context, in *pb.FooRequest) (*pb.FooReply, error) { 12 return &pb.FooReply{Message: "foo"}, nil 13} 14

gRPC.go

go

1package grpcservice 2 3import ( 4 "log" 5 "net" 6 7 // proto直下にまとめてimportをひとつにしたかったが 8 // 変換の際に内部で同名関数を生成しているため一階層掘ってわけた 9 pb1 "../../proto/hoge" 10 pb2 "../../proto/foo" 11 service "../../service" 12 13 "google.golang.org/grpc" 14 "google.golang.org/grpc/reflection" 15) 16 17func StartService(port string) { 18 lis, err := net.Listen("tcp", port) 19 if err != nil { 20 log.Fatalf("failed to listen: %v", err) 21 } 22 s := grpc.NewServer() 23 pb1.RegisterHogeServer(s, &service.HogeServer{}) 24 pb2.RegisterFooServer(s, &service.FooServer{}) 25 26 // Register reflection service on gRPC server. 27 reflection.Register(s) 28 if err := s.Serve(lis); err != nil { 29 log.Fatalf("failed to serve: %v", err) 30 } 31} 32

main.go

go

1package main 2 3import grpcservice "../sys/gRPC" 4 5const ( 6 port = ":50051" 7) 8 9func main() { 10 grpcservice.StartService(port) 11} 12

投稿2018/12/21 06:09

tanamochi

総合スコア83

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.37%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問