Skip to content

Commit

Permalink
add error handing
Browse files Browse the repository at this point in the history
  • Loading branch information
onewe committed May 11, 2024
1 parent 4fea4e4 commit aff4c63
Show file tree
Hide file tree
Showing 2 changed files with 166 additions and 43 deletions.
82 changes: 76 additions & 6 deletions dubbo/src/extension/invoker_extension.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ use async_trait::async_trait;
use bytes::Bytes;
use futures_core::Stream;
use std::{collections::HashMap, future::Future, marker::PhantomData, pin::Pin};
use thiserror::Error;

#[async_trait]
pub trait Invoker {
Expand Down Expand Up @@ -78,7 +79,9 @@ pub mod proxy {
use bytes::Bytes;
use futures_core::Stream;
use std::pin::Pin;
use thiserror::Error;
use tokio::sync::{mpsc::Sender, oneshot};
use tracing::error;

pub(super) enum InvokerOpt {
Invoke(
Expand All @@ -100,14 +103,36 @@ pub mod proxy {
invocation: GrpcInvocation,
) -> Result<Pin<Box<dyn Stream<Item = Bytes> + Send + 'static>>, StdError> {
let (tx, rx) = oneshot::channel();
let _ = self.tx.send(InvokerOpt::Invoke(invocation, tx));
let ret = self.tx.send(InvokerOpt::Invoke(invocation, tx)).await;
match ret {
Ok(_) => {}
Err(err) => {
error!(
"call invoke method failed by invoker proxy, error: {:?}",
err
);
return Err(InvokerProxyError::new(
"call invoke method failed by invoker proxy",
)
.into());
}
}
let ret = rx.await?;
ret
}

async fn url(&self) -> Result<Url, StdError> {
let (tx, rx) = oneshot::channel();
let _ = self.tx.send(InvokerOpt::Url(tx));
let ret = self.tx.send(InvokerOpt::Url(tx)).await;
match ret {
Ok(_) => {}
Err(err) => {
error!("call url method failed by invoker proxy, error: {:?}", err);
return Err(
InvokerProxyError::new("call url method failed by invoker proxy").into(),
);
}
}
let ret = rx.await?;
ret
}
Expand All @@ -121,17 +146,39 @@ pub mod proxy {
match opt {
InvokerOpt::Invoke(invocation, tx) => {
let result = invoker.invoke(invocation).await;
let _ = tx.send(result);
let callback_ret = tx.send(result);
match callback_ret {
Ok(_) => {}
Err(err) => {
error!("invoke method has been called, but callback to caller failed. {:?}", err);
}
}
}
InvokerOpt::Url(tx) => {
let _ = tx.send(invoker.url().await);
let ret = tx.send(invoker.url().await);
match ret {
Ok(_) => {}
Err(err) => {
error!("url method has been called, but callback to caller failed. {:?}", err);
}
}
}
}
}
});
InvokerProxy { tx }
}
}

#[derive(Error, Debug)]
#[error("invoker proxy error: {0}")]
pub struct InvokerProxyError(String);

impl InvokerProxyError {
pub fn new(msg: &str) -> Self {
InvokerProxyError(msg.to_string())
}
}
}

#[derive(Default)]
Expand All @@ -149,9 +196,22 @@ impl InvokerExtensionLoader {
}

pub fn load(&mut self, url: Url) -> Result<LoadExtensionPromise<InvokerProxy>, StdError> {
let extension_name = url.query::<ExtensionName>().unwrap();
let extension_name = url.query::<ExtensionName>();
let Some(extension_name) = extension_name else {
return Err(InvokerExtensionLoaderError::new(
"load invoker extension failed, extension mustn't be empty",
)
.into());
};
let extension_name = extension_name.value();
let factory = self.factories.get_mut(&extension_name).unwrap();
let factory = self.factories.get_mut(&extension_name);
let Some(factory) = factory else {
let err_msg = format!(
"load {} invoker extension failed, can not found extension factory",
extension_name
);
return Err(InvokerExtensionLoaderError(err_msg).into());
};
factory.create(url)
}
}
Expand Down Expand Up @@ -230,3 +290,13 @@ where
))
}
}

#[derive(Error, Debug)]
#[error("{0}")]
pub struct InvokerExtensionLoaderError(String);

