Compare commits
	
		
			3 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 16c66ee28c | |||
| 2e6ca48d22 | |||
| b1d6509294 | 
| @@ -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") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										171
									
								
								exerr/exerr.go
									
									
									
									
									
								
							
							
						
						
									
										171
									
								
								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 { | func (ee *ExErr) Unwrap() error { | ||||||
|  | 	return ee.OriginalError | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ee ExErr) Is(err error) bool { | 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 | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// fallback to self | ||||||
|  | 	return ee.Type | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 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.186" | const GoextVersion = "0.0.189" | ||||||
|  |  | ||||||
| const GoextVersionTimestamp = "2023-07-24T09:13:19+0200" | const GoextVersionTimestamp = "2023-07-24T11:11:15+0200" | ||||||
|   | |||||||
| @@ -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) { | ||||||
| @@ -16,13 +18,6 @@ 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) (TData, error) { | func (c *Coll[TData]) InsertMany(ctx context.Context, valueIn []TData) (*mongo.InsertManyResult, error) { | ||||||
| 	insRes, err := c.coll.InsertMany(ctx, langext.TovalueIn) | 	return c.coll.InsertMany(ctx, langext.ArrayToInterface(valueIn)) | ||||||
| 	if err != nil { |  | ||||||
| 		return *new(TData), err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	mongoRes := c.coll.FindOne(ctx, bson.M{"_id": insRes.InsertedID}) |  | ||||||
|  |  | ||||||
| 	return c.decodeSingle(ctx, mongoRes) |  | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user