Compare commits
5 Commits
Author | SHA1 | Date | |
---|---|---|---|
16c66ee28c
|
|||
2e6ca48d22
|
|||
b1d6509294 | |||
e909d656d9 | |||
0971f60c30
|
@@ -293,15 +293,7 @@ func fmtOutput(cs string, enums []EnumDef, pkgname string) string {
|
|||||||
str += "" + "\n"
|
str += "" + "\n"
|
||||||
|
|
||||||
str += "func (e " + enumdef.EnumTypeName + ") ValuesMeta() []EnumMetaValue {" + "\n"
|
str += "func (e " + enumdef.EnumTypeName + ") ValuesMeta() []EnumMetaValue {" + "\n"
|
||||||
str += " return []EnumMetaValue{" + "\n"
|
str += " return " + enumdef.EnumTypeName + "ValuesMeta()"
|
||||||
for _, v := range enumdef.Values {
|
|
||||||
if hasDescr {
|
|
||||||
str += " " + fmt.Sprintf("EnumMetaValue{VarName: \"%s\", Value: %s, Description: langext.Ptr(\"%s\")},", v.VarName, v.VarName, strings.TrimSpace(*v.Description)) + "\n"
|
|
||||||
} else {
|
|
||||||
str += " " + fmt.Sprintf("EnumMetaValue{VarName: \"%s\", Value: %s, Description: nil},", v.VarName, v.VarName) + "\n"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
str += " }" + "\n"
|
|
||||||
str += "}" + "\n"
|
str += "}" + "\n"
|
||||||
str += "" + "\n"
|
str += "" + "\n"
|
||||||
|
|
||||||
@@ -330,6 +322,15 @@ func fmtOutput(cs string, enums []EnumDef, pkgname string) string {
|
|||||||
str += "}" + "\n"
|
str += "}" + "\n"
|
||||||
str += "" + "\n"
|
str += "" + "\n"
|
||||||
|
|
||||||
|
str += "func (e " + enumdef.EnumTypeName + ") Meta() EnumMetaValue {" + "\n"
|
||||||
|
if hasDescr {
|
||||||
|
str += " return EnumMetaValue{VarName: e.VarName(), Value: e, Description: langext.Ptr(e.Description())}"
|
||||||
|
} else {
|
||||||
|
str += " return EnumMetaValue{VarName: e.VarName(), Value: e, Description: nil}"
|
||||||
|
}
|
||||||
|
str += "}" + "\n"
|
||||||
|
str += "" + "\n"
|
||||||
|
|
||||||
str += "func Parse" + enumdef.EnumTypeName + "(vv string) (" + enumdef.EnumTypeName + ", bool) {" + "\n"
|
str += "func Parse" + enumdef.EnumTypeName + "(vv string) (" + enumdef.EnumTypeName + ", bool) {" + "\n"
|
||||||
str += " for _, ev := range __" + enumdef.EnumTypeName + "Values {" + "\n"
|
str += " for _, ev := range __" + enumdef.EnumTypeName + "Values {" + "\n"
|
||||||
str += " if string(ev) == vv {" + "\n"
|
str += " if string(ev) == vv {" + "\n"
|
||||||
@@ -348,11 +349,7 @@ func fmtOutput(cs string, enums []EnumDef, pkgname string) string {
|
|||||||
str += "func " + enumdef.EnumTypeName + "ValuesMeta() []EnumMetaValue {" + "\n"
|
str += "func " + enumdef.EnumTypeName + "ValuesMeta() []EnumMetaValue {" + "\n"
|
||||||
str += " return []EnumMetaValue{" + "\n"
|
str += " return []EnumMetaValue{" + "\n"
|
||||||
for _, v := range enumdef.Values {
|
for _, v := range enumdef.Values {
|
||||||
if hasDescr {
|
str += " " + v.VarName + ".Meta(),\n"
|
||||||
str += " " + fmt.Sprintf("EnumMetaValue{VarName: \"%s\", Value: %s, Description: langext.Ptr(\"%s\")},", v.VarName, v.VarName, strings.TrimSpace(*v.Description)) + "\n"
|
|
||||||
} else {
|
|
||||||
str += " " + fmt.Sprintf("EnumMetaValue{VarName: \"%s\", Value: %s, Description: nil},", v.VarName, v.VarName) + "\n"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
str += " }" + "\n"
|
str += " }" + "\n"
|
||||||
str += "}" + "\n"
|
str += "}" + "\n"
|
||||||
|
@@ -2,10 +2,13 @@ package exerr
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/rs/zerolog"
|
"github.com/rs/zerolog"
|
||||||
"go.mongodb.org/mongo-driver/bson/primitive"
|
"go.mongodb.org/mongo-driver/bson/primitive"
|
||||||
|
"gogs.mikescher.com/BlackForestBytes/goext/dataext"
|
||||||
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
@@ -17,21 +20,21 @@ import (
|
|||||||
//
|
//
|
||||||
// ==== USAGE =====
|
// ==== USAGE =====
|
||||||
//
|
//
|
||||||
// If some method returns an error _always wrap it into an bmerror:
|
// If some method returns an error _always wrap it into an exerror:
|
||||||
// value, err := do_something(..)
|
// value, err := do_something(..)
|
||||||
// if err != nil {
|
// if err != nil {
|
||||||
// return nil, bmerror.Wrap(err, "do something failed").Build()
|
// return nil, exerror.Wrap(err, "do something failed").Build()
|
||||||
// }
|
// }
|
||||||
//
|
//
|
||||||
// If possible add metadata to the error (eg the id that was not found, ...), the methods are the same as in zerolog
|
// If possible add metadata to the error (eg the id that was not found, ...), the methods are the same as in zerolog
|
||||||
// return nil, bmerror.Wrap(err, "do something failed").Str("someid", id).Int("count", in.Count).Build()
|
// return nil, exerror.Wrap(err, "do something failed").Str("someid", id).Int("count", in.Count).Build()
|
||||||
//
|
//
|
||||||
// You can change the errortype with `.User()` and `.System()` (User-errors are 400 and System-errors 500)
|
// You can change the errortype with `.User()` and `.System()` (User-errors are 400 and System-errors 500)
|
||||||
// You can also manually set the statuscode with `.WithStatuscode(http.NotFound)`
|
// You can also manually set the statuscode with `.WithStatuscode(http.NotFound)`
|
||||||
// You can set the type with `WithType(..)`
|
// You can set the type with `WithType(..)`
|
||||||
//
|
//
|
||||||
// New Errors (that don't wrap an existing err object) are created with New
|
// New Errors (that don't wrap an existing err object) are created with New
|
||||||
// return nil, bmerror.New(bmerror.ErrInternal, "womethign wen horrible wrong").Build()
|
// return nil, exerror.New(exerror.TypeInternal, "womethign wen horrible wrong").Build()
|
||||||
// You can eitehr use an existing ErrorType, the "catch-all" ErrInternal, or add you own ErrType in consts.go
|
// You can eitehr use an existing ErrorType, the "catch-all" ErrInternal, or add you own ErrType in consts.go
|
||||||
//
|
//
|
||||||
// All errors should be handled one of the following four ways:
|
// All errors should be handled one of the following four ways:
|
||||||
@@ -64,37 +67,41 @@ func init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type Builder struct {
|
type Builder struct {
|
||||||
bmerror *bringmanError
|
errorData *ExErr
|
||||||
|
|
||||||
containsGinData bool
|
containsGinData bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func Get(err error) *Builder {
|
func Get(err error) *Builder {
|
||||||
return &Builder{bmerror: fromError(err)}
|
return &Builder{errorData: FromError(err)}
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(t ErrorType, msg string) *Builder {
|
func New(t ErrorType, msg string) *Builder {
|
||||||
return &Builder{bmerror: newBringmanErr(CatSystem, t, msg)}
|
return &Builder{errorData: newExErr(CatSystem, t, msg)}
|
||||||
}
|
}
|
||||||
|
|
||||||
func Wrap(err error, msg string) *Builder {
|
func Wrap(err error, msg string) *Builder {
|
||||||
return &Builder{bmerror: fromError(err).wrap(msg, CatWrap, 1)}
|
if !pkgconfig.RecursiveErrors {
|
||||||
|
v := FromError(err)
|
||||||
|
v.Message = msg
|
||||||
|
return &Builder{errorData: v}
|
||||||
|
}
|
||||||
|
return &Builder{errorData: wrapExErr(FromError(err), msg, CatWrap, 1)}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
func (b *Builder) WithType(t ErrorType) *Builder {
|
func (b *Builder) WithType(t ErrorType) *Builder {
|
||||||
b.bmerror.Type = t
|
b.errorData.Type = t
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) WithStatuscode(status int) *Builder {
|
func (b *Builder) WithStatuscode(status int) *Builder {
|
||||||
b.bmerror.StatusCode = status
|
b.errorData.StatusCode = &status
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) WithMessage(msg string) *Builder {
|
func (b *Builder) WithMessage(msg string) *Builder {
|
||||||
b.bmerror.Message = msg
|
b.errorData.Message = msg
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -119,7 +126,7 @@ func (b *Builder) WithMessage(msg string) *Builder {
|
|||||||
//
|
//
|
||||||
// - Send to the error-service
|
// - Send to the error-service
|
||||||
func (b *Builder) Err() *Builder {
|
func (b *Builder) Err() *Builder {
|
||||||
b.bmerror.Severity = SevErr
|
b.errorData.Severity = SevErr
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -138,7 +145,7 @@ func (b *Builder) Err() *Builder {
|
|||||||
//
|
//
|
||||||
// - Logged as Warn
|
// - Logged as Warn
|
||||||
func (b *Builder) Warn() *Builder {
|
func (b *Builder) Warn() *Builder {
|
||||||
b.bmerror.Severity = SevWarn
|
b.errorData.Severity = SevWarn
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -157,7 +164,7 @@ func (b *Builder) Warn() *Builder {
|
|||||||
//
|
//
|
||||||
// - -(nothing)-
|
// - -(nothing)-
|
||||||
func (b *Builder) Info() *Builder {
|
func (b *Builder) Info() *Builder {
|
||||||
b.bmerror.Severity = SevInfo
|
b.errorData.Severity = SevInfo
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -167,12 +174,12 @@ func (b *Builder) Info() *Builder {
|
|||||||
//
|
//
|
||||||
// Errors with category
|
// Errors with category
|
||||||
func (b *Builder) User() *Builder {
|
func (b *Builder) User() *Builder {
|
||||||
b.bmerror.Category = CatUser
|
b.errorData.Category = CatUser
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) System() *Builder {
|
func (b *Builder) System() *Builder {
|
||||||
b.bmerror.Category = CatSystem
|
b.errorData.Category = CatSystem
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -268,7 +275,7 @@ func (b *Builder) Stack() *Builder {
|
|||||||
|
|
||||||
func (b *Builder) Errs(key string, val []error) *Builder {
|
func (b *Builder) Errs(key string, val []error) *Builder {
|
||||||
for i, valerr := range val {
|
for i, valerr := range val {
|
||||||
b.addMeta(fmt.Sprintf("%v[%v]", key, i), MDTString, Get(valerr).toBMError().FormatLog(LogPrintFull))
|
b.addMeta(fmt.Sprintf("%v[%v]", key, i), MDTString, Get(valerr).errorData.FormatLog(LogPrintFull))
|
||||||
}
|
}
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
@@ -299,7 +306,7 @@ func (b *Builder) GinReq(ctx context.Context, g *gin.Context, req *http.Request)
|
|||||||
b.Str("gin.context.reqid", ctxVal)
|
b.Str("gin.context.reqid", ctxVal)
|
||||||
}
|
}
|
||||||
if req.Method != "GET" && req.Body != nil && req.Header.Get("Content-Type") == "application/json" {
|
if req.Method != "GET" && req.Body != nil && req.Header.Get("Content-Type") == "application/json" {
|
||||||
if brc, ok := req.Body.(langext.BufferedReadCloser); ok {
|
if brc, ok := req.Body.(dataext.BufferedReadCloser); ok {
|
||||||
if bin, err := brc.BufferedAll(); err == nil {
|
if bin, err := brc.BufferedAll(); err == nil {
|
||||||
if len(bin) < 16*1024 {
|
if len(bin) < 16*1024 {
|
||||||
var prettyJSON bytes.Buffer
|
var prettyJSON bytes.Buffer
|
||||||
@@ -348,13 +355,17 @@ func formatHeader(header map[string][]string) string {
|
|||||||
// Build creates a new error, ready to pass up the stack
|
// Build creates a new error, ready to pass up the stack
|
||||||
// If the errors is not SevWarn or SevInfo it gets also logged (in short form, without stacktrace) onto stdout
|
// If the errors is not SevWarn or SevInfo it gets also logged (in short form, without stacktrace) onto stdout
|
||||||
func (b *Builder) Build() error {
|
func (b *Builder) Build() error {
|
||||||
if b.bmerror.Severity == SevErr || b.bmerror.Severity == SevFatal {
|
warnOnPkgConfigNotInitialized()
|
||||||
b.bmerror.ShortLog(stackSkipLogger.Error())
|
|
||||||
|
if pkgconfig.ZeroLogErrTraces && (b.errorData.Severity == SevErr || b.errorData.Severity == SevFatal) {
|
||||||
|
b.errorData.ShortLog(stackSkipLogger.Error())
|
||||||
|
} else if pkgconfig.ZeroLogAllTraces {
|
||||||
|
b.errorData.ShortLog(stackSkipLogger.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
b.CallListener(MethodBuild)
|
b.CallListener(MethodBuild)
|
||||||
|
|
||||||
return b.bmerror.ToGrpcError()
|
return b.errorData
|
||||||
}
|
}
|
||||||
|
|
||||||
// Output prints the error onto the gin stdout.
|
// Output prints the error onto the gin stdout.
|
||||||
@@ -366,12 +377,12 @@ func (b *Builder) Output(ctx context.Context, g *gin.Context) {
|
|||||||
b.GinReq(ctx, g, g.Request)
|
b.GinReq(ctx, g, g.Request)
|
||||||
}
|
}
|
||||||
|
|
||||||
b.bmerror.Output(ctx, g)
|
b.errorData.Output(g)
|
||||||
|
|
||||||
if b.bmerror.Severity == SevErr || b.bmerror.Severity == SevFatal {
|
if b.errorData.Severity == SevErr || b.errorData.Severity == SevFatal {
|
||||||
b.bmerror.Log(stackSkipLogger.Error())
|
b.errorData.Log(stackSkipLogger.Error())
|
||||||
} else if b.bmerror.Severity == SevWarn {
|
} else if b.errorData.Severity == SevWarn {
|
||||||
b.bmerror.Log(stackSkipLogger.Warn())
|
b.errorData.Log(stackSkipLogger.Warn())
|
||||||
}
|
}
|
||||||
|
|
||||||
b.CallListener(MethodOutput)
|
b.CallListener(MethodOutput)
|
||||||
@@ -380,24 +391,24 @@ func (b *Builder) Output(ctx context.Context, g *gin.Context) {
|
|||||||
// Print prints the error
|
// Print prints the error
|
||||||
// If the error is SevErr we also send it to the error-service
|
// If the error is SevErr we also send it to the error-service
|
||||||
func (b *Builder) Print() {
|
func (b *Builder) Print() {
|
||||||
if b.bmerror.Severity == SevErr || b.bmerror.Severity == SevFatal {
|
if b.errorData.Severity == SevErr || b.errorData.Severity == SevFatal {
|
||||||
b.bmerror.Log(stackSkipLogger.Error())
|
b.errorData.Log(stackSkipLogger.Error())
|
||||||
} else if b.bmerror.Severity == SevWarn {
|
} else if b.errorData.Severity == SevWarn {
|
||||||
b.bmerror.ShortLog(stackSkipLogger.Warn())
|
b.errorData.ShortLog(stackSkipLogger.Warn())
|
||||||
}
|
}
|
||||||
|
|
||||||
b.CallListener(MethodPrint)
|
b.CallListener(MethodPrint)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) Format(level LogPrintLevel) string {
|
func (b *Builder) Format(level LogPrintLevel) string {
|
||||||
return b.bmerror.FormatLog(level)
|
return b.errorData.FormatLog(level)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fatal prints the error and terminates the program
|
// Fatal prints the error and terminates the program
|
||||||
// If the error is SevErr we also send it to the error-service
|
// If the error is SevErr we also send it to the error-service
|
||||||
func (b *Builder) Fatal() {
|
func (b *Builder) Fatal() {
|
||||||
b.bmerror.Severity = SevFatal
|
b.errorData.Severity = SevFatal
|
||||||
b.bmerror.Log(stackSkipLogger.WithLevel(zerolog.FatalLevel))
|
b.errorData.Log(stackSkipLogger.WithLevel(zerolog.FatalLevel))
|
||||||
|
|
||||||
b.CallListener(MethodFatal)
|
b.CallListener(MethodFatal)
|
||||||
|
|
||||||
@@ -407,10 +418,6 @@ func (b *Builder) Fatal() {
|
|||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
func (b *Builder) addMeta(key string, mdtype metaDataType, val interface{}) *Builder {
|
func (b *Builder) addMeta(key string, mdtype metaDataType, val interface{}) *Builder {
|
||||||
b.bmerror.Meta.add(key, mdtype, val)
|
b.errorData.Meta.add(key, mdtype, val)
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) toBMError() BMError {
|
|
||||||
return b.bmerror.ToBMError()
|
|
||||||
}
|
|
||||||
|
201
exerr/constructor.go
Normal file
201
exerr/constructor.go
Normal file
@@ -0,0 +1,201 @@
|
|||||||
|
package exerr
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"go.mongodb.org/mongo-driver/bson/primitive"
|
||||||
|
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
||||||
|
"reflect"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var reflectTypeStr = reflect.TypeOf("")
|
||||||
|
|
||||||
|
func FromError(err error) *ExErr {
|
||||||
|
if verr, ok := err.(*ExErr); ok {
|
||||||
|
// A simple ExErr
|
||||||
|
return verr
|
||||||
|
}
|
||||||
|
|
||||||
|
// A foreign error (eg a MongoDB exception)
|
||||||
|
return &ExErr{
|
||||||
|
UniqueID: newID(),
|
||||||
|
Category: CatForeign,
|
||||||
|
Type: TypeInternal,
|
||||||
|
Severity: SevErr,
|
||||||
|
Timestamp: time.Time{},
|
||||||
|
StatusCode: nil,
|
||||||
|
Message: err.Error(),
|
||||||
|
WrappedErrType: fmt.Sprintf("%T", err),
|
||||||
|
Caller: "",
|
||||||
|
OriginalError: nil,
|
||||||
|
Meta: getForeignMeta(err),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func newExErr(cat ErrorCategory, errtype ErrorType, msg string) *ExErr {
|
||||||
|
return &ExErr{
|
||||||
|
UniqueID: newID(),
|
||||||
|
Category: cat,
|
||||||
|
Type: errtype,
|
||||||
|
Severity: SevErr,
|
||||||
|
Timestamp: time.Now(),
|
||||||
|
StatusCode: nil,
|
||||||
|
Message: msg,
|
||||||
|
WrappedErrType: "",
|
||||||
|
Caller: callername(2),
|
||||||
|
OriginalError: nil,
|
||||||
|
Meta: make(map[string]MetaValue),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func wrapExErr(e *ExErr, msg string, cat ErrorCategory, stacktraceskip int) *ExErr {
|
||||||
|
return &ExErr{
|
||||||
|
UniqueID: newID(),
|
||||||
|
Category: cat,
|
||||||
|
Type: TypeWrap,
|
||||||
|
Severity: SevErr,
|
||||||
|
Timestamp: time.Now(),
|
||||||
|
StatusCode: e.StatusCode,
|
||||||
|
Message: msg,
|
||||||
|
WrappedErrType: "",
|
||||||
|
Caller: callername(1 + stacktraceskip),
|
||||||
|
OriginalError: e,
|
||||||
|
Meta: make(map[string]MetaValue),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getForeignMeta(err error) (mm MetaMap) {
|
||||||
|
mm = make(map[string]MetaValue)
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if panicerr := recover(); panicerr != nil {
|
||||||
|
New(TypePanic, "Panic while trying to get foreign meta").
|
||||||
|
Str("source", err.Error()).
|
||||||
|
Interface("panic-object", panicerr).
|
||||||
|
Stack().
|
||||||
|
Print()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
rval := reflect.ValueOf(err)
|
||||||
|
if rval.Kind() == reflect.Interface || rval.Kind() == reflect.Ptr {
|
||||||
|
rval = reflect.ValueOf(err).Elem()
|
||||||
|
}
|
||||||
|
|
||||||
|
mm.add("foreign.errortype", MDTString, rval.Type().String())
|
||||||
|
|
||||||
|
for k, v := range addMetaPrefix("foreign", getReflectedMetaValues(err, 8)) {
|
||||||
|
mm[k] = v
|
||||||
|
}
|
||||||
|
|
||||||
|
return mm
|
||||||
|
}
|
||||||
|
|
||||||
|
func getReflectedMetaValues(value interface{}, remainingDepth int) map[string]MetaValue {
|
||||||
|
|
||||||
|
if remainingDepth <= 0 {
|
||||||
|
return map[string]MetaValue{}
|
||||||
|
}
|
||||||
|
|
||||||
|
if langext.IsNil(value) {
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTNil, Value: nil}}
|
||||||
|
}
|
||||||
|
|
||||||
|
rval := reflect.ValueOf(value)
|
||||||
|
|
||||||
|
if rval.Type().Kind() == reflect.Ptr {
|
||||||
|
|
||||||
|
if rval.IsNil() {
|
||||||
|
return map[string]MetaValue{"*": {DataType: MDTNil, Value: nil}}
|
||||||
|
}
|
||||||
|
|
||||||
|
elem := rval.Elem()
|
||||||
|
|
||||||
|
return addMetaPrefix("*", getReflectedMetaValues(elem.Interface(), remainingDepth-1))
|
||||||
|
}
|
||||||
|
|
||||||
|
if !rval.CanInterface() {
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTString, Value: "<<no-interface>>"}}
|
||||||
|
}
|
||||||
|
|
||||||
|
raw := rval.Interface()
|
||||||
|
|
||||||
|
switch ifraw := raw.(type) {
|
||||||
|
case time.Time:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTTime, Value: ifraw}}
|
||||||
|
case time.Duration:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTDuration, Value: ifraw}}
|
||||||
|
case int:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTInt, Value: ifraw}}
|
||||||
|
case int8:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTInt8, Value: ifraw}}
|
||||||
|
case int16:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTInt16, Value: ifraw}}
|
||||||
|
case int32:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTInt32, Value: ifraw}}
|
||||||
|
case int64:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTInt64, Value: ifraw}}
|
||||||
|
case string:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTString, Value: ifraw}}
|
||||||
|
case bool:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTBool, Value: ifraw}}
|
||||||
|
case []byte:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTBytes, Value: ifraw}}
|
||||||
|
case float32:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTFloat32, Value: ifraw}}
|
||||||
|
case float64:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTFloat64, Value: ifraw}}
|
||||||
|
case []int:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTIntArray, Value: ifraw}}
|
||||||
|
case []int32:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTInt32Array, Value: ifraw}}
|
||||||
|
case primitive.ObjectID:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTObjectID, Value: ifraw}}
|
||||||
|
case []string:
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTStringArray, Value: ifraw}}
|
||||||
|
}
|
||||||
|
|
||||||
|
if rval.Type().Kind() == reflect.Struct {
|
||||||
|
m := make(map[string]MetaValue)
|
||||||
|
for i := 0; i < rval.NumField(); i++ {
|
||||||
|
fieldtype := rval.Type().Field(i)
|
||||||
|
|
||||||
|
fieldname := fieldtype.Name
|
||||||
|
|
||||||
|
if fieldtype.IsExported() {
|
||||||
|
for k, v := range addMetaPrefix(fieldname, getReflectedMetaValues(rval.Field(i).Interface(), remainingDepth-1)) {
|
||||||
|
m[k] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
|
||||||
|
if rval.Type().ConvertibleTo(reflectTypeStr) {
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTString, Value: rval.Convert(reflectTypeStr).String()}}
|
||||||
|
}
|
||||||
|
|
||||||
|
jsonval, err := json.Marshal(value)
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // gets recovered later up
|
||||||
|
}
|
||||||
|
|
||||||
|
return map[string]MetaValue{"": {DataType: MDTString, Value: string(jsonval)}}
|
||||||
|
}
|
||||||
|
|
||||||
|
func addMetaPrefix(prefix string, m map[string]MetaValue) map[string]MetaValue {
|
||||||
|
if len(m) == 1 {
|
||||||
|
for k, v := range m {
|
||||||
|
if k == "" {
|
||||||
|
return map[string]MetaValue{prefix: v}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
r := make(map[string]MetaValue, len(m))
|
||||||
|
for k, v := range m {
|
||||||
|
r[prefix+"."+k] = v
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
@@ -1,5 +1,9 @@
|
|||||||
package exerr
|
package exerr
|
||||||
|
|
||||||
|
import (
|
||||||
|
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
||||||
|
)
|
||||||
|
|
||||||
type ErrorCategory struct{ Category string }
|
type ErrorCategory struct{ Category string }
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@@ -24,9 +28,33 @@ var (
|
|||||||
|
|
||||||
var AllSeverities = []ErrorSeverity{SevTrace, SevDebug, SevInfo, SevWarn, SevErr, SevFatal}
|
var AllSeverities = []ErrorSeverity{SevTrace, SevDebug, SevInfo, SevWarn, SevErr, SevFatal}
|
||||||
|
|
||||||
type ErrorType struct{ Key string }
|
type ErrorType struct {
|
||||||
|
Key string
|
||||||
|
DefaultStatusCode *int
|
||||||
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
TypeInternal = ErrorType{"Internal"}
|
TypeInternal = ErrorType{"INTERNAL_ERROR", langext.Ptr(500)}
|
||||||
|
TypePanic = ErrorType{"PANIC", langext.Ptr(500)}
|
||||||
|
TypeNotImplemented = ErrorType{"NOT_IMPLEMENTED", langext.Ptr(500)}
|
||||||
|
|
||||||
|
TypeWrap = ErrorType{"Wrap", nil}
|
||||||
|
|
||||||
|
TypeBindFailURI = ErrorType{"BINDFAIL_URI", langext.Ptr(400)}
|
||||||
|
TypeBindFailQuery = ErrorType{"BINDFAIL_QUERY", langext.Ptr(400)}
|
||||||
|
TypeBindFailJSON = ErrorType{"BINDFAIL_JSON", langext.Ptr(400)}
|
||||||
|
TypeBindFailFormData = ErrorType{"BINDFAIL_FORMDATA", langext.Ptr(400)}
|
||||||
|
|
||||||
|
TypeUnauthorized = ErrorType{"UNAUTHORIZED", langext.Ptr(401)}
|
||||||
|
TypeAuthFailed = ErrorType{"AUTH_FAILED", langext.Ptr(401)}
|
||||||
|
|
||||||
// other values come from pkgconfig
|
// other values come from pkgconfig
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type LogPrintLevel string
|
||||||
|
|
||||||
|
const (
|
||||||
|
LogPrintFull LogPrintLevel = "Full"
|
||||||
|
LogPrintOverview LogPrintLevel = "Overview"
|
||||||
|
LogPrintShort LogPrintLevel = "Short"
|
||||||
|
)
|
||||||
|
@@ -1,23 +1,37 @@
|
|||||||
package exerr
|
package exerr
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
||||||
|
)
|
||||||
|
|
||||||
type ErrorPackageConfig struct {
|
type ErrorPackageConfig struct {
|
||||||
ZeroLogTraces bool // autom print zerolog logs on CreateError
|
ZeroLogErrTraces bool // autom print zerolog logs on .Build() (for SevErr and SevFatal)
|
||||||
|
ZeroLogAllTraces bool // autom print zerolog logs on .Build() (for all Severities)
|
||||||
RecursiveErrors bool // errors contains their Origin-Error
|
RecursiveErrors bool // errors contains their Origin-Error
|
||||||
Types []ErrorType // all available error-types
|
ExtendedGinOutput bool // Log extended data (trace, meta, ...) to gin in err.Output()
|
||||||
|
ExtendGinOutput func(json map[string]any) // (Optionally) extend the gin output with more fields
|
||||||
|
ExtendGinDataOutput func(json map[string]any) // (Optionally) extend the gin `__data` output with more fields
|
||||||
}
|
}
|
||||||
|
|
||||||
type ErrorPackageConfigInit struct {
|
type ErrorPackageConfigInit struct {
|
||||||
LogTraces bool
|
ZeroLogErrTraces bool
|
||||||
|
ZeroLogAllTraces bool
|
||||||
RecursiveErrors bool
|
RecursiveErrors bool
|
||||||
InitTypes func(_ func(_ string) ErrorType)
|
ExtendedGinOutput bool
|
||||||
|
ExtendGinOutput *func(json map[string]any)
|
||||||
|
ExtendGinDataOutput *func(json map[string]any)
|
||||||
}
|
}
|
||||||
|
|
||||||
var initialized = false
|
var initialized = false
|
||||||
|
|
||||||
var pkgconfig = ErrorPackageConfig{
|
var pkgconfig = ErrorPackageConfig{
|
||||||
ZeroLogTraces: true,
|
ZeroLogErrTraces: true,
|
||||||
|
ZeroLogAllTraces: false,
|
||||||
RecursiveErrors: true,
|
RecursiveErrors: true,
|
||||||
Types: []ErrorType{TypeInternal},
|
ExtendedGinOutput: false,
|
||||||
|
ExtendGinOutput: func(json map[string]any) {},
|
||||||
|
ExtendGinDataOutput: func(json map[string]any) {},
|
||||||
}
|
}
|
||||||
|
|
||||||
// Init initializes the exerr packages
|
// Init initializes the exerr packages
|
||||||
@@ -28,23 +42,25 @@ func Init(cfg ErrorPackageConfigInit) {
|
|||||||
panic("Cannot re-init error package")
|
panic("Cannot re-init error package")
|
||||||
}
|
}
|
||||||
|
|
||||||
types := pkgconfig.Types
|
|
||||||
|
|
||||||
fnAddType := func(v string) ErrorType {
|
|
||||||
et := ErrorType{v}
|
|
||||||
types = append(types, et)
|
|
||||||
return et
|
|
||||||
}
|
|
||||||
|
|
||||||
if cfg.InitTypes != nil {
|
|
||||||
cfg.InitTypes(fnAddType)
|
|
||||||
}
|
|
||||||
|
|
||||||
pkgconfig = ErrorPackageConfig{
|
pkgconfig = ErrorPackageConfig{
|
||||||
ZeroLogTraces: cfg.LogTraces,
|
ZeroLogErrTraces: cfg.ZeroLogErrTraces,
|
||||||
|
ZeroLogAllTraces: cfg.ZeroLogAllTraces,
|
||||||
RecursiveErrors: cfg.RecursiveErrors,
|
RecursiveErrors: cfg.RecursiveErrors,
|
||||||
Types: types,
|
ExtendedGinOutput: cfg.ExtendedGinOutput,
|
||||||
|
ExtendGinOutput: langext.Coalesce(cfg.ExtendGinOutput, func(json map[string]any) {}),
|
||||||
|
ExtendGinDataOutput: langext.Coalesce(cfg.ExtendGinDataOutput, func(json map[string]any) {}),
|
||||||
}
|
}
|
||||||
|
|
||||||
initialized = true
|
initialized = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func warnOnPkgConfigNotInitialized() {
|
||||||
|
if !initialized {
|
||||||
|
fmt.Printf("\n")
|
||||||
|
fmt.Printf("%s\n", langext.StrRepeat("=", 80))
|
||||||
|
fmt.Printf("%s\n", "[WARNING] exerr package used without initializiation")
|
||||||
|
fmt.Printf("%s\n", " call exerr.Init() in your main() function")
|
||||||
|
fmt.Printf("%s\n", langext.StrRepeat("=", 80))
|
||||||
|
fmt.Printf("\n")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
169
exerr/exerr.go
169
exerr/exerr.go
@@ -1,6 +1,10 @@
|
|||||||
package exerr
|
package exerr
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/rs/xid"
|
||||||
|
"github.com/rs/zerolog"
|
||||||
|
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -12,7 +16,10 @@ type ExErr struct {
|
|||||||
Severity ErrorSeverity `json:"severity"`
|
Severity ErrorSeverity `json:"severity"`
|
||||||
Type ErrorType `json:"type"`
|
Type ErrorType `json:"type"`
|
||||||
|
|
||||||
|
StatusCode *int `json:"statusCode"`
|
||||||
|
|
||||||
Message string `json:"message"`
|
Message string `json:"message"`
|
||||||
|
WrappedErrType string `json:"wrappedErrType"`
|
||||||
Caller string `json:"caller"`
|
Caller string `json:"caller"`
|
||||||
|
|
||||||
OriginalError *ExErr
|
OriginalError *ExErr
|
||||||
@@ -20,14 +27,166 @@ type ExErr struct {
|
|||||||
Meta MetaMap `json:"meta"`
|
Meta MetaMap `json:"meta"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ee ExErr) Error() string {
|
func (ee *ExErr) Error() string {
|
||||||
|
return ee.Message
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ee *ExErr) Unwrap() error {
|
||||||
|
return ee.OriginalError
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ee *ExErr) Log(evt *zerolog.Event) {
|
||||||
|
evt.Msg(ee.FormatLog(LogPrintFull))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ee *ExErr) FormatLog(lvl LogPrintLevel) string {
|
||||||
|
if lvl == LogPrintShort {
|
||||||
|
|
||||||
|
msg := ee.Message
|
||||||
|
if ee.OriginalError != nil && ee.OriginalError.Category == CatForeign {
|
||||||
|
msg = msg + " (" + strings.ReplaceAll(ee.OriginalError.Message, "\n", " ") + ")"
|
||||||
|
}
|
||||||
|
|
||||||
|
if ee.Type != TypeWrap {
|
||||||
|
return "[" + ee.Type.Key + "] " + msg
|
||||||
|
} else {
|
||||||
|
return msg
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if lvl == LogPrintOverview {
|
||||||
|
|
||||||
|
str := "[" + ee.RecursiveType().Key + "] <" + ee.UniqueID + "> " + strings.ReplaceAll(ee.RecursiveMessage(), "\n", " ") + "\n"
|
||||||
|
|
||||||
|
indent := ""
|
||||||
|
for curr := ee; curr != nil; curr = curr.OriginalError {
|
||||||
|
indent += " "
|
||||||
|
|
||||||
|
str += indent
|
||||||
|
str += "-> "
|
||||||
|
strmsg := strings.Trim(curr.Message, " \r\n\t")
|
||||||
|
if lbidx := strings.Index(curr.Message, "\n"); lbidx >= 0 {
|
||||||
|
strmsg = strmsg[0:lbidx]
|
||||||
|
}
|
||||||
|
strmsg = langext.StrLimit(strmsg, 61, "...")
|
||||||
|
str += strmsg
|
||||||
|
str += "\n"
|
||||||
|
|
||||||
|
}
|
||||||
|
return str
|
||||||
|
|
||||||
|
} else if lvl == LogPrintFull {
|
||||||
|
|
||||||
|
str := "[" + ee.RecursiveType().Key + "] <" + ee.UniqueID + "> " + strings.ReplaceAll(ee.RecursiveMessage(), "\n", " ") + "\n"
|
||||||
|
|
||||||
|
indent := ""
|
||||||
|
for curr := ee; curr != nil; curr = curr.OriginalError {
|
||||||
|
indent += " "
|
||||||
|
|
||||||
|
etype := ee.Type.Key
|
||||||
|
if ee.Type == TypeWrap {
|
||||||
|
etype = "~"
|
||||||
|
}
|
||||||
|
|
||||||
|
str += indent
|
||||||
|
str += "-> ["
|
||||||
|
str += etype
|
||||||
|
if curr.Category == CatForeign {
|
||||||
|
str += "|Foreign"
|
||||||
|
}
|
||||||
|
str += "] "
|
||||||
|
str += strings.ReplaceAll(curr.Message, "\n", " ")
|
||||||
|
if curr.Caller != "" {
|
||||||
|
str += " (@ "
|
||||||
|
str += curr.Caller
|
||||||
|
str += ")"
|
||||||
|
}
|
||||||
|
str += "\n"
|
||||||
|
|
||||||
|
if curr.Meta.Any() {
|
||||||
|
meta := indent + " {" + curr.Meta.FormatOneLine(240) + "}"
|
||||||
|
if len(meta) < 200 {
|
||||||
|
str += meta
|
||||||
|
str += "\n"
|
||||||
|
} else {
|
||||||
|
str += curr.Meta.FormatMultiLine(indent+" ", " ", 1024)
|
||||||
|
str += "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return str
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
return "[?[" + ee.UniqueID + "]?]"
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ee *ExErr) ShortLog(evt *zerolog.Event) {
|
||||||
|
ee.Meta.Apply(evt).Msg(ee.FormatLog(LogPrintShort))
|
||||||
|
}
|
||||||
|
|
||||||
|
// RecursiveMessage returns the message to show
|
||||||
|
// = first error (top-down) that is not wrapping/foreign/empty
|
||||||
|
func (ee *ExErr) RecursiveMessage() string {
|
||||||
|
for curr := ee; curr != nil; curr = curr.OriginalError {
|
||||||
|
if curr.Message != "" && curr.Category != CatWrap && curr.Category != CatForeign {
|
||||||
|
return curr.Message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// fallback to self
|
||||||
|
return ee.Message
|
||||||
|
}
|
||||||
|
|
||||||
|
// RecursiveType returns the statuscode to use
|
||||||
|
// = first error (top-down) that is not wrapping/empty
|
||||||
|
func (ee *ExErr) RecursiveType() ErrorType {
|
||||||
|
for curr := ee; curr != nil; curr = curr.OriginalError {
|
||||||
|
if curr.Type != TypeWrap {
|
||||||
|
return curr.Type
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ee ExErr) Unwrap() error {
|
// fallback to self
|
||||||
|
return ee.Type
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ee ExErr) Is(err error) bool {
|
// RecursiveStatuscode returns the HTTP Statuscode to use
|
||||||
|
// = first error (top-down) that has a statuscode set
|
||||||
|
func (ee *ExErr) RecursiveStatuscode() *int {
|
||||||
|
for curr := ee; curr != nil; curr = curr.OriginalError {
|
||||||
|
if curr.StatusCode != nil {
|
||||||
|
return langext.Ptr(*curr.StatusCode)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// fallback to <empty>
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RecursiveCategory returns the ErrorCategory to use
|
||||||
|
// = first error (top-down) that has a statuscode set
|
||||||
|
func (ee *ExErr) RecursiveCategory() ErrorCategory {
|
||||||
|
for curr := ee; curr != nil; curr = curr.OriginalError {
|
||||||
|
if curr.Category != CatWrap {
|
||||||
|
return curr.Category
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// fallback to <empty>
|
||||||
|
return ee.Category
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ee *ExErr) Depth() int {
|
||||||
|
if ee.OriginalError == nil {
|
||||||
|
return 1
|
||||||
|
} else {
|
||||||
|
return ee.OriginalError.Depth() + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func newID() string {
|
||||||
|
return xid.New().String()
|
||||||
}
|
}
|
||||||
|
84
exerr/gin.go
Normal file
84
exerr/gin.go
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
package exerr
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
json "gogs.mikescher.com/BlackForestBytes/goext/gojson"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (ee *ExErr) toJson() gin.H {
|
||||||
|
json := gin.H{}
|
||||||
|
|
||||||
|
if ee.UniqueID != "" {
|
||||||
|
json["id"] = ee.UniqueID
|
||||||
|
}
|
||||||
|
if ee.Category != CatWrap {
|
||||||
|
json["category"] = ee.Category
|
||||||
|
}
|
||||||
|
if ee.Type != TypeWrap {
|
||||||
|
json["type"] = ee.Type
|
||||||
|
}
|
||||||
|
if ee.StatusCode != nil {
|
||||||
|
json["statuscode"] = ee.StatusCode
|
||||||
|
}
|
||||||
|
if ee.Message != "" {
|
||||||
|
json["message"] = ee.Message
|
||||||
|
}
|
||||||
|
if ee.Caller != "" {
|
||||||
|
json["caller"] = ee.Caller
|
||||||
|
}
|
||||||
|
if ee.Severity != SevErr {
|
||||||
|
json["severity"] = ee.Severity
|
||||||
|
}
|
||||||
|
if ee.Timestamp != (time.Time{}) {
|
||||||
|
json["time"] = ee.Timestamp.Format(time.RFC3339)
|
||||||
|
}
|
||||||
|
if ee.WrappedErrType != "" {
|
||||||
|
json["wrappedErrType"] = ee.WrappedErrType
|
||||||
|
}
|
||||||
|
if ee.OriginalError != nil {
|
||||||
|
json["original"] = ee.OriginalError.toJson()
|
||||||
|
}
|
||||||
|
|
||||||
|
pkgconfig.ExtendGinDataOutput(json)
|
||||||
|
|
||||||
|
return json
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ee *ExErr) Output(g *gin.Context) {
|
||||||
|
var statuscode = http.StatusInternalServerError
|
||||||
|
|
||||||
|
var baseCat = ee.RecursiveCategory()
|
||||||
|
var baseType = ee.RecursiveType()
|
||||||
|
var baseStatuscode = ee.RecursiveStatuscode()
|
||||||
|
|
||||||
|
if baseCat == CatUser {
|
||||||
|
statuscode = http.StatusBadRequest
|
||||||
|
} else if baseCat == CatSystem {
|
||||||
|
statuscode = http.StatusInternalServerError
|
||||||
|
}
|
||||||
|
|
||||||
|
if baseStatuscode != nil {
|
||||||
|
statuscode = *ee.StatusCode
|
||||||
|
} else if baseType.DefaultStatusCode != nil {
|
||||||
|
statuscode = *baseType.DefaultStatusCode
|
||||||
|
}
|
||||||
|
|
||||||
|
warnOnPkgConfigNotInitialized()
|
||||||
|
|
||||||
|
ginOutput := gin.H{
|
||||||
|
"errorid": ee.UniqueID,
|
||||||
|
"message": ee.RecursiveMessage(),
|
||||||
|
"errorcode": ee.RecursiveType(),
|
||||||
|
"category": ee.RecursiveCategory(),
|
||||||
|
}
|
||||||
|
|
||||||
|
if pkgconfig.ExtendedGinOutput {
|
||||||
|
ginOutput["__data"] = ee.toJson()
|
||||||
|
}
|
||||||
|
|
||||||
|
pkgconfig.ExtendGinOutput(ginOutput)
|
||||||
|
|
||||||
|
g.Render(statuscode, json.GoJsonRender{Data: ginOutput, NilSafeSlices: true, NilSafeMaps: true})
|
||||||
|
}
|
86
exerr/helper.go
Normal file
86
exerr/helper.go
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
package exerr
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
// IsType test if the supplied error is of the specified ErrorType.
|
||||||
|
func IsType(err error, errType ErrorType) bool {
|
||||||
|
if err == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
bmerr := FromError(err)
|
||||||
|
for bmerr != nil {
|
||||||
|
if bmerr.Type == errType {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
bmerr = bmerr.OriginalError
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsFrom test if the supplied error stems originally from original
|
||||||
|
func IsFrom(e error, original error) bool {
|
||||||
|
if e == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if e == original {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
bmerr := FromError(e)
|
||||||
|
for bmerr == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for curr := bmerr; curr != nil; curr = curr.OriginalError {
|
||||||
|
if curr.Category == CatForeign && curr.Message == original.Error() && curr.WrappedErrType == fmt.Sprintf("%T", original) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// HasSourceMessage tests if the supplied error stems originally from an error with the message msg
|
||||||
|
func HasSourceMessage(e error, msg string) bool {
|
||||||
|
if e == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
bmerr := FromError(e)
|
||||||
|
for bmerr == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for curr := bmerr; curr != nil; curr = curr.OriginalError {
|
||||||
|
if curr.OriginalError == nil && curr.Message == msg {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func MessageMatch(e error, matcher func(string) bool) bool {
|
||||||
|
if e == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if matcher(e.Error()) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
bmerr := FromError(e)
|
||||||
|
for bmerr == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for curr := bmerr; curr != nil; curr = curr.OriginalError {
|
||||||
|
if matcher(curr.Message) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
@@ -13,7 +13,7 @@ const (
|
|||||||
MethodFatal Method = "FATAL"
|
MethodFatal Method = "FATAL"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Listener = func(method Method, v ExErr)
|
type Listener = func(method Method, v *ExErr)
|
||||||
|
|
||||||
var listenerLock = sync.Mutex{}
|
var listenerLock = sync.Mutex{}
|
||||||
var listener = make([]Listener, 0)
|
var listener = make([]Listener, 0)
|
||||||
@@ -26,7 +26,7 @@ func RegisterListener(l Listener) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (b *Builder) CallListener(m Method) {
|
func (b *Builder) CallListener(m Method) {
|
||||||
valErr := b.toBMError()
|
valErr := b.errorData
|
||||||
|
|
||||||
listenerLock.Lock()
|
listenerLock.Lock()
|
||||||
defer listenerLock.Unlock()
|
defer listenerLock.Unlock()
|
||||||
|
@@ -6,7 +6,6 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/rs/zerolog"
|
"github.com/rs/zerolog"
|
||||||
"github.com/rs/zerolog/log"
|
|
||||||
"go.mongodb.org/mongo-driver/bson"
|
"go.mongodb.org/mongo-driver/bson"
|
||||||
"go.mongodb.org/mongo-driver/bson/primitive"
|
"go.mongodb.org/mongo-driver/bson/primitive"
|
||||||
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
||||||
@@ -15,6 +14,10 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// This is a buffed up map[string]any
|
||||||
|
// we also save type information of the map-values
|
||||||
|
// which allows us to deserialize them back into te correct types later
|
||||||
|
|
||||||
type MetaMap map[string]MetaValue
|
type MetaMap map[string]MetaValue
|
||||||
|
|
||||||
type metaDataType string
|
type metaDataType string
|
||||||
@@ -350,11 +353,7 @@ func (v *MetaValue) Deserialize(value string, datatype metaDataType) error {
|
|||||||
v.DataType = datatype
|
v.DataType = datatype
|
||||||
return nil
|
return nil
|
||||||
} else {
|
} else {
|
||||||
r, err := valueFromProto(value[1:], MDTString)
|
v.Value = langext.Ptr(value[1:])
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
v.Value = langext.Ptr(r.Value.(string))
|
|
||||||
v.DataType = datatype
|
v.DataType = datatype
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@@ -586,51 +585,6 @@ func (v MetaValue) ValueString() string {
|
|||||||
return "(err)"
|
return "(err)"
|
||||||
}
|
}
|
||||||
|
|
||||||
func valueFromProto(value string, datatype metaDataType) (MetaValue, error) {
|
|
||||||
obj := MetaValue{}
|
|
||||||
err := obj.Deserialize(value, datatype)
|
|
||||||
if err != nil {
|
|
||||||
return MetaValue{}, err
|
|
||||||
}
|
|
||||||
return obj, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func metaFromProto(proto []*spbmodels.CustomError_MetaValue) MetaMap {
|
|
||||||
r := make(MetaMap)
|
|
||||||
|
|
||||||
for _, v := range proto {
|
|
||||||
mval, err := valueFromProto(v.Value, metaDataType(v.Type))
|
|
||||||
if err != nil {
|
|
||||||
log.Warn().Err(err).Msg("metaFromProto failed for " + v.Key)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
r[v.Key] = mval
|
|
||||||
}
|
|
||||||
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
func (mm MetaMap) ToProto() []*spbmodels.CustomError_MetaValue {
|
|
||||||
if mm == nil {
|
|
||||||
return make([]*spbmodels.CustomError_MetaValue, 0)
|
|
||||||
}
|
|
||||||
r := make([]*spbmodels.CustomError_MetaValue, 0, len(mm))
|
|
||||||
for k, v := range mm {
|
|
||||||
strval, err := v.SerializeValue()
|
|
||||||
if err != nil {
|
|
||||||
log.Warn().Err(err).Msg("MetaMap.ToProto failed for " + k)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
r = append(r, &spbmodels.CustomError_MetaValue{
|
|
||||||
Key: k,
|
|
||||||
Type: string(v.DataType),
|
|
||||||
Value: strval,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
func (mm MetaMap) FormatOneLine(singleMaxLen int) string {
|
func (mm MetaMap) FormatOneLine(singleMaxLen int) string {
|
||||||
r := ""
|
r := ""
|
||||||
|
|
||||||
|
14
exerr/stacktrace.go
Normal file
14
exerr/stacktrace.go
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
package exerr
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"runtime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func callername(skip int) string {
|
||||||
|
pc := make([]uintptr, 15)
|
||||||
|
n := runtime.Callers(skip+2, pc)
|
||||||
|
frames := runtime.CallersFrames(pc[:n])
|
||||||
|
frame, _ := frames.Next()
|
||||||
|
return fmt.Sprintf("%s:%d %s", frame.File, frame.Line, frame.Function)
|
||||||
|
}
|
@@ -1,16 +0,0 @@
|
|||||||
package ginext
|
|
||||||
|
|
||||||
type apiError struct {
|
|
||||||
ErrorCode string `json:"errorcode"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
FAPIErrorMessage *string `json:"fapiMessage,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type extAPIError struct {
|
|
||||||
ErrorCode string `json:"errorcode"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
FAPIErrorMessage *string `json:"fapiMessage,omitempty"`
|
|
||||||
|
|
||||||
RawError *string `json:"__error"`
|
|
||||||
Trace []string `json:"__trace"`
|
|
||||||
}
|
|
@@ -1,21 +0,0 @@
|
|||||||
package commonApiErr
|
|
||||||
|
|
||||||
type APIErrorCode struct {
|
|
||||||
HTTPStatusCode int
|
|
||||||
Key string
|
|
||||||
}
|
|
||||||
|
|
||||||
//goland:noinspection GoSnakeCaseUsage
|
|
||||||
var (
|
|
||||||
NotImplemented = APIErrorCode{500, "NOT_IMPLEMENTED"}
|
|
||||||
InternalError = APIErrorCode{500, "INTERNAL_ERROR"}
|
|
||||||
Panic = APIErrorCode{500, "PANIC"}
|
|
||||||
|
|
||||||
BindFailURI = APIErrorCode{400, "BINDFAIL_URI"}
|
|
||||||
BindFailQuery = APIErrorCode{400, "BINDFAIL_QUERY"}
|
|
||||||
BindFailJSON = APIErrorCode{400, "BINDFAIL_JSON"}
|
|
||||||
BindFailFormData = APIErrorCode{400, "BINDFAIL_FORMDATA"}
|
|
||||||
|
|
||||||
Unauthorized = APIErrorCode{401, "UNAUTHORIZED"}
|
|
||||||
AuthFailed = APIErrorCode{401, "AUTH_FAILED"}
|
|
||||||
)
|
|
@@ -12,11 +12,10 @@ type GinWrapper struct {
|
|||||||
|
|
||||||
allowCors bool
|
allowCors bool
|
||||||
ginDebug bool
|
ginDebug bool
|
||||||
returnRawErrors bool
|
|
||||||
requestTimeout time.Duration
|
requestTimeout time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewEngine(allowCors bool, ginDebug bool, returnRawErrors bool, timeout time.Duration) *GinWrapper {
|
func NewEngine(allowCors bool, ginDebug bool, timeout time.Duration) *GinWrapper {
|
||||||
engine := gin.New()
|
engine := gin.New()
|
||||||
|
|
||||||
wrapper := &GinWrapper{
|
wrapper := &GinWrapper{
|
||||||
@@ -24,7 +23,6 @@ func NewEngine(allowCors bool, ginDebug bool, returnRawErrors bool, timeout time
|
|||||||
SuppressGinLogs: false,
|
SuppressGinLogs: false,
|
||||||
allowCors: allowCors,
|
allowCors: allowCors,
|
||||||
ginDebug: ginDebug,
|
ginDebug: ginDebug,
|
||||||
returnRawErrors: returnRawErrors,
|
|
||||||
requestTimeout: timeout,
|
requestTimeout: timeout,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,10 +1,9 @@
|
|||||||
package ginext
|
package ginext
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/rs/zerolog/log"
|
"gogs.mikescher.com/BlackForestBytes/goext/exerr"
|
||||||
)
|
)
|
||||||
|
|
||||||
type WHandlerFunc func(PreContext) HTTPResponse
|
type WHandlerFunc func(PreContext) HTTPResponse
|
||||||
@@ -13,18 +12,20 @@ func Wrap(w *GinWrapper, fn WHandlerFunc) gin.HandlerFunc {
|
|||||||
|
|
||||||
return func(g *gin.Context) {
|
return func(g *gin.Context) {
|
||||||
|
|
||||||
g.Set("__returnRawErrors", w.returnRawErrors)
|
|
||||||
|
|
||||||
reqctx := g.Request.Context()
|
reqctx := g.Request.Context()
|
||||||
|
|
||||||
wrap, stackTrace, panicObj := callPanicSafe(fn, PreContext{wrapper: w, ginCtx: g})
|
wrap, stackTrace, panicObj := callPanicSafe(fn, PreContext{wrapper: w, ginCtx: g})
|
||||||
if panicObj != nil {
|
if panicObj != nil {
|
||||||
|
|
||||||
fmt.Printf("\n======== ======== STACKTRACE ======== ========\n%s\n======== ======== ======== ========\n\n", stackTrace)
|
fmt.Printf("\n======== ======== STACKTRACE ======== ========\n%s\n======== ======== ======== ========\n\n", stackTrace)
|
||||||
log.Error().
|
|
||||||
Interface("panicObj", panicObj).
|
err := exerr.
|
||||||
|
New(exerr.TypePanic, "Panic occured (in gin handler)").
|
||||||
|
Any("panicObj", panicObj).
|
||||||
Str("trace", stackTrace).
|
Str("trace", stackTrace).
|
||||||
Msg("Panic occured (in gin handler)")
|
Build()
|
||||||
wrap = APIError(g, commonApiErr.Panic, "A panic occured in the HTTP handler", errors.New(fmt.Sprintf("%+v", panicObj)))
|
|
||||||
|
wrap = APIError(g, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if g.Writer.Written() {
|
if g.Writer.Written() {
|
||||||
|
@@ -3,11 +3,8 @@ package ginext
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/rs/zerolog/log"
|
"gogs.mikescher.com/BlackForestBytes/goext/exerr"
|
||||||
json "gogs.mikescher.com/BlackForestBytes/goext/gojson"
|
json "gogs.mikescher.com/BlackForestBytes/goext/gojson"
|
||||||
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
|
||||||
"runtime/debug"
|
|
||||||
"strings"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type HTTPResponse interface {
|
type HTTPResponse interface {
|
||||||
@@ -74,6 +71,14 @@ func (j redirectHTTPResponse) Write(g *gin.Context) {
|
|||||||
g.Redirect(j.statusCode, j.url)
|
g.Redirect(j.statusCode, j.url)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type jsonAPIErrResponse struct {
|
||||||
|
err *exerr.ExErr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (j jsonAPIErrResponse) Write(g *gin.Context) {
|
||||||
|
j.err.Output(g)
|
||||||
|
}
|
||||||
|
|
||||||
func Status(sc int) HTTPResponse {
|
func Status(sc int) HTTPResponse {
|
||||||
return &emptyHTTPResponse{statusCode: sc}
|
return &emptyHTTPResponse{statusCode: sc}
|
||||||
}
|
}
|
||||||
@@ -102,52 +107,12 @@ func Redirect(sc int, newURL string) HTTPResponse {
|
|||||||
return &redirectHTTPResponse{statusCode: sc, url: newURL}
|
return &redirectHTTPResponse{statusCode: sc, url: newURL}
|
||||||
}
|
}
|
||||||
|
|
||||||
func APIError(g *gin.Context, errcode commonApiErr.APIErrorCode, msg string, e error) HTTPResponse {
|
func APIError(g *gin.Context, e error) HTTPResponse {
|
||||||
return createApiError(g, errcode, msg, e)
|
return &jsonAPIErrResponse{
|
||||||
|
err: exerr.FromError(e),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func NotImplemented(g *gin.Context) HTTPResponse {
|
func NotImplemented(g *gin.Context) HTTPResponse {
|
||||||
return createApiError(g, commonApiErr.NotImplemented, "", nil)
|
return APIError(g, exerr.New(exerr.TypeNotImplemented, "").Build())
|
||||||
}
|
|
||||||
|
|
||||||
func createApiError(g *gin.Context, errcode commonApiErr.APIErrorCode, msg string, e error) HTTPResponse {
|
|
||||||
reqUri := ""
|
|
||||||
if g != nil && g.Request != nil {
|
|
||||||
reqUri = g.Request.Method + " :: " + g.Request.RequestURI
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Error().
|
|
||||||
Str("errorcode.key", errcode.Key).
|
|
||||||
Int("errcode.status", errcode.HTTPStatusCode).
|
|
||||||
Str("uri", reqUri).
|
|
||||||
AnErr("err", e).
|
|
||||||
Stack().
|
|
||||||
Msg(msg)
|
|
||||||
|
|
||||||
var fapiMessage *string = nil
|
|
||||||
if v, ok := e.(interface{ FAPIMessage() string }); ok {
|
|
||||||
fapiMessage = langext.Ptr(v.FAPIMessage())
|
|
||||||
}
|
|
||||||
|
|
||||||
if g.GetBool("__returnRawErrors") {
|
|
||||||
return &jsonHTTPResponse{
|
|
||||||
statusCode: errcode.HTTPStatusCode,
|
|
||||||
data: extAPIError{
|
|
||||||
ErrorCode: errcode.Key,
|
|
||||||
Message: msg,
|
|
||||||
RawError: langext.Ptr(langext.Conditional(e == nil, "", fmt.Sprintf("%+v", e))),
|
|
||||||
FAPIErrorMessage: fapiMessage,
|
|
||||||
Trace: strings.Split(string(debug.Stack()), "\n"),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return &jsonHTTPResponse{
|
|
||||||
statusCode: errcode.HTTPStatusCode,
|
|
||||||
data: apiError{
|
|
||||||
ErrorCode: errcode.Key,
|
|
||||||
Message: msg,
|
|
||||||
FAPIErrorMessage: fapiMessage,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
1
go.mod
1
go.mod
@@ -5,6 +5,7 @@ go 1.19
|
|||||||
require (
|
require (
|
||||||
github.com/gin-gonic/gin v1.9.1
|
github.com/gin-gonic/gin v1.9.1
|
||||||
github.com/jmoiron/sqlx v1.3.5
|
github.com/jmoiron/sqlx v1.3.5
|
||||||
|
github.com/rs/xid v1.5.0
|
||||||
github.com/rs/zerolog v1.29.1
|
github.com/rs/zerolog v1.29.1
|
||||||
go.mongodb.org/mongo-driver v1.12.0
|
go.mongodb.org/mongo-driver v1.12.0
|
||||||
golang.org/x/crypto v0.11.0
|
golang.org/x/crypto v0.11.0
|
||||||
|
2
go.sum
2
go.sum
@@ -79,6 +79,8 @@ github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
|||||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/rs/xid v1.4.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
|
github.com/rs/xid v1.4.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
|
||||||
|
github.com/rs/xid v1.5.0 h1:mKX4bl4iPYJtEIxp6CYiUuLQ/8DYMoz0PUdtGgMFRVc=
|
||||||
|
github.com/rs/xid v1.5.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
|
||||||
github.com/rs/zerolog v1.29.0 h1:Zes4hju04hjbvkVkOhdl2HpZa+0PmVwigmo8XoORE5w=
|
github.com/rs/zerolog v1.29.0 h1:Zes4hju04hjbvkVkOhdl2HpZa+0PmVwigmo8XoORE5w=
|
||||||
github.com/rs/zerolog v1.29.0/go.mod h1:NILgTygv/Uej1ra5XxGf82ZFSLk58MFGAUS2o6usyD0=
|
github.com/rs/zerolog v1.29.0/go.mod h1:NILgTygv/Uej1ra5XxGf82ZFSLk58MFGAUS2o6usyD0=
|
||||||
github.com/rs/zerolog v1.29.1 h1:cO+d60CHkknCbvzEWxP0S9K6KqyTjrCNUy1LdQLCGPc=
|
github.com/rs/zerolog v1.29.1 h1:cO+d60CHkknCbvzEWxP0S9K6KqyTjrCNUy1LdQLCGPc=
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
package goext
|
package goext
|
||||||
|
|
||||||
const GoextVersion = "0.0.184"
|
const GoextVersion = "0.0.189"
|
||||||
|
|
||||||
const GoextVersionTimestamp = "2023-07-19T19:29:59+0200"
|
const GoextVersionTimestamp = "2023-07-24T11:11:15+0200"
|
||||||
|
@@ -459,3 +459,11 @@ func ArrExcept[T comparable](arr []T, needles ...T) []T {
|
|||||||
}
|
}
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ArrayToInterface[T any](t []T) []interface{} {
|
||||||
|
res := make([]interface{}, 0, len(t))
|
||||||
|
for i, _ := range t {
|
||||||
|
res = append(res, t[i])
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
@@ -3,6 +3,8 @@ package wmo
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"go.mongodb.org/mongo-driver/bson"
|
"go.mongodb.org/mongo-driver/bson"
|
||||||
|
"go.mongodb.org/mongo-driver/mongo"
|
||||||
|
"gogs.mikescher.com/BlackForestBytes/goext/langext"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (c *Coll[TData]) InsertOne(ctx context.Context, valueIn TData) (TData, error) {
|
func (c *Coll[TData]) InsertOne(ctx context.Context, valueIn TData) (TData, error) {
|
||||||
@@ -15,3 +17,7 @@ func (c *Coll[TData]) InsertOne(ctx context.Context, valueIn TData) (TData, erro
|
|||||||
|
|
||||||
return c.decodeSingle(ctx, mongoRes)
|
return c.decodeSingle(ctx, mongoRes)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Coll[TData]) InsertMany(ctx context.Context, valueIn []TData) (*mongo.InsertManyResult, error) {
|
||||||
|
return c.coll.InsertMany(ctx, langext.ArrayToInterface(valueIn))
|
||||||
|
}
|
||||||
|
Reference in New Issue
Block a user