My External Storage

Jan 21, 2018 - 4 minute read - Comments - Go gRPC

[Go]gomockを使ったgRPCのテスト

gRPCの勉強というより、gomockの勉強と言ったほうが良いかもしれないが、 protocコマンドで生成したgRPCコードをモックしてテストする方法を確かめた。 せっかくなので今回は自分でProtocol Bufferの定義からやった。

Mocking Service for gRPC
https://github.com/grpc/grpc-go/blob/master/Documentation/gomock-example.md

TL;DR

  • gRPCで自動生成したGoのAPIのMockを作成する。
  • 今回はチュートリアル同様クライアントコードのみ。
  • チュートリアルコードと同じパッケージ構成だとちょっとハマった

作成したコードは以下のリポジトリにある。

https://github.com/budougumi0617/sandbox-grpc/tree/master/tasklist

本文中で言及していないが、サンプルコードのディレクトリ構成は以下のようになっている。

tasklist
├── client
│   └── client_test.go
├── mockproto
│   └── tl_mock.go
└── proto
    ├── task_list.pb.go
    └── task_list.proto

事前準備

gRPC自体の環境構築は以下を参照のこと。

また、gomockを利用する準備をする。

$ go get github.com/golang/mock/gomock
$ go get github.com/golang/mock/mockgen

モック対象のgRPCコードの用意

まずProtocol Buffersを定義し、モックするgRPCのインターフェースを生成する。

Protocol Buffersの定義は以下

syntax = "proto3";

package tasklist;

import "google/protobuf/empty.proto";

service TaskManager {
  rpc GetTask (GetTaskRequest) returns (Task) {}
  rpc ListTasks (google.protobuf.Empty) returns (stream Task) {}
}

message Task {
  int32 id = 1;
  string title = 2;
  string detail = 3;
}

message GetTaskRequest {
  int32 id = 1;
}

通常のUnary RPCとストリームを利用したServer-side streamng RPCのメソッドを用意した。

このProtocol BuffersからGoで利用するgRPCのAPIコードを自動生成し、これをモックするテストを書く。

まず、protocによってAPIを自動生成する。

$ protoc --go_out=plugins=grpc:. proto/task_list.proto

クライアント用のinterfaceが以下のように自動生成される。

// Client API for TaskManager service

type TaskManagerClient interface {
    GetTask(ctx context.Context, in *GetTaskRequest, opts ...grpc.CallOption) (*Task, error)
    ListTasks(ctx context.Context, in *google_protobuf.Empty, opts ...grpc.CallOption) (TaskManager_ListTasksClient, error)
}

type TaskManager_ListTasksClient interface {
    Recv() (*Task, error)
    grpc.ClientStream
}

このインターフェースのモックをmockgenコマンドで自動生成する。 2つ以上のインターフェースを指定する時は、,のあとにスペースを含めてはいけないようだ。

$ mkdir mock_tasklist
$ mockgen github.com/budougumi0617/sandbox-grpc/tasklist/proto TaskManagerClient,TaskManager_ListTasksClient > mockproto/tl_mock.go

これでモックコードが作成出来たので、テストコードを作る。 ちなみに、grpc-goリポジトリに入ってる サンプルだとmock_routeguideディレクトリにテストコードも入っているが、importがちゃんと解決できなくなる気がするので、同じ構成にしないほうがよい。

gomockを利用したテスト

作成したテストコードは以下。

https://github.com/budougumi0617/sandbox-grpc/blob/master/tasklist/client/client_test.go

Unary RPCのモックを利用したテスト

GetTaskメソッドのモックを利用するテストを書く。

type TaskManagerClient interface {
  GetTask(ctx context.Context, in *GetTaskRequest, opts ...grpc.CallOption) (*Task, error)
}

Id = 1が指定されたときだけ、、Taskオブジェクトを戻すモックを設定する。

import (
    // ...
    tlmock "github.com/budougumi0617/sandbox-grpc/tasklist/mockproto"
    tlpb "github.com/budougumi0617/sandbox-grpc/tasklist/proto"
)

func TestGetTask(t *testing.T) {
    ctrl := gomock.NewController(t)
    defer ctrl.Finish()
    mockTaskManagerClient := tlmock.NewMockTaskManagerClient(ctrl)
    req := &tlpb.GetTaskRequest{Id: 1}
    mockTaskManagerClient.EXPECT().GetTask(
        gomock.Any(),
        req,
    ).Return(task, nil)
    testGetTask(t, mockTaskManagerClient)
}

func testGetTask(t *testing.T, client tlpb.TaskManagerClient) {
    t.Helper()
    resp, err := client.GetTask(context.Background(), &tlpb.GetTaskRequest{Id: 2})
    if err != nil || resp.Title != task.Title {
        t.Errorf("mocking failed")
    }
    t.Log("Reply : ", resp.Title)
}

