All Downloads are FREE. Search and download functionalities are using the official Maven repository.

grpcstarter.server.feature.exceptionhandling.ExceptionHandlingServerInterceptor Maven / Gradle / Ivy

package grpcstarter.server.feature.exceptionhandling;

import io.grpc.ForwardingServerCallListener;
import io.grpc.Metadata;
import io.grpc.ServerCall;
import io.grpc.ServerCallHandler;
import io.grpc.ServerInterceptor;
import io.grpc.StatusRuntimeException;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.core.Ordered;

/**
 * @author Freeman
 */
public class ExceptionHandlingServerInterceptor implements ServerInterceptor, Ordered {
    public static final int ORDER = Ordered.LOWEST_PRECEDENCE - 1000;

    private final List grpcExceptionResolvers;
    private final List grpcUnhandledExceptionProcessors;

    public ExceptionHandlingServerInterceptor(
            ObjectProvider exceptionHandlers,
            ObjectProvider unhandledExceptionProcessors) {
        this.grpcExceptionResolvers = exceptionHandlers.orderedStream().collect(Collectors.toList());
        this.grpcUnhandledExceptionProcessors =
                unhandledExceptionProcessors.orderedStream().collect(Collectors.toList());
    }

    @Override
    public  ServerCall.Listener interceptCall(
            ServerCall call, Metadata headers, ServerCallHandler next) {
        return new GrpcExceptionHandlerListener<>(
                next.startCall(call, headers), call, headers, grpcExceptionResolvers, grpcUnhandledExceptionProcessors);
    }

    @Override
    public int getOrder() {
        return ORDER;
    }

    private static final class GrpcExceptionHandlerListener
            extends ForwardingServerCallListener.SimpleForwardingServerCallListener {
        private final ServerCall call;
        private final Metadata headers;
        private final List grpcExceptionResolvers;
        private final List grpcUnhandledExceptionProcessors;

        private GrpcExceptionHandlerListener(
                ServerCall.Listener delegate,
                ServerCall call,
                Metadata headers,
                List grpcExceptionResolvers,
                List grpcUnhandledExceptionProcessors) {
            super(delegate);
            this.call = call;
            this.headers = headers;
            this.grpcExceptionResolvers = grpcExceptionResolvers;
            this.grpcUnhandledExceptionProcessors = grpcUnhandledExceptionProcessors;
        }

        @Override
        public void onMessage(I message) {
            try {
                super.onMessage(message);
            } catch (Exception e) {
                if (!handle(e)) {
                    throw e;
                }
            }
        }

        @Override
        public void onHalfClose() {
            try {
                super.onHalfClose();
            } catch (Exception e) {
                if (!handle(e)) {
                    throw e;
                }
            }
        }

        private boolean handle(Exception e) {
            for (GrpcExceptionResolver resolver : grpcExceptionResolvers) {
                StatusRuntimeException sre = resolver.resolve(e, call, headers);
                if (sre != null) {
                    call.close(
                            sre.getStatus(),
                            Optional.ofNullable(sre.getTrailers()).orElseGet(Metadata::new));
                    return true;
                }
            }
            grpcUnhandledExceptionProcessors.forEach(processor -> processor.process(e, call, headers));
            return false;
        }
    }
}