gRPC over HTTP/2 in Rust
If you're keen on enhancing your networked applications in Rust, you've come to the right place. Today, we're exploring Tonic, Rust's take on the gRPC framework. With its high-performance and reliable infrastructure, Tonic can significantly improve the efficiency of your web services.
Ready to learn more?
Let's begin with what Tonic and gRPC in Rust offer.
What is Tonic?
Tonic is a gRPC over HTTP/2 implementation focused on high performance, interoperability, and flexibility. Built on top of the hyper, tower, and prost libraries, it offers first-class support for async/await syntax, making it easy to craft high-performance gRPC servers and clients.
Key Features of Tonic:
1. Async/Await Support:
2. Interceptors:
3. Code Generation:
4. HTTP/2 Support:
5. Streaming:
6. Extensibility:
7. Robust Error Handling:
8. Efficient Serialization:
Setting Up Tonic
Before diving into examples, ensure you add the required dependencies to your Cargo.toml:
[dependencies]
tonic = "0.5"
prost = "0.8"
tokio = { version = "1", features = ["full"] }
You'd also want to include the build dependencies to generate Rust code from .proto files:
[build-dependencies]
tonic-build = "0.5"
Examples:
1. Defining the Protocol
Start by defining your service in a .proto file, for instance, hello_world.proto:
syntax = "proto3";
package hello_world;
service Greeter {
rpc SayHello (HelloRequest) returns (HelloReply);
}
message HelloRequest {
string name = 1;
}
message HelloReply {
string message = 1;
}
Run the build script to generate Rust code:
fn main() -> Result<(), Box<dyn std::error::Error>> {
tonic_build::compile_protos("proto/hello_world.proto")?;
Ok(())
}
2. Implementing the Server
Here's a simple implementation using Tonic:
use tonic::{transport::Server, Request, Response, Status};
use hello_world::greeter_server::{Greeter, GreeterServer};
use hello_world::{HelloReply, HelloRequest};
pub mod hello_world {
tonic::include_proto!("hello_world");
}
#[derive(Debug, Default)]
pub struct MyGreeter;
#[tonic::async_trait]
impl Greeter for MyGreeter {
async fn say_hello(
&self,
request: Request<HelloRequest>,
) -> Result<Response<HelloReply>, Status> {
let reply = hello_world::HelloReply {
message: format!("Hello, {}!", request.into_inner().name),
};
Ok(Response::new(reply))
}
}
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
let addr = "[::1]:50051".parse()?;
let greeter = MyGreeter::default();
Server::builder()
.add_service(GreeterServer::new(greeter))
.serve(addr)
.await?;
Ok(())
}
3. Crafting the Client
Once you have a server, a client is easy to create:
use tonic::transport::Channel;
use hello_world::greeter_client::GreeterClient;
use hello_world::HelloRequest;
pub mod hello_world {
tonic::include_proto!("hello_world");
}
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
let channel = Channel::from_static("http://[::1]:50051")
.connect()
.await?;
let mut client = GreeterClient::new(channel);
let request = tonic::Request::new(HelloRequest {
name: "Tonic".into(),
});
let response = client.say_hello(request).await?;
println!("RESPONSE={:?}", response);
Ok(())
}
Advanced Tonic Features:
Streaming
gRPC allows for server-streaming, client-streaming, and bidirectional-streaming. Tonic leverages async streams in Rust to support these features, making it ergonomic to implement complex data flows.
Recommended by LinkedIn
Server-Streaming Example:
Suppose we wish to send multiple HelloReply messages for a single client request.
service Greeter {
rpc StreamHello (HelloRequest) returns (stream HelloReply);
}
In the server:
#[tonic::async_trait]
impl Greeter for MyGreeter {
type StreamHelloStream = Pin<Box<dyn Stream<Item = Result<HelloReply, Status>> + Send + Sync + 'static>>;
async fn stream_hello(
&self,
request: Request<HelloRequest>,
) -> Result<Response<Self::StreamHelloStream>, Status> {
let name = request.into_inner().name;
let messages = vec![
format!("Hello, {}!", name),
format!("How are you, {}?", name),
format!("Goodbye, {}!", name)
];
let output = futures::stream::iter(
messages.into_iter().map(|msg| {
Ok(hello_world::HelloReply { message: msg })
})
);
Ok(Response::new(Box::pin(output)))
}
}
On the client side, you can asynchronously iterate over the streamed messages.
let mut response_stream = client.stream_hello(request).await?.into_inner();
while let Some(response) = response_stream.message().await? {
println!("Streamed Message: {:?}", response);
}
Interceptors
Interceptors allow developers to add custom logic to the request/response lifecycle. This is useful for implementing functionalities like authentication, logging, and metrics collection without altering the core service logic.
Example: Logging Interceptor
fn log_request(request: Request<()>) -> Result<Request<()>, Status> {
println!("Incoming request: {:?}", request.metadata());
Ok(request)
}
let greeter = MyGreeter::default();
let greeter_with_interceptor = GreeterServer::with_interceptor(greeter, log_request);
Performance and Production Readiness
Tonic is optimized for performance, making the most of Rust's zero-cost abstractions and async runtime. gRPC, with its HTTP/2 foundation, provides multiplexing, efficient binary data transmission, and header compression. Combined with Tonic's Rust implementation, users can expect low-latency and high-throughput communication.
For production readiness, it's crucial to:
Integration with Other Libraries
One of Tonic's strengths is its extensibility and its compatibility with other Rust libraries. Whether you're interested in telemetry, authentication, or extended serialization formats, Tonic likely has a way to integrate.
Telemetry with tracing and metrics
Tonic, built on the tower service framework, naturally supports the tracing ecosystem. This allows users to gain deep insights into application behaviour, performance bottlenecks, and trace requests end-to-end.
To integrate tracing:
tracing = "0.1"
tracing-subscriber = "0.2"
let subscriber = tracing_subscriber::fmt::Subscriber::builder()
.finish();
tracing::subscriber::set_global_default(subscriber)
.expect("Setting global default failed");
Authentication with rust-jwt or rust-oauth2
Integrating JWT or OAuth2 can be achieved smoothly with Tonic's interceptors to secure your services.
Example with JWT:
Custom Serialization with serde
While prost is the default serialization library Tonic uses, you might be in situations where custom serialization, perhaps with serde, is necessary. While this requires more manual steps, it's feasible by crafting custom codecs.
Tips for Deployment
While developing with Tonic is a pleasure, deploying gRPC services also requires some considerations:
Read more articles about Rust in my Rust Programming Library!
All right, there we have it!
Happy coding, and keep those gears turning! 🦀🔧🚀
Read more articles about Rust in my Rust Programming Library!
Visit my Blog for more articles, news, and software engineering stuff!
All the best,
CTO | Tech Lead | Senior Software Engineer | Cloud Solutions Architect | Rust 🦀 | Golang | Java | ML AI & Statistics | Web3 & Blockchain