impl InvokerExtensionLoaderError {
pub fn new(msg: &str) -> Self {
InvokerExtensionLoaderError(msg.to_string())
}
}
127 changes: 90 additions & 37 deletions examples/echo/src/generated/grpc.examples.echo.rs
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,9 @@ pub mod echo_client {
let invocation = RpcInvocation::default()
.with_service_unique_name(String::from("grpc.examples.echo.Echo"))
.with_method_name(String::from("UnaryEcho"));
let path = http::uri::PathAndQuery::from_static("/grpc.examples.echo.Echo/UnaryEcho");
let path = http::uri::PathAndQuery::from_static(
"/grpc.examples.echo.Echo/UnaryEcho",
);
self.inner.unary(request, path, invocation).await
}
/// ServerStreamingEcho is server side streaming.
Expand Down Expand Up @@ -100,7 +102,9 @@ pub mod echo_server {
request: Request<super::EchoRequest>,
) -> Result<Response<super::EchoResponse>, dubbo::status::Status>;
///Server streaming response type for the ServerStreamingEcho method.
type ServerStreamingEchoStream: futures_util::Stream<Item = Result<super::EchoResponse, dubbo::status::Status>>
type ServerStreamingEchoStream: futures_util::Stream<
Item = Result<super::EchoResponse, dubbo::status::Status>,
>
+ Send
+ 'static;
/// ServerStreamingEcho is server side streaming.
Expand All @@ -114,14 +118,19 @@ pub mod echo_server {
request: Request<Decoding<super::EchoRequest>>,
) -> Result<Response<super::EchoResponse>, dubbo::status::Status>;
///Server streaming response type for the BidirectionalStreamingEcho method.
type BidirectionalStreamingEchoStream: futures_util::Stream<Item = Result<super::EchoResponse, dubbo::status::Status>>
type BidirectionalStreamingEchoStream: futures_util::Stream<
Item = Result<super::EchoResponse, dubbo::status::Status>,
>
+ Send
+ 'static;
/// BidirectionalStreamingEcho is bidi streaming.
async fn bidirectional_streaming_echo(
&self,
request: Request<Decoding<super::EchoRequest>>,
) -> Result<Response<Self::BidirectionalStreamingEchoStream>, dubbo::status::Status>;
) -> Result<
Response<Self::BidirectionalStreamingEchoStream>,
dubbo::status::Status,
>;
}
/// Echo is the echo service.
#[derive(Debug)]
Expand Down Expand Up @@ -151,7 +160,10 @@ pub mod echo_server {
type Response = http::Response<BoxBody>;
type Error = std::convert::Infallible;
type Future = BoxFuture<Self::Response, Self::Error>;
fn poll_ready(&mut self, _cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
fn poll_ready(
&mut self,
_cx: &mut Context<'_>,
) -> Poll<Result<(), Self::Error>> {
Poll::Ready(Ok(()))
}
fn call(&mut self, req: http::Request<B>) -> Self::Future {
Expand All @@ -164,16 +176,24 @@ pub mod echo_server {
}
impl<T: Echo> UnarySvc<super::EchoRequest> for UnaryEchoServer<T> {
type Response = super::EchoResponse;
type Future = BoxFuture<Response<Self::Response>, dubbo::status::Status>;
fn call(&mut self, request: Request<super::EchoRequest>) -> Self::Future {
type Future = BoxFuture<
Response<Self::Response>,
dubbo::status::Status,
>;
fn call(
&mut self,
request: Request<super::EchoRequest>,
) -> Self::Future {
let inner = self.inner.0.clone();
let fut = async move { inner.unary_echo(request).await };
Box::pin(fut)
}
}
let fut = async move {
let mut server =
TripleServer::<super::EchoRequest, super::EchoResponse>::new();
let mut server = TripleServer::<
super::EchoRequest,
super::EchoResponse,
>::new();
let res = server.unary(UnaryEchoServer { inner }, req).await;
Ok(res)
};
Expand All @@ -184,20 +204,30 @@ pub mod echo_server {
struct ServerStreamingEchoServer<T: Echo> {
inner: _Inner<T>,
}
impl<T: Echo> ServerStreamingSvc<super::EchoRequest> for ServerStreamingEchoServer<T> {
impl<T: Echo> ServerStreamingSvc<super::EchoRequest>
for ServerStreamingEchoServer<T> {
type Response = super::EchoResponse;
type ResponseStream = T::ServerStreamingEchoStream;
type Future =
BoxFuture<Response<Self::ResponseStream>, dubbo::status::Status>;
fn call(&mut self, request: Request<super::EchoRequest>) -> Self::Future {
type Future = BoxFuture<
Response<Self::ResponseStream>,
dubbo::status::Status,
>;
fn call(
&mut self,
request: Request<super::EchoRequest>,
) -> Self::Future {
let inner = self.inner.0.clone();
let fut = async move { inner.server_streaming_echo(request).await };
let fut = async move {
inner.server_streaming_echo(request).await
};
Box::pin(fut)
}
}
let fut = async move {
let mut server =
TripleServer::<super::EchoRequest, super::EchoResponse>::new();
let mut server = TripleServer::<
super::EchoRequest,
super::EchoResponse,
>::new();
let res = server
.server_streaming(ServerStreamingEchoServer { inner }, req)
.await;
Expand All @@ -210,21 +240,29 @@ pub mod echo_server {
struct ClientStreamingEchoServer<T: Echo> {
inner: _Inner<T>,
}
impl<T: Echo> ClientStreamingSvc<super::EchoRequest> for ClientStreamingEchoServer<T> {
impl<T: Echo> ClientStreamingSvc<super::EchoRequest>
for ClientStreamingEchoServer<T> {
type Response = super::EchoResponse;
type Future = BoxFuture<Response<Self::Response>, dubbo::status::Status>;
type Future = BoxFuture<
Response<Self::Response>,
dubbo::status::Status,
>;
fn call(
&mut self,
request: Request<Decoding<super::EchoRequest>>,
) -> Self::Future {
let inner = self.inner.0.clone();
let fut = async move { inner.client_streaming_echo(request).await };
let fut = async move {
inner.client_streaming_echo(request).await
};
Box::pin(fut)
}
}
let fut = async move {
let mut server =
TripleServer::<super::EchoRequest, super::EchoResponse>::new();
let mut server = TripleServer::<
super::EchoRequest,
super::EchoResponse,
>::new();
let res = server
.client_streaming(ClientStreamingEchoServer { inner }, req)
.await;
Expand All @@ -237,39 +275,54 @@ pub mod echo_server {
struct BidirectionalStreamingEchoServer<T: Echo> {
inner: _Inner<T>,
}
impl<T: Echo> StreamingSvc<super::EchoRequest> for BidirectionalStreamingEchoServer<T> {
impl<T: Echo> StreamingSvc<super::EchoRequest>
for BidirectionalStreamingEchoServer<T> {
type Response = super::EchoResponse;
type ResponseStream = T::BidirectionalStreamingEchoStream;
type Future =
BoxFuture<Response<Self::ResponseStream>, dubbo::status::Status>;
type Future = BoxFuture<
Response<Self::ResponseStream>,
dubbo::status::Status,
>;
fn call(
&mut self,
request: Request<Decoding<super::EchoRequest>>,
) -> Self::Future {
let inner = self.inner.0.clone();
let fut =
async move { inner.bidirectional_streaming_echo(request).await };
let fut = async move {
inner.bidirectional_streaming_echo(request).await
};
Box::pin(fut)
}
}
let fut = async move {
let mut server =
TripleServer::<super::EchoRequest, super::EchoResponse>::new();
let mut server = TripleServer::<
super::EchoRequest,
super::EchoResponse,
>::new();
let res = server
.bidi_streaming(BidirectionalStreamingEchoServer { inner }, req)
.bidi_streaming(
BidirectionalStreamingEchoServer {
inner,
},
req,
)
.await;
Ok(res)
};
Box::pin(fut)
}
_ => Box::pin(async move {
Ok(http::Response::builder()
.status(200)
.header("grpc-status", "12")
.header("content-type", "application/grpc")
.body(empty_body())
.unwrap())
}),
_ => {
Box::pin(async move {
Ok(
http::Response::builder()
.status(200)
.header("grpc-status", "12")
.header("content-type", "application/grpc")
.body(empty_body())
.unwrap(),
)
})
}
}
}
}
Expand Down

0 comments on commit aff4c63

Please sign in to comment.