pub mod pb { tonic::include_proto!("grpc.examples.echo"); } use futures::Stream; use std::net::SocketAddr; use std::pin::Pin; use tokio::sync::mpsc; use tonic::{transport::Server, Request, Response, Status, Streaming}; use pb::{EchoRequest, EchoResponse}; type EchoResult = Result, Status>; type ResponseStream = Pin> + Send + Sync>>; #[derive(Debug)] pub struct EchoServer { addr: SocketAddr, } #[tonic::async_trait] impl pb::echo_server::Echo for EchoServer { async fn unary_echo(&self, request: Request) -> EchoResult { let message = format!("{} (from {})", request.into_inner().message, self.addr); Ok(Response::new(EchoResponse { message })) } type ServerStreamingEchoStream = ResponseStream; async fn server_streaming_echo( &self, _: Request, ) -> EchoResult { Err(Status::unimplemented("not implemented")) } async fn client_streaming_echo( &self, _: Request>, ) -> EchoResult { Err(Status::unimplemented("not implemented")) } type BidirectionalStreamingEchoStream = ResponseStream; async fn bidirectional_streaming_echo( &self, _: Request>, ) -> EchoResult { Err(Status::unimplemented("not implemented")) } } #[tokio::main] async fn main() -> Result<(), Box> { let addrs = ["[::1]:50051", "[::1]:50052"]; let (tx, mut rx) = mpsc::unbounded_channel(); for addr in &addrs { let addr = addr.parse()?; let tx = tx.clone(); let server = EchoServer { addr }; let serve = Server::builder() .add_service(pb::echo_server::EchoServer::new(server)) .serve(addr); tokio::spawn(async move { if let Err(e) = serve.await { eprintln!("Error = {:?}", e); } tx.send(()).unwrap(); }); } rx.recv().await; Ok(()) }