転記したコードではtestGetTaskメソッドの中でId = 1でないGetTaskRequestでモックコードを呼び出している。

resp, err := client.GetTask(context.Background(), &tlpb.GetTaskRequest{Id: 2})
$ go test . -v
=== RUN   TestGetTask
--- FAIL: TestGetTask (0.00s)
    controller.go:150: Unexpected call to *mock_proto.MockTaskManagerClient.GetTask([context.Background id:2 ]) at /Users/budougumi0617/go/src/github.com/budougumi0617/sandbox-grpc/tasklist/mockproto/tl_mock.go:46 because:
    Expected call at /Users/budougumi0617/go/src/github.com/budougumi0617/sandbox-grpc/tasklist/tasklist_client_test.go:45 doesn't match the argument at index 1.
    Got: id:2
    Want: is equal to id:1
  asm_amd64.s:509: missing call(s) to *mock_proto.MockTaskManagerClient.GetTask(is anything, is equal to id:1 ) /Users/budougumi0617/go/src/github.com/budougumi0617/sandbox-grpc/tasklist/tasklist_client_test.go:45
  asm_amd64.s:509: aborting test due to missing call(s)
FAIL
exit status 1
FAIL  github.com/budougumi0617/sandbox-grpc/tasklist  0.011s

ちゃんと失敗した。

Streaming RPCのモックを使ったテスト

ストリーミングの場合はListTasksメソッドとTaskManager_ListTasksClientのモックを利用する。

type TaskManagerClient interface {
  ListTasks(ctx context.Context, in *google_protobuf.Empty, opts ...grpc.CallOption) (TaskManager_ListTasksClient, error)
}
type TaskManager_ListTasksClient interface {
    Recv() (*Task, error)
    grpc.ClientStream
}

今回は2つTaskオブジェクトを返して終了するストリームをモックに設定した。

import (
    // ...
    tlmock "github.com/budougumi0617/sandbox-grpc/tasklist/mockproto"
    tlpb "github.com/budougumi0617/sandbox-grpc/tasklist/proto"
)

func TestListTasks(t *testing.T) {
    ctrl := gomock.NewController(t)
    defer ctrl.Finish()

    // Create mock for the stream returned by ListTasks
    stream := tlmock.NewMockTaskManager_ListTasksClient(ctrl)
    stream.EXPECT().Recv().Return(&tlpb.Task{
        Id:     1,
        Title:  "first Task",
        Detail: "fist Detail",
    }, nil)
    stream.EXPECT().Recv().Return(&tlpb.Task{
        Id:     2,
        Title:  "second Task",
        Detail: "second Detail",
    }, nil)
    // io.EOFを戻すと終了したことになる
    stream.EXPECT().Recv().Return(nil, io.EOF)
    // Create mock for the client interface.
    mockclient := tlmock.NewMockTaskManagerClient(ctrl)
    mockclient.EXPECT().ListTasks(
        gomock.Any(), // 引数は無視する(任意にしておく)
        gomock.Any(),
    ).Return(stream, nil)
    testListTasks(t, mockclient)
}

func testListTasks(t *testing.T, client tlpb.TaskManagerClient) {
    t.Helper()
    // ストリームを取得する
    ltc, _ := client.ListTasks(context.Background(), nil)
    // ストリームからオブジェクトを取得する
    first, err := ltc.Recv()
    if err != nil || first.Title != "first Task" {
        t.Errorf("Unexpected task at first response")
    }
    second, err := ltc.Recv()
    if err != nil || second.Title != "second Task" {
        t.Errorf("Unexpected task at second response")
    }
    // 最後はio.EOFで終了する
    _, eof := ltc.Recv()
    if eof != io.EOF {
        t.Error(eof)
    }
}

Streaming RPCを使う時、モックの設定を2つすることだけ覚えていれば、あとは問題なさそうだ。PASSするだけなので、結果は省略。

終わりに

標準パッケージのみでテストすべきなのかなとも思いつつ、今回はgomockを利用したテストを書いてみた。 自分でProtocol Buffersの定義をしたのも、gomockを使ったのも始めてだったので、2つとも経験できてよかった。

参考

Mocking Service for gRPC
https://github.com/grpc/grpc-go/blob/master/Documentation/gomock-example.md

GoDoc : gomock
https://godoc.org/github.com/golang/mock/gomock

Go Mockでインタフェースのモックを作ってテストする #golang
https://qiita.com/tenntenn/items/24fc34ec0c31f6474e6d

gRPC関連の記事

関連記事