mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00
140 lines
4.3 KiB
Go
140 lines
4.3 KiB
Go
// Copyright 2016 The etcd Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package v3rpc
|
|
|
|
import (
|
|
"strings"
|
|
|
|
"github.com/coreos/etcd/auth"
|
|
"github.com/coreos/etcd/etcdserver"
|
|
"github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes"
|
|
"github.com/coreos/etcd/etcdserver/membership"
|
|
"github.com/coreos/etcd/lease"
|
|
"github.com/coreos/etcd/mvcc"
|
|
|
|
"google.golang.org/grpc"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
)
|
|
|
|
func togRPCError(err error) error {
|
|
switch err {
|
|
case membership.ErrIDRemoved:
|
|
return rpctypes.ErrGRPCMemberNotFound
|
|
case membership.ErrIDNotFound:
|
|
return rpctypes.ErrGRPCMemberNotFound
|
|
case membership.ErrIDExists:
|
|
return rpctypes.ErrGRPCMemberExist
|
|
case membership.ErrPeerURLexists:
|
|
return rpctypes.ErrGRPCPeerURLExist
|
|
case etcdserver.ErrNotEnoughStartedMembers:
|
|
return rpctypes.ErrMemberNotEnoughStarted
|
|
|
|
case mvcc.ErrCompacted:
|
|
return rpctypes.ErrGRPCCompacted
|
|
case mvcc.ErrFutureRev:
|
|
return rpctypes.ErrGRPCFutureRev
|
|
case etcdserver.ErrRequestTooLarge:
|
|
return rpctypes.ErrGRPCRequestTooLarge
|
|
case etcdserver.ErrNoSpace:
|
|
return rpctypes.ErrGRPCNoSpace
|
|
case etcdserver.ErrTooManyRequests:
|
|
return rpctypes.ErrTooManyRequests
|
|
|
|
case etcdserver.ErrNoLeader:
|
|
return rpctypes.ErrGRPCNoLeader
|
|
case etcdserver.ErrStopped:
|
|
return rpctypes.ErrGRPCStopped
|
|
case etcdserver.ErrTimeout:
|
|
return rpctypes.ErrGRPCTimeout
|
|
case etcdserver.ErrTimeoutDueToLeaderFail:
|
|
return rpctypes.ErrGRPCTimeoutDueToLeaderFail
|
|
case etcdserver.ErrTimeoutDueToConnectionLost:
|
|
return rpctypes.ErrGRPCTimeoutDueToConnectionLost
|
|
case etcdserver.ErrUnhealthy:
|
|
return rpctypes.ErrGRPCUnhealthy
|
|
case etcdserver.ErrKeyNotFound:
|
|
return rpctypes.ErrGRPCKeyNotFound
|
|
|
|
case lease.ErrLeaseNotFound:
|
|
return rpctypes.ErrGRPCLeaseNotFound
|
|
case lease.ErrLeaseExists:
|
|
return rpctypes.ErrGRPCLeaseExist
|
|
|
|
case auth.ErrRootUserNotExist:
|
|
return rpctypes.ErrGRPCRootUserNotExist
|
|
case auth.ErrRootRoleNotExist:
|
|
return rpctypes.ErrGRPCRootRoleNotExist
|
|
case auth.ErrUserAlreadyExist:
|
|
return rpctypes.ErrGRPCUserAlreadyExist
|
|
case auth.ErrUserEmpty:
|
|
return rpctypes.ErrGRPCUserEmpty
|
|
case auth.ErrUserNotFound:
|
|
return rpctypes.ErrGRPCUserNotFound
|
|
case auth.ErrRoleAlreadyExist:
|
|
return rpctypes.ErrGRPCRoleAlreadyExist
|
|
case auth.ErrRoleNotFound:
|
|
return rpctypes.ErrGRPCRoleNotFound
|
|
case auth.ErrAuthFailed:
|
|
return rpctypes.ErrGRPCAuthFailed
|
|
case auth.ErrPermissionDenied:
|
|
return rpctypes.ErrGRPCPermissionDenied
|
|
case auth.ErrRoleNotGranted:
|
|
return rpctypes.ErrGRPCRoleNotGranted
|
|
case auth.ErrPermissionNotGranted:
|
|
return rpctypes.ErrGRPCPermissionNotGranted
|
|
case auth.ErrAuthNotEnabled:
|
|
return rpctypes.ErrGRPCAuthNotEnabled
|
|
case auth.ErrInvalidAuthToken:
|
|
return rpctypes.ErrGRPCInvalidAuthToken
|
|
case auth.ErrInvalidAuthMgmt:
|
|
return rpctypes.ErrGRPCInvalidAuthMgmt
|
|
default:
|
|
return grpc.Errorf(codes.Unknown, err.Error())
|
|
}
|
|
}
|
|
|
|
func isClientCtxErr(ctxErr error, err error) bool {
|
|
if ctxErr != nil {
|
|
return true
|
|
}
|
|
|
|
ev, ok := status.FromError(err)
|
|
if !ok {
|
|
return false
|
|
}
|
|
|
|
switch ev.Code() {
|
|
case codes.Canceled, codes.DeadlineExceeded:
|
|
// client-side context cancel or deadline exceeded
|
|
// "rpc error: code = Canceled desc = context canceled"
|
|
// "rpc error: code = DeadlineExceeded desc = context deadline exceeded"
|
|
return true
|
|
case codes.Unavailable:
|
|
msg := ev.Message()
|
|
// client-side context cancel or deadline exceeded with TLS ("http2.errClientDisconnected")
|
|
// "rpc error: code = Unavailable desc = client disconnected"
|
|
if msg == "client disconnected" {
|
|
return true
|
|
}
|
|
// "grpc/transport.ClientTransport.CloseStream" on canceled streams
|
|
// "rpc error: code = Unavailable desc = stream error: stream ID 21; CANCEL")
|
|
if strings.HasPrefix(msg, "stream error: ") && strings.HasSuffix(msg, "; CANCEL") {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|