Files
Ivana Huckova a69fac6e16 Mark grpc data source timeouts as cancelled queries (#100573)
* Set up to reproduce issue locally

* add check for deadline exceeded

* Revert "Set up to reproduce issue locally"

This reverts commit d8d9b354cab93e0e88edc739c4227cc75541b867.

* Trigger build

---------

Co-authored-by: Will Browne <will.browne@grafana.com>
2025-02-13 13:40:53 +01:00

107 lines
2.1 KiB
Go

package instrumentationutils
import (
"context"
"errors"
"strings"
grpccodes "google.golang.org/grpc/codes"
grpcstatus "google.golang.org/grpc/status"
"github.com/grafana/grafana-plugin-sdk-go/backend"
"github.com/grafana/grafana-plugin-sdk-go/genproto/pluginv2"
)
type RequestStatus int
const (
RequestStatusOK RequestStatus = iota
RequestStatusCancelled
RequestStatusError
)
func (status RequestStatus) String() string {
names := [...]string{"ok", "cancelled", "error"}
if status < RequestStatusOK || status > RequestStatusError {
return ""
}
return names[status]
}
func RequestStatusFromError(err error) RequestStatus {
status := RequestStatusOK
if err != nil {
status = RequestStatusError
if errors.Is(err, context.Canceled) {
status = RequestStatusCancelled
} else if s, ok := grpcstatus.FromError(err); ok && s.Code() == grpccodes.Canceled || s.Code() == grpccodes.DeadlineExceeded {
status = RequestStatusCancelled
}
}
return status
}
func RequestStatusFromErrorString(errString string) RequestStatus {
status := RequestStatusOK
if errString != "" {
status = RequestStatusError
if strings.Contains(errString, context.Canceled.Error()) || strings.Contains(errString, "code = Canceled") {
status = RequestStatusCancelled
}
}
return status
}
func RequestStatusFromQueryDataResponse(res *backend.QueryDataResponse, err error) RequestStatus {
if err != nil {
return RequestStatusFromError(err)
}
status := RequestStatusOK
if res != nil {
for _, dr := range res.Responses {
if dr.Error != nil {
s := RequestStatusFromError(dr.Error)
if s > status {
status = s
}
if status == RequestStatusError {
break
}
}
}
}
return status
}
func RequestStatusFromProtoQueryDataResponse(res *pluginv2.QueryDataResponse, err error) RequestStatus {
if err != nil {
return RequestStatusFromError(err)
}
status := RequestStatusOK
if res != nil {
for _, dr := range res.Responses {
if dr.Error != "" {
s := RequestStatusFromErrorString(dr.Error)
if s > status {
status = s
}
if status == RequestStatusError {
break
}
}
}
}
return status
}