crashinterceptor.go 1.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142
  1. package serverinterceptors
  2. import (
  3. "context"
  4. "runtime/debug"
  5. "github.com/zeromicro/go-zero/core/logx"
  6. "google.golang.org/grpc"
  7. "google.golang.org/grpc/codes"
  8. "google.golang.org/grpc/status"
  9. )
  10. // StreamCrashInterceptor catches panics in processing stream requests and recovers.
  11. func StreamCrashInterceptor(svr interface{}, stream grpc.ServerStream, _ *grpc.StreamServerInfo,
  12. handler grpc.StreamHandler) (err error) {
  13. defer handleCrash(func(r interface{}) {
  14. err = toPanicError(r)
  15. })
  16. return handler(svr, stream)
  17. }
  18. // UnaryCrashInterceptor catches panics in processing unary requests and recovers.
  19. func UnaryCrashInterceptor(ctx context.Context, req interface{}, _ *grpc.UnaryServerInfo,
  20. handler grpc.UnaryHandler) (resp interface{}, err error) {
  21. defer handleCrash(func(r interface{}) {
  22. err = toPanicError(r)
  23. })
  24. return handler(ctx, req)
  25. }
  26. func handleCrash(handler func(interface{})) {
  27. if r := recover(); r != nil {
  28. handler(r)
  29. }
  30. }
  31. func toPanicError(r interface{}) error {
  32. logx.Errorf("%+v\n\n%s", r, debug.Stack())
  33. return status.Errorf(codes.Internal, "panic: %v", r)
  34. }