use std::{
future::Future,
pin::Pin,
task::{Context, Poll},
};
use http::Request;
use tower::Service;
use crate::app::{context::run_with_app, layer::AppService};
type GrpcRequest = Request<tonic::body::Body>;
impl<S> Service<GrpcRequest> for AppService<S>
where
S: Service<GrpcRequest> + Clone + Send + 'static,
S::Future: Send,
{
type Response = S::Response;
type Error = S::Error;
type Future = Pin<Box<dyn Future<Output = Result<Self::Response, Self::Error>> + Send>>;
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
self.inner.poll_ready(cx)
}
fn call(&mut self, req: GrpcRequest) -> Self::Future {
let inner = self.inner.clone();
let mut inner = std::mem::replace(&mut self.inner, inner);
let app = self.app.clone();
Box::pin(async move { run_with_app(app, inner.call(req)).await })
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{
app::{App, get_app, layer::AppLayer},
container::Container,
};
use tower::{Layer, ServiceExt};
#[tokio::test]
async fn test_grpc_app_layer() {
#[derive(Clone)]
struct TestService;
let mut app = App::new();
app.container.bind(TestService);
let layer = AppLayer(app.clone());
let service = layer.layer(tower::service_fn(|_req: GrpcRequest| async {
let app = get_app().expect("get_app should work within app layer");
let _service = app.get::<TestService>().unwrap();
Ok::<_, std::convert::Infallible>(http::Response::new(tonic::body::Body::empty()))
}));
let req = Request::builder()
.uri("/test.Service/Method")
.method("POST")
.body(tonic::body::Body::empty())
.unwrap();
let response: http::Response<tonic::body::Body> = service.oneshot(req).await.unwrap();
assert_eq!(response.status(), 200);
}
}