2 releases

0.1.1 Dec 4, 2024
0.1.0 Dec 3, 2024

#189 in Testing

36 downloads per month

Apache-2.0

31KB
612 lines

grpcmock

grpcmock is a minimal framework for mocking gRPC services in Rust, supporting unary, client-streaming, server-streaming, and bidirectional-streaming methods.

Crates.io Documentation Crates.io

Table of contents

Features

  • Mocks tonic gRPC services
  • Mocks defined in Rust or YAML files using simple, intuitive spec
  • Supports unary, client-streaming, server-streaming, and bidirectional-streaming methods
  • Performs basic "full body" (equals) matching

Stubbing

Reference service spec for examples below.

syntax = "proto3";

package example;

service Hello {
  rpc HelloUnary(HelloRequest) returns (HelloResponse) {}
  rpc HelloClientStreaming(stream HelloRequest) returns (HelloResponse) {}
}

message HelloRequest { string name = 1; }
message HelloResponse { string message = 1; }

In Rust

Mocks can be defined in Rust using prost-generated types and grpcmock abstractions.

Insert a single Mock into a MockSet:

let mut mocks = MockSet::new();
mocks.insert(
    GrpcMethod::new("example.Hello", "HelloUnary")?, 
    Mock::new(HelloRequest {}, HelloResponse {})
);

Create a MockSet with a batch of mocks:

let mocks = MockSet::with_mocks(
    [
        // Mocks for HelloUnary method
        (
            GrpcMethod::new("example.Hello", "HelloUnary")?,
            vec![Mock::new(HelloRequest {}, HelloResponse {})],
        ),
        // Mocks for HelloClientStreaming method
        (
            GrpcMethod::new("example.Hello", "HelloClientStreaming")?,
            vec![Mock::new(HelloRequest {}, HelloResponse {})],
        ),
    ]
);

Mock Files

Mocks can be defined in YAML files. A Mock File defines mocks for a single method.

Spec:

service: 'package.ServiceName' # fully-qualified gRPC service name
method: 'MethodName' # gRPC method name
mocks:
- request:
    body: '' # JSON string, [''] for streaming
  response:
    code: 200 # optional, default=200
    body: '' # JSON string, [''] for streaming
    headers: {} # optional
    error: '' # optional
  • service is the fully-qualified gRPC service name (<package>.<name>) as defined in the proto file.
    • name starts with an uppercase letter, e.g. example.Hello. method is the method name
    • Starts with an uppercase letter, e.g. HelloUnary
  • mocks is a list of mocks for the method
  • request.body / response.body is a JSON representation of the protobuf message
    • string for unary, array<string> for streaming
    • Currently, values must be set (even if empty) for all repeated and map fields
  • response.code is a HTTP status code that is converted to an equivalent gRPC status code
  • response.error is an optional error message for error responses
  • response.headers is an optional map of header key-value pairs

Examples:

  1. Client-streaming method with success response
    service: example.Hello
    method: HelloClientStreaming
    mocks:
    - request:
        body: # a list body represents a stream of messages
        - '{"name": "Dan"}' # HelloRequest
        - '{"name": "Gaurav"}'
        - '{"name": "Paul"}'
      response:
        code: 200
        body: '{"message": "Hello Dan, Gaurav, and Paul!"}' # HelloResponse
    
  2. Unary method with error response
    service: example.Hello
    method: HelloUnary
    mocks:
    - request:
        body: '{"name": "Error"}' # HelloRequest
      response:
        code: 400
        error: 'some error message'
    

Insert mocks from a file into a MockSet:

MockSet::insert_from_file<I, O>() generic type parameters correspond to prost-generated input and output types of the method defined in the mock file.

let mut mocks = MockSet::new();
mocks.insert_from_file::<HelloRequest, HelloResponse>("/path/to/file.yaml")?;

Usage

  1. Add grpcmock to Cargo.toml as a development dependency:

    [dev-dependencies]
    grpcmock = "0.1.0"
    
  2. Add required type attributes to your tonic_build configuration in build.rs. This is to enable JSON->T deserialization of prost-generated types via serde.

    tonic_build::configure()
        .type_attribute(
            ".",
            "#[derive(serde::Deserialize)] #[serde(rename_all = \"snake_case\")]",
        )
    
  3. Define stubs for your service following Stubbing guidance above.

  4. In a test context, follow the example below:

    #[cfg(test)]
    mod tests {
        use pb::{HelloRequest, HelloResponse, hello_client::HelloClient};
        use grpcmock::prelude::*;
        use tonic::transport::Channel;
    
        // Generate server `MockHelloServer` for the `example.Hello` service.
        generate_server!("example.Hello", MockHelloServer);
    
        #[tokio::test]
        async fn test_grpcmock() -> Result<(), anyhow::Error> {
            // Create a new MockSet
            let mut mocks = MockSet::new();
    
            // Load and insert mocks from mock files
            // NOTE: generic type parameters correspond to prost-generated input and output types of the method.
            mocks.insert_from_file::<HelloRequest, HelloResponse>("stubs/hello/unary.yaml")?;
            mocks.insert_from_file::<HelloRequest, HelloResponse>("stubs/hello/client_streaming.yaml")?;
    
            // Alternatively, define mocks in Rust and use MockSet::insert() or MockSet::with_mocks() to build a MockSet.
    
            // Start mock server
            let server = MockHelloServer::start(mocks).await?;
    
            // Create mock client
            let channel = Channel::from_shared(format!("http://0.0.0.0:{}", server.addr().port()))?
                .connect()
                .await?;
            let mut client = HelloClient::new(channel);
    
            // Send unary request
            let response = client
                .hello_unary(HelloRequest { name: "Dan".into() })
                .await;
            dbg!(response);
    
            // Send client-streaming request
            let request_stream = futures::stream::iter(vec![
                HelloRequest { name: "Dan".into() },
                HelloRequest { name: "Gaurav".into() },
                HelloRequest { name: "Paul".into() },
            ]);
            let response = client.hello_client_streaming(request_stream).await;
            dbg!(response);
    
            Ok(())
        }
    }
    

Examples

See grpcmock-test crate for more examples.

Dependencies

~7–15MB
~177K SLoC