mirror of
https://github.com/SigNoz/signoz.git
synced 2026-06-20 15:20:31 +01:00
Some checks failed
build-staging / js-build (push) Has been cancelled
build-staging / prepare (push) Has been cancelled
build-staging / go-build (push) Has been cancelled
build-staging / staging (push) Has been cancelled
Release Drafter / update_release_draft (push) Has been cancelled
* feat: extend error responses with new error struct
* fix: enriched error for dashboard api
* fix: merge issues
* fix: reverted dashboards changes and add for cloud integrations
* fix: delete file
* fix: add back file
* fix: added a helper
* fix: removed invlaid referencess
* fix: generate openapi
* fix: keeping additional along with suggestion
* Revert "fix: keeping additional along with suggestion"
This reverts commit be30e2ffd2.
* fix: added suggestions per additonal error
* fix: generate openapi
* fix: remove valid references
* fix: removeg valid references for select and group by and only did you mean is kept
* fix: unit test
* fix: use binding for deconding for both ee and community
* fix: trim down suggestions methods
* fix: added renamed methods and moved stuff around
* fix: typo
* fix: removed json decoder
* fix: added empty check
* fix: retain addtional
* fix: reverted re-structing of file
131 lines
3.7 KiB
Go
131 lines
3.7 KiB
Go
package binding
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io"
|
|
"reflect"
|
|
"strings"
|
|
|
|
"github.com/SigNoz/signoz/pkg/errors"
|
|
)
|
|
|
|
const (
|
|
ErrMessageInvalidJSON string = "request body contains invalid JSON, please verify the format and try again."
|
|
ErrMessageInvalidField string = "request body contains invalid field value"
|
|
)
|
|
|
|
var _ BindingBody = (*jsonBinding)(nil)
|
|
|
|
type jsonBinding struct{}
|
|
|
|
func (b *jsonBinding) BindBody(body io.Reader, obj any, opts ...BindBodyOption) error {
|
|
bindBodyOptions := &bindBodyOptions{
|
|
DisallowUnknownFields: false,
|
|
UseNumber: false,
|
|
}
|
|
|
|
for _, opt := range opts {
|
|
opt(bindBodyOptions)
|
|
}
|
|
|
|
decoder := json.NewDecoder(body)
|
|
|
|
if bindBodyOptions.DisallowUnknownFields {
|
|
decoder.DisallowUnknownFields()
|
|
}
|
|
|
|
if bindBodyOptions.UseNumber {
|
|
decoder.UseNumber()
|
|
}
|
|
|
|
if err := decoder.Decode(obj); err != nil {
|
|
var syntaxError *json.SyntaxError
|
|
var unmarshalError *json.InvalidUnmarshalError
|
|
var unmarshalTypeError *json.UnmarshalTypeError
|
|
|
|
if errors.As(err, &syntaxError) || errors.As(err, &unmarshalError) || errors.Is(err, io.EOF) || errors.Is(err, io.ErrUnexpectedEOF) {
|
|
return errors.
|
|
New(errors.TypeInvalidInput, ErrCodeInvalidRequestBody, ErrMessageInvalidJSON).
|
|
WithAdditional(err.Error())
|
|
}
|
|
|
|
if errors.As(err, &unmarshalTypeError) {
|
|
if unmarshalTypeError.Field != "" {
|
|
return errors.
|
|
New(errors.TypeInvalidInput, ErrCodeInvalidRequestField, ErrMessageInvalidField).
|
|
WithAdditional("value of type '" + unmarshalTypeError.Value + "' was received for field '" + unmarshalTypeError.Field + "', try sending '" + unmarshalTypeError.Type.String() + "' instead?")
|
|
}
|
|
|
|
return errors.
|
|
New(errors.TypeInvalidInput, ErrCodeInvalidRequestField, ErrMessageInvalidField).
|
|
WithAdditional("value of type '" + unmarshalTypeError.Value + "' was received, try sending '" + unmarshalTypeError.Type.String() + "' instead?")
|
|
}
|
|
|
|
// DisallowUnknownFields surfaces a bare `json: unknown field "x"`; turn it
|
|
// into a structured invalid-input error with did-you-mean/valid-reference
|
|
// suggestions drawn from obj's own JSON field names. Gated on the strict
|
|
// flag so an already-structured "unknown field" error bubbling up from a
|
|
// nested UnmarshalJSON is passed through unchanged, not re-wrapped here with
|
|
// the wrong (outer) field set.
|
|
if bindBodyOptions.DisallowUnknownFields && strings.Contains(err.Error(), "unknown field") {
|
|
if field := extractUnknownField(err.Error()); field != "" {
|
|
message := "unknown field %q"
|
|
if bindBodyOptions.UnknownFieldContext != "" {
|
|
message = "unknown field %q in " + bindBodyOptions.UnknownFieldContext
|
|
}
|
|
|
|
return errors.
|
|
NewInvalidInputf(errors.CodeInvalidInput, message, field).
|
|
WithSuggestions(errors.SuggestionsOnLevenshteinDistance(field, JSONFieldNames(obj))...)
|
|
}
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// JSONFieldNames returns the JSON field names of a struct (or pointer to one),
|
|
// skipping fields tagged "-" or without a json tag.
|
|
func JSONFieldNames(v any) []string {
|
|
var fields []string
|
|
|
|
t := reflect.TypeOf(v)
|
|
if t.Kind() == reflect.Pointer {
|
|
t = t.Elem()
|
|
}
|
|
|
|
if t.Kind() != reflect.Struct {
|
|
return fields
|
|
}
|
|
|
|
for i := 0; i < t.NumField(); i++ {
|
|
field := t.Field(i)
|
|
jsonTag := field.Tag.Get("json")
|
|
|
|
if jsonTag == "" || jsonTag == "-" {
|
|
continue
|
|
}
|
|
|
|
fieldName := strings.Split(jsonTag, ",")[0]
|
|
if fieldName != "" {
|
|
fields = append(fields, fieldName)
|
|
}
|
|
}
|
|
|
|
return fields
|
|
}
|
|
|
|
// extractUnknownField pulls fieldname out of a `json: unknown field "fieldname"`
|
|
// decoder message, or returns "" when the message has no quoted field.
|
|
func extractUnknownField(errMsg string) string {
|
|
parts := strings.Split(errMsg, `"`)
|
|
|
|
if len(parts) >= 2 {
|
|
return parts[1]
|
|
}
|
|
|
|
return ""
|
|
}
|