GripMock
GripMock is a mock server for GRPC services. It's using .proto
file to generate implementation of gRPC service for you.
If you already familiar with Apiary or WireMock for mocking API service and looking for similiar thing for GRPC then this is the perfect fit for that.
How It Works
GripMock has 2 main components:
- GRPC server that serving on
tcp://localhost:4770
. It's main job is to serve incoming rpc call from client then parse the input so that can be posted to Stub service to find the perfect stub match. - Stub server that serving on
http://localhost:4771
. It's main job is to store all the stub mapping. We can add a new stub or list existing stub using http request.
Matched stub will be returned to GRPC service then further parse it to response the rpc call.
Quick Usage
First, prepare your .proto
file. or you can use hello.proto
in example/pb/
folder. Suppose you put it in /mypath/hello.proto
. We gonna use Docker image for easier example test.
basic syntax to run GripMock is
gripmock <protofile>
- Install Docker
- Run
docker pull tkpd/gripmock
to pull the image - We gonna mount
/mypath/hello.proto
(it must a fullpath) into container and also we expose ports needed. Rundocker run -p 4770:4770 -p 4771:4771 -v /mypath:/proto tkpd/gripmock /proto/hello.proto
- On separate terminal we gonna add stub into stub service. Run
curl -X POST -d '{"service":"Greeter","method":"SayHello","input":{"equals":{"name":"gripmock"}},"output":{"data":{"message":"Hello GripMock"}}}' localhost:4771/add
- Now we are ready to test it with our client. you can find client example file under
example/client/
. Execute one of your preferred language. Example for go:go run example/client/go/*.go
Check example
folder for various usecase of gripmock.
Stubbing
Stubbing is the essential mocking of GripMock. It will match and return the expected result into GRPC service. This is where you put all your request expectation and response
Dynamic stubbing
You could add stubbing on the fly with simple REST. HTTP stub server running on port :4771
GET /
Will list all stubs mapping.POST /add
Will add stub with provided stub dataPOST /find
Find matching stub with provided input. see Input Matching below.GET /clear
Clear stub mappings.
Stub Format is JSON text format. It has skeleton like below:
{
"service":"<servicename>", // name of service defined in proto
"method":"<methodname>", // name of method that we want to mock
"input":{ // input matching rule. see Input Matching Rule section below
// put rule here
},
"output":{ // output json if input were matched
"data":{
// put result fields here
},
"error":"<error message>" // Optional. if you want to return error instead.
}
}
For our hello
service example we put stub with below text:
{
"service":"Greeter",
"method":"SayHello",
"input":{
"equals":{
"name":"gripmock"
}
},
"output":{
"data":{
"message":"Hello GripMock"
}
}
}
Static stubbing
You could initialize gripmock with stub json files and provide the path using --stub
argument. For example you may
mount your stub file in /mystubs
folder then mount it to docker like
docker run -p 4770:4770 -p 4771:4771 -v /mypath:/proto -v /mystubs:/stub tkpd/gripmock --stub=/stub /proto/hello.proto
Please note that Gripmock still serve http stubbing to modify stored stubs on the fly.
Input Matching
Stub will responding the expected response if only requested with matching rule of input. Stub service will serve /find
endpoint with format:
{
"service":"<service name>",
"method":"<method name>",
"data":{
// input that suppose to match with stored stubs
}
}
So if you do curl -X POST -d '{"service":"Greeter","method":"SayHello","data":{"name":"gripmock"}}' localhost:4771/find
stub service will find a match from listed stubs stored there.
Input Matching Rule
Input matching has 3 rules to match an input. which is equals,contains and regex
equals will match the exact field name and value of input into expected stub. example stub JSON:
{
.
.
"input":{
"equals":{
"name":"gripmock"
}
}
.
.
}
contains will match input that has the value declared expected fields. example stub JSON:
{
.
.
"input":{
"contains":{
"field2":"hello"
}
}
.
.
}
matches using regex for matching fields expectation. example:
{
.
.
"input":{
"matches":{
"name":"^grip.*$"
}
}
.
.
}