为了账号安全,请及时绑定邮箱和手机立即绑定

Spring Boot gRPC:发生业务错误时如何返回错误码?

Spring Boot gRPC:发生业务错误时如何返回错误码?

杨__羊羊 2023-03-09 15:31:39
我正在使用LogNet grpc-spring-boot-starter实现 gRPC API 。例如,INVALID_ARGUMENT当传递了不正确的参数时,我想返回一个错误代码。如果我抛出自定义异常,它会以io.grpc.StatusRuntimeException: UNKNOWN.问:是否可以定义一些异常处理机制,使特定类型的异常始终导致正确的 gRPC 状态?不幸的是,项目中没有那么多文档。
查看完整描述

4 回答

?
MM们

TA贡献1886条经验 获得超2个赞

gRPC 不鼓励您抛出异常以将该错误传达给用户。这是因为意外泄露您可能没有考虑发送给客户的信息是微不足道的。

相反,我们鼓励您将StatusExceptionor传递StatusRuntimeExceptionstreamObserver.onError(Throwable). 如果您使用异常在您自己的代码中传达此信息,您可以在您的代码中放置一个 try-catch 并将异常传递给onError(). 例如,这对于 可能是公平的StatusException,因为它是一个已检查的异常。

TransmitStatusRuntimeExceptionInterceptor将在回调期间捕获异常,如果它是StatusRuntimeException,则关闭异常状态的调用。这与您的要求非常匹配,但默认情况下并未故意启用。


查看完整回答
反对 回复 2023-03-09
?
桃花长相依

TA贡献1860条经验 获得超8个赞

我刚刚发表了一篇关于gRPC Java 中的异常处理和错误传播主题的文章。


您可以使用拦截器处理异常,例如:


public class ExceptionHandler implements ServerInterceptor {


    @Override

    public <ReqT, RespT> ServerCall.Listener<ReqT> interceptCall(ServerCall<ReqT, RespT> serverCall, Metadata metadata,

                                                                 ServerCallHandler<ReqT, RespT> serverCallHandler) {

        ServerCall.Listener<ReqT> listener = serverCallHandler.startCall(serverCall, metadata);

        return new ExceptionHandlingServerCallListener<>(listener, serverCall, metadata);

    }


    private class ExceptionHandlingServerCallListener<ReqT, RespT>

            extends ForwardingServerCallListener.SimpleForwardingServerCallListener<ReqT> {

        private ServerCall<ReqT, RespT> serverCall;

        private Metadata metadata;


        ExceptionHandlingServerCallListener(ServerCall.Listener<ReqT> listener, ServerCall<ReqT, RespT> serverCall,

                                            Metadata metadata) {

            super(listener);

            this.serverCall = serverCall;

            this.metadata = metadata;

        }


        @Override

        public void onHalfClose() {

            try {

                super.onHalfClose();

            } catch (RuntimeException ex) {

                handleException(ex, serverCall, metadata);

                throw ex;

            }

        }


        @Override

        public void onReady() {

            try {

                super.onReady();

            } catch (RuntimeException ex) {

                handleException(ex, serverCall, metadata);

                throw ex;

            }

        }


        private void handleException(RuntimeException exception, ServerCall<ReqT, RespT> serverCall, Metadata metadata) {

            if (exception instanceof IllegalArgumentException) {

                serverCall.close(Status.INVALID_ARGUMENT.withDescription(exception.getMessage()), metadata);

            } else {

                serverCall.close(Status.UNKNOWN, metadata);

            }

        }

    }

}


查看完整回答
反对 回复 2023-03-09
?
绝地无双

TA贡献1946条经验 获得超4个赞

gRPC 不会传播错误。来自官方文档-

使用给定原因创建 Status 的派生实例。但是,原因不会从服务器传输到客户端。

如果您想将自定义信息从服务器传递到客户端,那么您有几个选择 -

  1. 使用元数据将错误信息从服务器传播到客户端

  2. 用于google.rpc.Status传递错误详细信息repeated google.protobuf.Any details

您需要在这两种情况下捕获异常,准备一条错误消息,并将其发送回客户端。

我写了一篇关于gRPC 错误处理的详细博客文章。


查看完整回答
反对 回复 2023-03-09
?
江户川乱折腾

TA贡献1851条经验 获得超5个赞

启动器的最新版本集成了 spring 验证支持。如果验证失败,它返回 INVALID_ARGUMENT。

披露:我是这个启动器的创建者。


查看完整回答
反对 回复 2023-03-09
  • 4 回答
  • 0 关注
  • 184 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信