Compare commits
	
		
			56 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 202afc9068 | |||
| 56094b3cb6 | |||
| 0da098e9f9 | |||
| f0881c9fd6 | |||
| 029b408749 | |||
| 84b2be3169 | |||
| c872cecc67 | |||
| 99cd92729e | |||
| ac416f7b69 | |||
| e10140e143 | |||
| e165f0f62f | |||
| 655d4daad9 | |||
| 87a004e577 | |||
| 376c6cab50 | |||
| 4a3f25baa0 | |||
| aa33bc8df3 | |||
| 96b3718375 | |||
| 5f9b55933b | |||
| 74d42637e7 | |||
| 0c05bcf29b | |||
| 9136143f2f | |||
| 2f1b784dc2 | |||
| 190584e0e6 | |||
| b7003b9ec9 | |||
| 4f871271e8 | |||
| 91f4793678 | |||
| 3b30bb049e | |||
| f0c5b36ea9 | |||
| 647ec64c3b | |||
| b5f9b6b638 | |||
| c7949febf2 | |||
| 15a4b2a713 | |||
| 493c6ebae8 | |||
| fb847b03af | |||
| f826633e6e | |||
| edeae23bf1 | |||
| a038b86147 | |||
| ede0b99d3a | |||
| d04ce18eb0 | |||
| 8ae9a0f107 | |||
| a259bb6dbc | |||
| adf32568ee | |||
| 0cfa159cb1 | |||
| 0ead99608a | |||
| 7fe3e66cad | |||
| a73d7d1654 | |||
| bbd7a7bc2c | |||
| f5151eb214 | |||
| eefb9ac9f5 | |||
| 468a7d212d | |||
| a4def75d06 | |||
| 16c66ee28c | |||
| 2e6ca48d22 | |||
| b1d6509294 | |||
| e909d656d9 | |||
| 0971f60c30 | 
| @@ -31,13 +31,13 @@ type EnumDef struct { | |||||||
| 	Values       []EnumDefVal | 	Values       []EnumDefVal | ||||||
| } | } | ||||||
|  |  | ||||||
| var rexPackage = rext.W(regexp.MustCompile("^package\\s+(?P<name>[A-Za-z0-9_]+)\\s*$")) | var rexPackage = rext.W(regexp.MustCompile(`^package\s+(?P<name>[A-Za-z0-9_]+)\s*$`)) | ||||||
|  |  | ||||||
| var rexEnumDef = rext.W(regexp.MustCompile("^\\s*type\\s+(?P<name>[A-Za-z0-9_]+)\\s+(?P<type>[A-Za-z0-9_]+)\\s*//\\s*(@enum:type).*$")) | var rexEnumDef = rext.W(regexp.MustCompile(`^\s*type\s+(?P<name>[A-Za-z0-9_]+)\s+(?P<type>[A-Za-z0-9_]+)\s*//\s*(@enum:type).*$`)) | ||||||
|  |  | ||||||
| var rexValueDef = rext.W(regexp.MustCompile("^\\s*(?P<name>[A-Za-z0-9_]+)\\s+(?P<type>[A-Za-z0-9_]+)\\s*=\\s*(?P<value>(\"[A-Za-z0-9_:]+\"|[0-9]+))\\s*(//(?P<descr>.*))?.*$")) | var rexValueDef = rext.W(regexp.MustCompile(`^\s*(?P<name>[A-Za-z0-9_]+)\s+(?P<type>[A-Za-z0-9_]+)\s*=\s*(?P<value>("[A-Za-z0-9_:]+"|[0-9]+))\s*(//(?P<descr>.*))?.*$`)) | ||||||
|  |  | ||||||
| var rexChecksumConst = rext.W(regexp.MustCompile("const ChecksumGenerator = \"(?P<cs>[A-Za-z0-9_]*)\"")) | var rexChecksumConst = rext.W(regexp.MustCompile(`const ChecksumGenerator = "(?P<cs>[A-Za-z0-9_]*)"`)) | ||||||
|  |  | ||||||
| func GenerateEnumSpecs(sourceDir string, destFile string) error { | func GenerateEnumSpecs(sourceDir string, destFile string) error { | ||||||
|  |  | ||||||
| @@ -209,36 +209,10 @@ func fmtOutput(cs string, enums []EnumDef, pkgname string) string { | |||||||
| 	str += "\n" | 	str += "\n" | ||||||
|  |  | ||||||
| 	str += "import \"gogs.mikescher.com/BlackForestBytes/goext/langext\"" + "\n" | 	str += "import \"gogs.mikescher.com/BlackForestBytes/goext/langext\"" + "\n" | ||||||
|  | 	str += "import \"gogs.mikescher.com/BlackForestBytes/goext/enums\"" + "\n" | ||||||
| 	str += "\n" | 	str += "\n" | ||||||
|  |  | ||||||
| 	str += "const ChecksumGenerator = \"" + cs + "\"" + "\n" | 	str += "const ChecksumGenerator = \"" + cs + "\" // GoExtVersion: " + goext.GoextVersion + "\n" | ||||||
| 	str += "\n" |  | ||||||
|  |  | ||||||
| 	str += "type Enum interface {" + "\n" |  | ||||||
| 	str += "    Valid() bool" + "\n" |  | ||||||
| 	str += "    ValuesAny() []any" + "\n" |  | ||||||
| 	str += "    ValuesMeta() []EnumMetaValue" + "\n" |  | ||||||
| 	str += "    VarName() string" + "\n" |  | ||||||
| 	str += "}" + "\n" |  | ||||||
| 	str += "" + "\n" |  | ||||||
|  |  | ||||||
| 	str += "type StringEnum interface {" + "\n" |  | ||||||
| 	str += "    Enum" + "\n" |  | ||||||
| 	str += "    String() string" + "\n" |  | ||||||
| 	str += "}" + "\n" |  | ||||||
| 	str += "" + "\n" |  | ||||||
|  |  | ||||||
| 	str += "type DescriptionEnum interface {" + "\n" |  | ||||||
| 	str += "    Enum" + "\n" |  | ||||||
| 	str += "    Description() string" + "\n" |  | ||||||
| 	str += "}" + "\n" |  | ||||||
| 	str += "\n" |  | ||||||
|  |  | ||||||
| 	str += "type EnumMetaValue struct {" + "\n" |  | ||||||
| 	str += "    VarName     string  `json:\"varName\"`" + "\n" |  | ||||||
| 	str += "    Value       any     `json:\"value\"`" + "\n" |  | ||||||
| 	str += "    Description *string `json:\"description\"`" + "\n" |  | ||||||
| 	str += "}" + "\n" |  | ||||||
| 	str += "\n" | 	str += "\n" | ||||||
|  |  | ||||||
| 	for _, enumdef := range enums { | 	for _, enumdef := range enums { | ||||||
| @@ -292,16 +266,8 @@ func fmtOutput(cs string, enums []EnumDef, pkgname string) string { | |||||||
| 		str += "}" + "\n" | 		str += "}" + "\n" | ||||||
| 		str += "" + "\n" | 		str += "" + "\n" | ||||||
|  |  | ||||||
| 		str += "func (e " + enumdef.EnumTypeName + ") ValuesMeta() []EnumMetaValue {" + "\n" | 		str += "func (e " + enumdef.EnumTypeName + ") ValuesMeta() []enums.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 +296,15 @@ func fmtOutput(cs string, enums []EnumDef, pkgname string) string { | |||||||
| 		str += "}" + "\n" | 		str += "}" + "\n" | ||||||
| 		str += "" + "\n" | 		str += "" + "\n" | ||||||
|  |  | ||||||
|  | 		str += "func (e " + enumdef.EnumTypeName + ") Meta() enums.EnumMetaValue {" + "\n" | ||||||
|  | 		if hasDescr { | ||||||
|  | 			str += "    return enums.EnumMetaValue{VarName: e.VarName(), Value: e, Description: langext.Ptr(e.Description())}" | ||||||
|  | 		} else { | ||||||
|  | 			str += "    return enums.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" | ||||||
| @@ -345,14 +320,10 @@ func fmtOutput(cs string, enums []EnumDef, pkgname string) string { | |||||||
| 		str += "}" + "\n" | 		str += "}" + "\n" | ||||||
| 		str += "" + "\n" | 		str += "" + "\n" | ||||||
|  |  | ||||||
| 		str += "func " + enumdef.EnumTypeName + "ValuesMeta() []EnumMetaValue {" + "\n" | 		str += "func " + enumdef.EnumTypeName + "ValuesMeta() []enums.EnumMetaValue {" + "\n" | ||||||
| 		str += "    return []EnumMetaValue{" + "\n" | 		str += "    return []enums.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" | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ type CommandRunner struct { | |||||||
| 	listener         []CommandListener | 	listener         []CommandListener | ||||||
| 	enforceExitCodes *[]int | 	enforceExitCodes *[]int | ||||||
| 	enforceNoTimeout bool | 	enforceNoTimeout bool | ||||||
|  | 	enforceNoStderr  bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func Runner(program string) *CommandRunner { | func Runner(program string) *CommandRunner { | ||||||
| @@ -25,6 +26,7 @@ func Runner(program string) *CommandRunner { | |||||||
| 		listener:         make([]CommandListener, 0), | 		listener:         make([]CommandListener, 0), | ||||||
| 		enforceExitCodes: nil, | 		enforceExitCodes: nil, | ||||||
| 		enforceNoTimeout: false, | 		enforceNoTimeout: false, | ||||||
|  | 		enforceNoStderr:  false, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -73,6 +75,11 @@ func (r *CommandRunner) FailOnTimeout() *CommandRunner { | |||||||
| 	return r | 	return r | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (r *CommandRunner) FailOnStderr() *CommandRunner { | ||||||
|  | 	r.enforceNoStderr = true | ||||||
|  | 	return r | ||||||
|  | } | ||||||
|  |  | ||||||
| func (r *CommandRunner) Listen(lstr CommandListener) *CommandRunner { | func (r *CommandRunner) Listen(lstr CommandListener) *CommandRunner { | ||||||
| 	r.listener = append(r.listener, lstr) | 	r.listener = append(r.listener, lstr) | ||||||
| 	return r | 	return r | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ import ( | |||||||
|  |  | ||||||
| var ErrExitCode = errors.New("process exited with an unexpected exitcode") | var ErrExitCode = errors.New("process exited with an unexpected exitcode") | ||||||
| var ErrTimeout = errors.New("process did not exit after the specified timeout") | var ErrTimeout = errors.New("process did not exit after the specified timeout") | ||||||
|  | var ErrStderrPrint = errors.New("process did print to stderr stream") | ||||||
|  |  | ||||||
| type CommandResult struct { | type CommandResult struct { | ||||||
| 	StdOut          string | 	StdOut          string | ||||||
| @@ -53,12 +54,27 @@ func run(opt CommandRunner) (CommandResult, error) { | |||||||
| 		err         error | 		err         error | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	stderrFailChan := make(chan bool) | ||||||
|  |  | ||||||
| 	outputChan := make(chan resultObj) | 	outputChan := make(chan resultObj) | ||||||
| 	go func() { | 	go func() { | ||||||
| 		// we need to first fully read the pipes and then call Wait | 		// we need to first fully read the pipes and then call Wait | ||||||
| 		// see https://pkg.go.dev/os/exec#Cmd.StdoutPipe | 		// see https://pkg.go.dev/os/exec#Cmd.StdoutPipe | ||||||
|  |  | ||||||
| 		stdout, stderr, stdcombined, err := preader.Read(opt.listener) | 		listener := make([]CommandListener, 0) | ||||||
|  | 		listener = append(listener, opt.listener...) | ||||||
|  |  | ||||||
|  | 		if opt.enforceNoStderr { | ||||||
|  | 			listener = append(listener, genericCommandListener{ | ||||||
|  | 				_readRawStderr: langext.Ptr(func(v []byte) { | ||||||
|  | 					if len(v) > 0 { | ||||||
|  | 						stderrFailChan <- true | ||||||
|  | 					} | ||||||
|  | 				}), | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		stdout, stderr, stdcombined, err := preader.Read(listener) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			outputChan <- resultObj{stdout, stderr, stdcombined, err} | 			outputChan <- resultObj{stdout, stderr, stdcombined, err} | ||||||
| 			_ = cmd.Process.Kill() | 			_ = cmd.Process.Kill() | ||||||
| @@ -115,6 +131,34 @@ func run(opt CommandRunner) (CommandResult, error) { | |||||||
| 			return res, nil | 			return res, nil | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 	case <-stderrFailChan: | ||||||
|  | 		_ = cmd.Process.Kill() | ||||||
|  | 		for _, lstr := range opt.listener { | ||||||
|  | 			lstr.Timeout() | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if fallback, ok := syncext.ReadChannelWithTimeout(outputChan, 32*time.Millisecond); ok { | ||||||
|  | 			// most of the time the cmd.Process.Kill() should also have finished the pipereader | ||||||
|  | 			// and we can at least return the already collected stdout, stderr, etc | ||||||
|  | 			res := CommandResult{ | ||||||
|  | 				StdOut:          fallback.stdout, | ||||||
|  | 				StdErr:          fallback.stderr, | ||||||
|  | 				StdCombined:     fallback.stdcombined, | ||||||
|  | 				ExitCode:        -1, | ||||||
|  | 				CommandTimedOut: false, | ||||||
|  | 			} | ||||||
|  | 			return res, ErrStderrPrint | ||||||
|  | 		} else { | ||||||
|  | 			res := CommandResult{ | ||||||
|  | 				StdOut:          "", | ||||||
|  | 				StdErr:          "", | ||||||
|  | 				StdCombined:     "", | ||||||
|  | 				ExitCode:        -1, | ||||||
|  | 				CommandTimedOut: false, | ||||||
|  | 			} | ||||||
|  | 			return res, ErrStderrPrint | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 	case outobj := <-outputChan: | 	case outobj := <-outputChan: | ||||||
| 		if exiterr, ok := outobj.err.(*exec.ExitError); ok { | 		if exiterr, ok := outobj.err.(*exec.ExitError); ok { | ||||||
| 			excode := exiterr.ExitCode() | 			excode := exiterr.ExitCode() | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| package cmdext | package cmdext | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"testing" | 	"testing" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -289,16 +290,40 @@ func TestLongStdout(t *testing.T) { | |||||||
| func TestFailOnTimeout(t *testing.T) { | func TestFailOnTimeout(t *testing.T) { | ||||||
|  |  | ||||||
| 	_, err := Runner("sleep").Arg("2").Timeout(200 * time.Millisecond).FailOnTimeout().Run() | 	_, err := Runner("sleep").Arg("2").Timeout(200 * time.Millisecond).FailOnTimeout().Run() | ||||||
| 	if err != ErrTimeout { | 	if !errors.Is(err, ErrTimeout) { | ||||||
| 		t.Errorf("wrong err := %v", err) | 		t.Errorf("wrong err := %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestFailOnStderr(t *testing.T) { | ||||||
|  |  | ||||||
|  | 	res1, err := Runner("python").Arg("-c").Arg("import sys; print(\"error\", file=sys.stderr, end='')").FailOnStderr().Run() | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Errorf("no err") | ||||||
|  | 	} | ||||||
|  | 	if res1.CommandTimedOut { | ||||||
|  | 		t.Errorf("Timeout") | ||||||
|  | 	} | ||||||
|  | 	if res1.ExitCode != -1 { | ||||||
|  | 		t.Errorf("res1.ExitCode == %v", res1.ExitCode) | ||||||
|  | 	} | ||||||
|  | 	if res1.StdErr != "error" { | ||||||
|  | 		t.Errorf("res1.StdErr == '%v'", res1.StdErr) | ||||||
|  | 	} | ||||||
|  | 	if res1.StdOut != "" { | ||||||
|  | 		t.Errorf("res1.StdOut == '%v'", res1.StdOut) | ||||||
|  | 	} | ||||||
|  | 	if res1.StdCombined != "error\n" { | ||||||
|  | 		t.Errorf("res1.StdCombined == '%v'", res1.StdCombined) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
| func TestFailOnExitcode(t *testing.T) { | func TestFailOnExitcode(t *testing.T) { | ||||||
|  |  | ||||||
| 	_, err := Runner("false").Timeout(200 * time.Millisecond).FailOnExitCode().Run() | 	_, err := Runner("false").Timeout(200 * time.Millisecond).FailOnExitCode().Run() | ||||||
| 	if err != ErrExitCode { | 	if !errors.Is(err, ErrExitCode) { | ||||||
| 		t.Errorf("wrong err := %v", err) | 		t.Errorf("wrong err := %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,6 +7,9 @@ type SyncSet[TData comparable] struct { | |||||||
| 	lock sync.Mutex | 	lock sync.Mutex | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Add adds `value` to the set | ||||||
|  | // returns true  if the value was actually inserted | ||||||
|  | // returns false if the value already existed | ||||||
| func (s *SyncSet[TData]) Add(value TData) bool { | func (s *SyncSet[TData]) Add(value TData) bool { | ||||||
| 	s.lock.Lock() | 	s.lock.Lock() | ||||||
| 	defer s.lock.Unlock() | 	defer s.lock.Unlock() | ||||||
| @@ -15,10 +18,10 @@ func (s *SyncSet[TData]) Add(value TData) bool { | |||||||
| 		s.data = make(map[TData]bool) | 		s.data = make(map[TData]bool) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	_, ok := s.data[value] | 	_, existsInPreState := s.data[value] | ||||||
| 	s.data[value] = true | 	s.data[value] = true | ||||||
|  |  | ||||||
| 	return !ok | 	return !existsInPreState | ||||||
| } | } | ||||||
|  |  | ||||||
| func (s *SyncSet[TData]) AddAll(values []TData) { | func (s *SyncSet[TData]) AddAll(values []TData) { | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								enums/enum.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								enums/enum.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | package enums | ||||||
|  |  | ||||||
|  | type Enum interface { | ||||||
|  | 	Valid() bool | ||||||
|  | 	ValuesAny() []any | ||||||
|  | 	ValuesMeta() []EnumMetaValue | ||||||
|  | 	VarName() string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type StringEnum interface { | ||||||
|  | 	Enum | ||||||
|  | 	String() string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type DescriptionEnum interface { | ||||||
|  | 	Enum | ||||||
|  | 	Description() string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type EnumMetaValue struct { | ||||||
|  | 	VarName     string  `json:"varName"` | ||||||
|  | 	Value       any     `json:"value"` | ||||||
|  | 	Description *string `json:"description"` | ||||||
|  | } | ||||||
							
								
								
									
										138
									
								
								exerr/builder.go
									
									
									
									
									
								
							
							
						
						
									
										138
									
								
								exerr/builder.go
									
									
									
									
									
								
							| @@ -2,10 +2,14 @@ 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/enums" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
| @@ -17,21 +21,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 +68,45 @@ 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 err == nil { | ||||||
|  | 		return &Builder{errorData: newExErr(CatSystem, TypeInternal, msg)} // prevent NPE if we call Wrap with err==nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	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 +131,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 +150,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 +169,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 +179,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 | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -262,13 +274,25 @@ func (b *Builder) Any(key string, val any) *Builder { | |||||||
| 	return b.addMeta(key, MDTAny, newAnyWrap(val)) | 	return b.addMeta(key, MDTAny, newAnyWrap(val)) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *Builder) Stringer(key string, val fmt.Stringer) *Builder { | ||||||
|  | 	if val == nil { | ||||||
|  | 		return b.addMeta(key, MDTString, "(!nil)") | ||||||
|  | 	} else { | ||||||
|  | 		return b.addMeta(key, MDTString, val.String()) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Builder) Enum(key string, val enums.Enum) *Builder { | ||||||
|  | 	return b.addMeta(key, MDTEnum, newEnumWrap(val)) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (b *Builder) Stack() *Builder { | func (b *Builder) Stack() *Builder { | ||||||
| 	return b.addMeta("@Stack", MDTString, string(debug.Stack())) | 	return b.addMeta("@Stack", MDTString, string(debug.Stack())) | ||||||
| } | } | ||||||
|  |  | ||||||
| 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 | ||||||
| } | } | ||||||
| @@ -282,7 +306,7 @@ func (b *Builder) GinReq(ctx context.Context, g *gin.Context, req *http.Request) | |||||||
| 	} | 	} | ||||||
| 	b.Str("gin.method", req.Method) | 	b.Str("gin.method", req.Method) | ||||||
| 	b.Str("gin.path", g.FullPath()) | 	b.Str("gin.path", g.FullPath()) | ||||||
| 	b.Str("gin.header", formatHeader(g.Request.Header)) | 	b.Strs("gin.header", extractHeader(g.Request.Header)) | ||||||
| 	if req.URL != nil { | 	if req.URL != nil { | ||||||
| 		b.Str("gin.url", req.URL.String()) | 		b.Str("gin.url", req.URL.String()) | ||||||
| 	} | 	} | ||||||
| @@ -298,8 +322,10 @@ func (b *Builder) GinReq(ctx context.Context, g *gin.Context, req *http.Request) | |||||||
| 	if ctxVal := g.GetString("reqid"); ctxVal != "" { | 	if ctxVal := g.GetString("reqid"); ctxVal != "" { | ||||||
| 		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 { | ||||||
| 		if brc, ok := req.Body.(langext.BufferedReadCloser); ok { |  | ||||||
|  | 		if req.Header.Get("Content-Type") == "application/json" { | ||||||
|  | 			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 | ||||||
| @@ -310,12 +336,26 @@ func (b *Builder) GinReq(ctx context.Context, g *gin.Context, req *http.Request) | |||||||
| 							b.Bytes("gin.body", bin) | 							b.Bytes("gin.body", bin) | ||||||
| 						} | 						} | ||||||
| 					} else { | 					} else { | ||||||
| 					b.Str("gin.body", fmt.Sprintf("[[%v bytes]]", len(bin))) | 						b.Str("gin.body", fmt.Sprintf("[[%v bytes | %s]]", len(bin), req.Header.Get("Content-Type"))) | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		if req.Header.Get("Content-Type") == "multipart/form-data" || req.Header.Get("Content-Type") == "x-www-form-urlencoded" { | ||||||
|  | 			if brc, ok := req.Body.(dataext.BufferedReadCloser); ok { | ||||||
|  | 				if bin, err := brc.BufferedAll(); err == nil { | ||||||
|  | 					if len(bin) < 16*1024 { | ||||||
|  | 						b.Bytes("gin.body", bin) | ||||||
|  | 					} else { | ||||||
|  | 						b.Str("gin.body", fmt.Sprintf("[[%v bytes | %s]]", len(bin), req.Header.Get("Content-Type"))) | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	b.containsGinData = true | 	b.containsGinData = true | ||||||
| 	return b | 	return b | ||||||
| } | } | ||||||
| @@ -343,18 +383,36 @@ func formatHeader(header map[string][]string) string { | |||||||
| 	return r | 	return r | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func extractHeader(header map[string][]string) []string { | ||||||
|  | 	r := make([]string, 0, len(header)) | ||||||
|  | 	for k, v := range header { | ||||||
|  | 		for _, hval := range v { | ||||||
|  | 			value := hval | ||||||
|  | 			value = strings.ReplaceAll(value, "\n", "\\n") | ||||||
|  | 			value = strings.ReplaceAll(value, "\r", "\\r") | ||||||
|  | 			value = strings.ReplaceAll(value, "\t", "\\t") | ||||||
|  | 			r = append(r, k+": "+value) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return r | ||||||
|  | } | ||||||
|  |  | ||||||
| // ---------------------------------------------------------------------------- | // ---------------------------------------------------------------------------- | ||||||
|  |  | ||||||
| // 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 +424,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 +438,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 +465,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() |  | ||||||
| } |  | ||||||
|   | |||||||
							
								
								
									
										204
									
								
								exerr/constructor.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								exerr/constructor.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,204 @@ | |||||||
|  | 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), | ||||||
|  | 		WrappedErr:     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: "", | ||||||
|  | 		WrappedErr:     nil, | ||||||
|  | 		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: "", | ||||||
|  | 		WrappedErr:     nil, | ||||||
|  | 		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,10 @@ | |||||||
| package exerr | package exerr | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | ) | ||||||
|  |  | ||||||
| type ErrorCategory struct{ Category string } | type ErrorCategory struct{ Category string } | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| @@ -9,6 +14,7 @@ var ( | |||||||
| 	CatForeign = ErrorCategory{"Foreign"} // A foreign error that some component threw (e.g. an unknown mongodb error), happens if we call Wrap(..) on an non-bmerror value | 	CatForeign = ErrorCategory{"Foreign"} // A foreign error that some component threw (e.g. an unknown mongodb error), happens if we call Wrap(..) on an non-bmerror value | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | //goland:noinspection GoUnusedGlobalVariable | ||||||
| var AllCategories = []ErrorCategory{CatWrap, CatSystem, CatUser, CatForeign} | var AllCategories = []ErrorCategory{CatWrap, CatSystem, CatUser, CatForeign} | ||||||
|  |  | ||||||
| type ErrorSeverity struct{ Severity string } | type ErrorSeverity struct{ Severity string } | ||||||
| @@ -22,11 +28,49 @@ var ( | |||||||
| 	SevFatal = ErrorSeverity{"Fatal"} | 	SevFatal = ErrorSeverity{"Fatal"} | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | //goland:noinspection GoUnusedGlobalVariable | ||||||
| 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 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | //goland:noinspection GoUnusedGlobalVariable | ||||||
| var ( | var ( | ||||||
| 	TypeInternal = ErrorType{"Internal"} | 	TypeInternal       = NewType("INTERNAL_ERROR", langext.Ptr(500)) | ||||||
|  | 	TypePanic          = NewType("PANIC", langext.Ptr(500)) | ||||||
|  | 	TypeNotImplemented = NewType("NOT_IMPLEMENTED", langext.Ptr(500)) | ||||||
|  |  | ||||||
|  | 	TypeWrap = NewType("Wrap", nil) | ||||||
|  |  | ||||||
|  | 	TypeBindFailURI      = NewType("BINDFAIL_URI", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailQuery    = NewType("BINDFAIL_QUERY", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailJSON     = NewType("BINDFAIL_JSON", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailFormData = NewType("BINDFAIL_FORMDATA", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailHeader   = NewType("BINDFAIL_HEADER", langext.Ptr(400)) | ||||||
|  |  | ||||||
|  | 	TypeUnauthorized = NewType("UNAUTHORIZED", langext.Ptr(401)) | ||||||
|  | 	TypeAuthFailed   = NewType("AUTH_FAILED", langext.Ptr(401)) | ||||||
|  |  | ||||||
| 	// other values come from pkgconfig | 	// other values come from pkgconfig | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | var registeredTypes = dataext.SyncSet[string]{} | ||||||
|  |  | ||||||
|  | func NewType(key string, defStatusCode *int) ErrorType { | ||||||
|  | 	insertOkay := registeredTypes.Add(key) | ||||||
|  | 	if !insertOkay { | ||||||
|  | 		panic("Cannot register same ErrType ('" + key + "') more than once") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return ErrorType{key, defStatusCode} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type LogPrintLevel string | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	LogPrintFull     LogPrintLevel = "Full" | ||||||
|  | 	LogPrintOverview LogPrintLevel = "Overview" | ||||||
|  | 	LogPrintShort    LogPrintLevel = "Short" | ||||||
|  | ) | ||||||
|   | |||||||
| @@ -1,23 +1,40 @@ | |||||||
| 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() | ||||||
|  | 	IncludeMetaInGinOutput bool                                             // Log meta fields ( from e.g. `.Str(key, val).Build()` ) to gin in err.Output() | ||||||
|  | 	ExtendGinOutput        func(err *ExErr, json map[string]any)            // (Optionally) extend the gin output with more fields | ||||||
|  | 	ExtendGinDataOutput    func(err *ExErr, depth int, 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 | ||||||
|  | 	IncludeMetaInGinOutput bool | ||||||
|  | 	ExtendGinOutput        func(err *ExErr, json map[string]any) | ||||||
|  | 	ExtendGinDataOutput    func(err *ExErr, depth int, 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, | ||||||
|  | 	IncludeMetaInGinOutput: true, | ||||||
|  | 	ExtendGinOutput:        func(err *ExErr, json map[string]any) {}, | ||||||
|  | 	ExtendGinDataOutput:    func(err *ExErr, depth int, json map[string]any) {}, | ||||||
| } | } | ||||||
|  |  | ||||||
| // Init initializes the exerr packages | // Init initializes the exerr packages | ||||||
| @@ -28,23 +45,36 @@ func Init(cfg ErrorPackageConfigInit) { | |||||||
| 		panic("Cannot re-init error package") | 		panic("Cannot re-init error package") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	types := pkgconfig.Types | 	ego := func(err *ExErr, json map[string]any) {} | ||||||
|  | 	egdo := func(err *ExErr, depth int, json map[string]any) {} | ||||||
|  |  | ||||||
| 	fnAddType := func(v string) ErrorType { | 	if cfg.ExtendGinOutput != nil { | ||||||
| 		et := ErrorType{v} | 		ego = cfg.ExtendGinOutput | ||||||
| 		types = append(types, et) |  | ||||||
| 		return et |  | ||||||
| 	} | 	} | ||||||
|  | 	if cfg.ExtendGinDataOutput != nil { | ||||||
| 	if cfg.InitTypes != nil { | 		egdo = cfg.ExtendGinDataOutput | ||||||
| 		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, | ||||||
|  | 		IncludeMetaInGinOutput: cfg.IncludeMetaInGinOutput, | ||||||
|  | 		ExtendGinOutput:        ego, | ||||||
|  | 		ExtendGinDataOutput:    egdo, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	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") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										277
									
								
								exerr/exerr.go
									
									
									
									
									
								
							
							
						
						
									
										277
									
								
								exerr/exerr.go
									
									
									
									
									
								
							| @@ -1,6 +1,11 @@ | |||||||
| package exerr | package exerr | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"github.com/rs/xid" | ||||||
|  | 	"github.com/rs/zerolog" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"reflect" | ||||||
|  | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -12,22 +17,282 @@ 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"` | ||||||
|  | 	WrappedErr     any    `json:"-"` | ||||||
| 	Caller         string `json:"caller"` | 	Caller         string `json:"caller"` | ||||||
|  |  | ||||||
| 	OriginalError *ExErr | 	OriginalError *ExErr `json:"originalError"` | ||||||
|  |  | ||||||
| 	Meta MetaMap `json:"meta"` | 	Meta MetaMap `json:"meta"` | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ee ExErr) Error() string { | func (ee *ExErr) Error() string { | ||||||
|  | 	return ee.Message | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Unwrap must be implemented so that some error.XXX methods work | ||||||
|  | func (ee *ExErr) Unwrap() error { | ||||||
|  | 	if ee.OriginalError == nil { | ||||||
|  | 		return nil // this is neccessary - otherwise we return a wrapped nil and the `x == nil` comparison fails (= panic in errors.Is and other failures) | ||||||
|  | 	} | ||||||
|  | 	return ee.OriginalError | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Is must be implemented so that error.Is(x) works | ||||||
|  | func (ee *ExErr) Is(e error) bool { | ||||||
|  | 	return IsFrom(ee, e) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // As must be implemented so that error.As(x) works | ||||||
|  | // | ||||||
|  | //goland:noinspection GoTypeAssertionOnErrors | ||||||
|  | func (ee *ExErr) As(target any) bool { | ||||||
|  | 	if dstErr, ok := target.(*ExErr); ok { | ||||||
|  |  | ||||||
|  | 		if dst0, ok := ee.contains(dstErr); ok { | ||||||
|  | 			dstErr = dst0 | ||||||
|  | 			return true | ||||||
|  | 		} else { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	} else { | ||||||
|  |  | ||||||
|  | 		val := reflect.ValueOf(target) | ||||||
|  |  | ||||||
|  | 		typStr := val.Type().Elem().String() | ||||||
|  |  | ||||||
|  | 		for curr := ee; curr != nil; curr = curr.OriginalError { | ||||||
|  | 			if curr.Category == CatForeign && curr.WrappedErrType == typStr && curr.WrappedErr != nil { | ||||||
|  | 				val.Elem().Set(reflect.ValueOf(curr.WrappedErr)) | ||||||
|  | 				return true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	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 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // RecursiveMeta searches (top-down) for teh first error that has a meta value with teh specified key | ||||||
|  | // and returns its value (or nil) | ||||||
|  | func (ee *ExErr) RecursiveMeta(key string) *MetaValue { | ||||||
|  | 	for curr := ee; curr != nil; curr = curr.OriginalError { | ||||||
|  | 		if metaval, ok := curr.Meta[key]; ok { | ||||||
|  | 			return langext.Ptr(metaval) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Depth returns the depth of recursively contained errors | ||||||
|  | func (ee *ExErr) Depth() int { | ||||||
|  | 	if ee.OriginalError == nil { | ||||||
|  | 		return 1 | ||||||
|  | 	} else { | ||||||
|  | 		return ee.OriginalError.Depth() + 1 | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // contains test if the supplied error is contained in this error (anywhere in the chain) | ||||||
|  | func (ee *ExErr) contains(original *ExErr) (*ExErr, bool) { | ||||||
|  | 	if original == nil { | ||||||
|  | 		return nil, false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if ee == original { | ||||||
|  | 		return ee, true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for curr := ee; curr != nil; curr = curr.OriginalError { | ||||||
|  | 		if curr.equalsDirectProperties(curr) { | ||||||
|  | 			return curr, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil, false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // equalsDirectProperties tests if ee and other are equals, but only looks at primary properties (not `OriginalError` or `Meta`) | ||||||
|  | func (ee *ExErr) equalsDirectProperties(other *ExErr) bool { | ||||||
|  |  | ||||||
|  | 	if ee.UniqueID != other.UniqueID { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.Timestamp != other.Timestamp { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.Category != other.Category { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.Severity != other.Severity { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.Type != other.Type { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.StatusCode != other.StatusCode { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.Message != other.Message { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.WrappedErrType != other.WrappedErrType { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if ee.Caller != other.Caller { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func newID() string { | ||||||
|  | 	return xid.New().String() | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										93
									
								
								exerr/exerr_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								exerr/exerr_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,93 @@ | |||||||
|  | package exerr | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"errors" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/tst" | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type golangErr struct { | ||||||
|  | 	Message string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g golangErr) Error() string { | ||||||
|  | 	return g.Message | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type golangErr2 struct { | ||||||
|  | 	Message string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g golangErr2) Error() string { | ||||||
|  | 	return g.Message | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type simpleError struct { | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g simpleError) Error() string { | ||||||
|  | 	return "Something simple went wroong" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type simpleError2 struct { | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g simpleError2) Error() string { | ||||||
|  | 	return "Something simple went wroong" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestExErrIs1(t *testing.T) { | ||||||
|  | 	e0 := simpleError{} | ||||||
|  |  | ||||||
|  | 	wrap := Wrap(e0, "something went wrong").Str("test", "123").Build() | ||||||
|  |  | ||||||
|  | 	tst.AssertTrue(t, errors.Is(wrap, simpleError{})) | ||||||
|  | 	tst.AssertFalse(t, errors.Is(wrap, golangErr{})) | ||||||
|  | 	tst.AssertFalse(t, errors.Is(wrap, golangErr{"error1"})) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestExErrIs2(t *testing.T) { | ||||||
|  | 	e0 := golangErr{"error1"} | ||||||
|  |  | ||||||
|  | 	wrap := Wrap(e0, "something went wrong").Str("test", "123").Build() | ||||||
|  |  | ||||||
|  | 	tst.AssertTrue(t, errors.Is(wrap, e0)) | ||||||
|  | 	tst.AssertTrue(t, errors.Is(wrap, golangErr{"error1"})) | ||||||
|  | 	tst.AssertFalse(t, errors.Is(wrap, golangErr{"error2"})) | ||||||
|  | 	tst.AssertFalse(t, errors.Is(wrap, simpleError{})) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestExErrAs(t *testing.T) { | ||||||
|  |  | ||||||
|  | 	e0 := golangErr{"error1"} | ||||||
|  |  | ||||||
|  | 	w0 := Wrap(e0, "something went wrong").Str("test", "123").Build() | ||||||
|  |  | ||||||
|  | 	{ | ||||||
|  | 		out := golangErr{} | ||||||
|  | 		ok := errors.As(w0, &out) | ||||||
|  | 		tst.AssertTrue(t, ok) | ||||||
|  | 		tst.AssertEqual(t, out.Message, "error1") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	w1 := Wrap(w0, "outher error").Build() | ||||||
|  |  | ||||||
|  | 	{ | ||||||
|  | 		out := golangErr{} | ||||||
|  | 		ok := errors.As(w1, &out) | ||||||
|  | 		tst.AssertTrue(t, ok) | ||||||
|  | 		tst.AssertEqual(t, out.Message, "error1") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	{ | ||||||
|  | 		out := golangErr2{} | ||||||
|  | 		ok := errors.As(w1, &out) | ||||||
|  | 		tst.AssertFalse(t, ok) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	{ | ||||||
|  | 		out := simpleError2{} | ||||||
|  | 		ok := errors.As(w1, &out) | ||||||
|  | 		tst.AssertFalse(t, ok) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										112
									
								
								exerr/gin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								exerr/gin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,112 @@ | |||||||
|  | package exerr | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	json "gogs.mikescher.com/BlackForestBytes/goext/gojson" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"net/http" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (ee *ExErr) toJson(depth int, applyExtendListener bool, outputMeta bool) langext.H { | ||||||
|  | 	ginJson := langext.H{} | ||||||
|  |  | ||||||
|  | 	if ee.UniqueID != "" { | ||||||
|  | 		ginJson["id"] = ee.UniqueID | ||||||
|  | 	} | ||||||
|  | 	if ee.Category != CatWrap { | ||||||
|  | 		ginJson["category"] = ee.Category | ||||||
|  | 	} | ||||||
|  | 	if ee.Type != TypeWrap { | ||||||
|  | 		ginJson["type"] = ee.Type | ||||||
|  | 	} | ||||||
|  | 	if ee.StatusCode != nil { | ||||||
|  | 		ginJson["statuscode"] = ee.StatusCode | ||||||
|  | 	} | ||||||
|  | 	if ee.Message != "" { | ||||||
|  | 		ginJson["message"] = ee.Message | ||||||
|  | 	} | ||||||
|  | 	if ee.Caller != "" { | ||||||
|  | 		ginJson["caller"] = ee.Caller | ||||||
|  | 	} | ||||||
|  | 	if ee.Severity != SevErr { | ||||||
|  | 		ginJson["severity"] = ee.Severity | ||||||
|  | 	} | ||||||
|  | 	if ee.Timestamp != (time.Time{}) { | ||||||
|  | 		ginJson["time"] = ee.Timestamp.Format(time.RFC3339) | ||||||
|  | 	} | ||||||
|  | 	if ee.WrappedErrType != "" { | ||||||
|  | 		ginJson["wrappedErrType"] = ee.WrappedErrType | ||||||
|  | 	} | ||||||
|  | 	if ee.OriginalError != nil { | ||||||
|  | 		ginJson["original"] = ee.OriginalError.toJson(depth+1, applyExtendListener, outputMeta) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if outputMeta { | ||||||
|  | 		metaJson := langext.H{} | ||||||
|  | 		for metaKey, metaVal := range ee.Meta { | ||||||
|  | 			metaJson[metaKey] = metaVal.rawValueForJson() | ||||||
|  | 		} | ||||||
|  | 		ginJson["meta"] = metaJson | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if applyExtendListener { | ||||||
|  | 		pkgconfig.ExtendGinDataOutput(ee, depth, ginJson) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return ginJson | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToAPIJson converts the ExError to a json object | ||||||
|  | // (the same object as used in the Output(gin) method) | ||||||
|  | // | ||||||
|  | // Parameters: | ||||||
|  | //   - [applyExtendListener]:  if false the pkgconfig.ExtendGinOutput / pkgconfig.ExtendGinDataOutput will not be applied | ||||||
|  | //   - [includeWrappedErrors]: if false we do not include the recursive/wrapped errors in `__data` | ||||||
|  | //   - [includeMetaFields]:    if true  we also include meta-values (aka from `.Str(key, value).Build()`), needs includeWrappedErrors=true | ||||||
|  | func (ee *ExErr) ToAPIJson(applyExtendListener bool, includeWrappedErrors bool, includeMetaFields bool) langext.H { | ||||||
|  |  | ||||||
|  | 	apiOutput := langext.H{ | ||||||
|  | 		"errorid":   ee.UniqueID, | ||||||
|  | 		"message":   ee.RecursiveMessage(), | ||||||
|  | 		"errorcode": ee.RecursiveType().Key, | ||||||
|  | 		"category":  ee.RecursiveCategory().Category, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if includeWrappedErrors { | ||||||
|  | 		apiOutput["__data"] = ee.toJson(0, applyExtendListener, includeMetaFields) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if applyExtendListener { | ||||||
|  | 		pkgconfig.ExtendGinOutput(ee, apiOutput) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return apiOutput | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ee *ExErr) Output(g *gin.Context) { | ||||||
|  |  | ||||||
|  | 	warnOnPkgConfigNotInitialized() | ||||||
|  |  | ||||||
|  | 	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 | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ginOutput := ee.ToAPIJson(true, pkgconfig.ExtendedGinOutput, pkgconfig.IncludeMetaInGinOutput) | ||||||
|  |  | ||||||
|  | 	g.Render(statuscode, json.GoJsonRender{Data: ginOutput, NilSafeSlices: true, NilSafeMaps: true}) | ||||||
|  | } | ||||||
							
								
								
									
										88
									
								
								exerr/helper.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								exerr/helper.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | |||||||
|  | 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 | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	//goland:noinspection GoDirectComparisonOfErrors | ||||||
|  | 	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 | ||||||
| @@ -40,6 +43,7 @@ const ( | |||||||
| 	MDTID          metaDataType = "ID" | 	MDTID          metaDataType = "ID" | ||||||
| 	MDTAny         metaDataType = "Interface" | 	MDTAny         metaDataType = "Interface" | ||||||
| 	MDTNil         metaDataType = "Nil" | 	MDTNil         metaDataType = "Nil" | ||||||
|  | 	MDTEnum        metaDataType = "Enum" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type MetaValue struct { | type MetaValue struct { | ||||||
| @@ -128,6 +132,8 @@ func (v MetaValue) SerializeValue() (string, error) { | |||||||
| 		return string(r), nil | 		return string(r), nil | ||||||
| 	case MDTNil: | 	case MDTNil: | ||||||
| 		return "", nil | 		return "", nil | ||||||
|  | 	case MDTEnum: | ||||||
|  | 		return v.Value.(EnumWrap).Serialize(), nil | ||||||
| 	} | 	} | ||||||
| 	return "", errors.New("Unknown type: " + string(v.DataType)) | 	return "", errors.New("Unknown type: " + string(v.DataType)) | ||||||
| } | } | ||||||
| @@ -205,6 +211,8 @@ func (v MetaValue) ShortString(lim int) string { | |||||||
| 		return langext.StrLimit(string(r), lim, "...") | 		return langext.StrLimit(string(r), lim, "...") | ||||||
| 	case MDTNil: | 	case MDTNil: | ||||||
| 		return "<<null>>" | 		return "<<null>>" | ||||||
|  | 	case MDTEnum: | ||||||
|  | 		return v.Value.(EnumWrap).String() | ||||||
| 	} | 	} | ||||||
| 	return "(err)" | 	return "(err)" | ||||||
| } | } | ||||||
| @@ -267,6 +275,14 @@ func (v MetaValue) Apply(key string, evt *zerolog.Event) *zerolog.Event { | |||||||
| 		return evt.Ints32(key, v.Value.([]int32)) | 		return evt.Ints32(key, v.Value.([]int32)) | ||||||
| 	case MDTNil: | 	case MDTNil: | ||||||
| 		return evt.Str(key, "<<null>>") | 		return evt.Str(key, "<<null>>") | ||||||
|  | 	case MDTEnum: | ||||||
|  | 		if v.Value.(EnumWrap).IsNil { | ||||||
|  | 			return evt.Any(key, nil) | ||||||
|  | 		} else if v.Value.(EnumWrap).ValueRaw != nil { | ||||||
|  | 			return evt.Any(key, v.Value.(EnumWrap).ValueRaw) | ||||||
|  | 		} else { | ||||||
|  | 			return evt.Str(key, v.Value.(EnumWrap).ValueString) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return evt.Str(key, "(err)") | 	return evt.Str(key, "(err)") | ||||||
| } | } | ||||||
| @@ -350,11 +366,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 | ||||||
| 		} | 		} | ||||||
| @@ -512,6 +524,10 @@ func (v *MetaValue) Deserialize(value string, datatype metaDataType) error { | |||||||
| 		v.Value = nil | 		v.Value = nil | ||||||
| 		v.DataType = datatype | 		v.DataType = datatype | ||||||
| 		return nil | 		return nil | ||||||
|  | 	case MDTEnum: | ||||||
|  | 		v.Value = deserializeEnumWrap(value) | ||||||
|  | 		v.DataType = datatype | ||||||
|  | 		return nil | ||||||
| 	} | 	} | ||||||
| 	return errors.New("Unknown type: " + string(datatype)) | 	return errors.New("Unknown type: " + string(datatype)) | ||||||
| } | } | ||||||
| @@ -582,53 +598,53 @@ func (v MetaValue) ValueString() string { | |||||||
| 		return string(r) | 		return string(r) | ||||||
| 	case MDTNil: | 	case MDTNil: | ||||||
| 		return "<<null>>" | 		return "<<null>>" | ||||||
|  | 	case MDTEnum: | ||||||
|  | 		return v.Value.(EnumWrap).String() | ||||||
| 	} | 	} | ||||||
| 	return "(err)" | 	return "(err)" | ||||||
| } | } | ||||||
|  |  | ||||||
| func valueFromProto(value string, datatype metaDataType) (MetaValue, error) { | // rawValueForJson returns most-of-the-time the `Value` field | ||||||
| 	obj := MetaValue{} | // but for some datatyes we do special processing | ||||||
| 	err := obj.Deserialize(value, datatype) | // all, so we can pluck the output value in json.Marshal without any suprises | ||||||
| 	if err != nil { | func (v MetaValue) rawValueForJson() any { | ||||||
| 		return MetaValue{}, err | 	if v.DataType == MDTAny { | ||||||
|  | 		if v.Value.(AnyWrap).IsNil { | ||||||
|  | 			return nil | ||||||
| 		} | 		} | ||||||
| 	return obj, nil | 		return v.Value.(AnyWrap).Serialize() | ||||||
| 	} | 	} | ||||||
|  | 	if v.DataType == MDTID { | ||||||
| func metaFromProto(proto []*spbmodels.CustomError_MetaValue) MetaMap { | 		if v.Value.(IDWrap).IsNil { | ||||||
| 	r := make(MetaMap) | 			return nil | ||||||
|  |  | ||||||
| 	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 v.Value.(IDWrap).Value | ||||||
| 	} | 	} | ||||||
|  | 	if v.DataType == MDTBytes { | ||||||
| 	return r | 		return hex.EncodeToString(v.Value.([]byte)) | ||||||
| 	} | 	} | ||||||
|  | 	if v.DataType == MDTDuration { | ||||||
| func (mm MetaMap) ToProto() []*spbmodels.CustomError_MetaValue { | 		return v.Value.(time.Duration).String() | ||||||
| 	if mm == nil { |  | ||||||
| 		return make([]*spbmodels.CustomError_MetaValue, 0) |  | ||||||
| 	} | 	} | ||||||
| 	r := make([]*spbmodels.CustomError_MetaValue, 0, len(mm)) | 	if v.DataType == MDTTime { | ||||||
| 	for k, v := range mm { | 		return v.Value.(time.Time).Format(time.RFC3339Nano) | ||||||
| 		strval, err := v.SerializeValue() |  | ||||||
| 		if err != nil { |  | ||||||
| 			log.Warn().Err(err).Msg("MetaMap.ToProto failed for " + k) |  | ||||||
| 			continue |  | ||||||
| 	} | 	} | ||||||
|  | 	if v.DataType == MDTObjectID { | ||||||
| 		r = append(r, &spbmodels.CustomError_MetaValue{ | 		return v.Value.(primitive.ObjectID).Hex() | ||||||
| 			Key:   k, |  | ||||||
| 			Type:  string(v.DataType), |  | ||||||
| 			Value: strval, |  | ||||||
| 		}) |  | ||||||
| 	} | 	} | ||||||
| 	return r | 	if v.DataType == MDTNil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	if v.DataType == MDTEnum { | ||||||
|  | 		if v.Value.(EnumWrap).IsNil { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 		if v.Value.(EnumWrap).ValueRaw != nil { | ||||||
|  | 			return v.Value.(EnumWrap).ValueRaw | ||||||
|  | 		} | ||||||
|  | 		return v.Value.(EnumWrap).ValueString | ||||||
|  | 	} | ||||||
|  | 	return v.Value | ||||||
| } | } | ||||||
|  |  | ||||||
| func (mm MetaMap) FormatOneLine(singleMaxLen int) string { | func (mm MetaMap) FormatOneLine(singleMaxLen int) string { | ||||||
|   | |||||||
							
								
								
									
										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) | ||||||
|  | } | ||||||
| @@ -4,6 +4,7 @@ import ( | |||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/enums" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"strings" | 	"strings" | ||||||
| ) | ) | ||||||
| @@ -131,3 +132,58 @@ func deserializeAnyWrap(v string) AnyWrap { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | type EnumWrap struct { | ||||||
|  | 	Type        string | ||||||
|  | 	ValueString string | ||||||
|  | 	ValueRaw    enums.Enum // `ValueRaw` is lost during serialization roundtrip | ||||||
|  | 	IsNil       bool | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func newEnumWrap(val enums.Enum) EnumWrap { | ||||||
|  | 	t := fmt.Sprintf("%T", val) | ||||||
|  | 	arr := strings.Split(t, ".") | ||||||
|  | 	if len(arr) > 0 { | ||||||
|  | 		t = arr[len(arr)-1] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if langext.IsNil(val) { | ||||||
|  | 		return EnumWrap{Type: t, ValueString: "", ValueRaw: val, IsNil: true} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if enumstr, ok := val.(enums.StringEnum); ok { | ||||||
|  | 		return EnumWrap{Type: t, ValueString: enumstr.String(), ValueRaw: val, IsNil: false} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return EnumWrap{Type: t, ValueString: fmt.Sprintf("%v", val), ValueRaw: val, IsNil: false} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (w EnumWrap) Serialize() string { | ||||||
|  | 	if w.IsNil { | ||||||
|  | 		return "!nil" + ":" + w.Type | ||||||
|  | 	} | ||||||
|  | 	return w.Type + ":" + w.ValueString | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (w EnumWrap) String() string { | ||||||
|  | 	if w.IsNil { | ||||||
|  | 		return w.Type + "<<nil>>" | ||||||
|  | 	} | ||||||
|  | 	return "[" + w.Type + "] " + w.ValueString | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func deserializeEnumWrap(v string) EnumWrap { | ||||||
|  | 	r := strings.SplitN(v, ":", 2) | ||||||
|  | 
 | ||||||
|  | 	if len(r) == 2 && r[0] == "!nil" { | ||||||
|  | 		return EnumWrap{Type: r[1], ValueString: v, ValueRaw: nil, IsNil: true} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if len(r) == 0 { | ||||||
|  | 		return EnumWrap{} | ||||||
|  | 	} else if len(r) == 1 { | ||||||
|  | 		return EnumWrap{Type: "", ValueString: v, ValueRaw: nil, IsNil: false} | ||||||
|  | 	} else { | ||||||
|  | 		return EnumWrap{Type: r[0], ValueString: r[1], ValueRaw: nil, IsNil: false} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -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"` |  | ||||||
| } |  | ||||||
| @@ -14,6 +14,9 @@ type AppContext struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| func CreateAppContext(g *gin.Context, innerCtx context.Context, cancelFn context.CancelFunc) *AppContext { | func CreateAppContext(g *gin.Context, innerCtx context.Context, cancelFn context.CancelFunc) *AppContext { | ||||||
|  | 	for key, value := range g.Keys { | ||||||
|  | 		innerCtx = context.WithValue(innerCtx, key, value) | ||||||
|  | 	} | ||||||
| 	return &AppContext{ | 	return &AppContext{ | ||||||
| 		inner:      innerCtx, | 		inner:      innerCtx, | ||||||
| 		cancelFunc: cancelFn, | 		cancelFunc: cancelFn, | ||||||
| @@ -38,6 +41,10 @@ func (ac *AppContext) Value(key any) any { | |||||||
| 	return ac.inner.Value(key) | 	return ac.inner.Value(key) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (ac *AppContext) Set(key, value any) { | ||||||
|  | 	ac.inner = context.WithValue(ac.inner, key, value) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (ac *AppContext) Cancel() { | func (ac *AppContext) Cancel() { | ||||||
| 	ac.cancelled = true | 	ac.cancelled = true | ||||||
| 	ac.cancelFunc() | 	ac.cancelFunc() | ||||||
| @@ -50,10 +57,3 @@ func (ac *AppContext) RequestURI() string { | |||||||
| 		return "" | 		return "" | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) FinishSuccess(res HTTPResponse) HTTPResponse { |  | ||||||
| 	if ac.cancelled { |  | ||||||
| 		panic("Cannot finish a cancelled request") |  | ||||||
| 	} |  | ||||||
| 	return res |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -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
									
								
								ginext/commonMiddlewares.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								ginext/commonMiddlewares.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | package ginext | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func BodyBuffer(g *gin.Context) { | ||||||
|  | 	if g.Request.Body != nil { | ||||||
|  | 		g.Request.Body = dataext.NewBufferedReadCloser(g.Request.Body) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										107
									
								
								ginext/engine.go
									
									
									
									
									
								
							
							
						
						
									
										107
									
								
								ginext/engine.go
									
									
									
									
									
								
							| @@ -1,8 +1,14 @@ | |||||||
| package ginext | package ginext | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | ||||||
|  | 	"net" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -12,11 +18,26 @@ type GinWrapper struct { | |||||||
|  |  | ||||||
| 	allowCors      bool | 	allowCors      bool | ||||||
| 	ginDebug       bool | 	ginDebug       bool | ||||||
| 	returnRawErrors bool | 	bufferBody     bool | ||||||
| 	requestTimeout time.Duration | 	requestTimeout time.Duration | ||||||
|  |  | ||||||
|  | 	routeSpecs []ginRouteSpec | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewEngine(allowCors bool, ginDebug bool, returnRawErrors bool, timeout time.Duration) *GinWrapper { | type ginRouteSpec struct { | ||||||
|  | 	Method      string | ||||||
|  | 	URL         string | ||||||
|  | 	Middlewares []string | ||||||
|  | 	Handler     string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // NewEngine creates a new (wrapped) ginEngine | ||||||
|  | // Parameters are: | ||||||
|  | // - [allowCors]    Add cors handler to allow all CORS requests on the default http methods | ||||||
|  | // - [ginDebug]     Set gin.debug to true (adds more logs) | ||||||
|  | // - [bufferBody]   Buffers the input body stream, this way the ginext error handler can later include the whole request body | ||||||
|  | // - [timeout]      The default handler timeout | ||||||
|  | func NewEngine(allowCors bool, ginDebug bool, bufferBody bool, timeout time.Duration) *GinWrapper { | ||||||
| 	engine := gin.New() | 	engine := gin.New() | ||||||
|  |  | ||||||
| 	wrapper := &GinWrapper{ | 	wrapper := &GinWrapper{ | ||||||
| @@ -24,7 +45,7 @@ func NewEngine(allowCors bool, ginDebug bool, returnRawErrors bool, timeout time | |||||||
| 		SuppressGinLogs: false, | 		SuppressGinLogs: false, | ||||||
| 		allowCors:       allowCors, | 		allowCors:       allowCors, | ||||||
| 		ginDebug:        ginDebug, | 		ginDebug:        ginDebug, | ||||||
| 		returnRawErrors: returnRawErrors, | 		bufferBody:      bufferBody, | ||||||
| 		requestTimeout:  timeout, | 		requestTimeout:  timeout, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -35,18 +56,94 @@ func NewEngine(allowCors bool, ginDebug bool, returnRawErrors bool, timeout time | |||||||
| 		engine.Use(CorsMiddleware()) | 		engine.Use(CorsMiddleware()) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// do not debug-print routes | ||||||
|  | 	gin.DebugPrintRouteFunc = func(_, _, _ string, _ int) {} | ||||||
|  |  | ||||||
| 	if ginDebug { | 	if ginDebug { | ||||||
|  | 		gin.SetMode(gin.ReleaseMode) | ||||||
|  |  | ||||||
| 		ginlogger := gin.Logger() | 		ginlogger := gin.Logger() | ||||||
| 		engine.Use(func(context *gin.Context) { | 		engine.Use(func(context *gin.Context) { | ||||||
| 			if !wrapper.SuppressGinLogs { | 			if !wrapper.SuppressGinLogs { | ||||||
| 				ginlogger(context) | 				ginlogger(context) | ||||||
| 			} | 			} | ||||||
| 		}) | 		}) | ||||||
|  | 	} else { | ||||||
|  | 		gin.SetMode(gin.DebugMode) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return wrapper | 	return wrapper | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinWrapper) ServeHTTP(writer http.ResponseWriter, request *http.Request) { | func (w *GinWrapper) ListenAndServeHTTP(addr string, postInit func(port string)) (chan error, *http.Server) { | ||||||
| 	w.engine.ServeHTTP(writer, request) |  | ||||||
|  | 	w.DebugPrintRoutes() | ||||||
|  |  | ||||||
|  | 	httpserver := &http.Server{ | ||||||
|  | 		Addr:    addr, | ||||||
|  | 		Handler: w.engine, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	errChan := make(chan error) | ||||||
|  |  | ||||||
|  | 	go func() { | ||||||
|  |  | ||||||
|  | 		ln, err := net.Listen("tcp", httpserver.Addr) | ||||||
|  | 		if err != nil { | ||||||
|  | 			errChan <- err | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		_, port, err := net.SplitHostPort(ln.Addr().String()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			errChan <- err | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		log.Info().Str("address", httpserver.Addr).Msg("HTTP-Server started on http://localhost:" + port) | ||||||
|  |  | ||||||
|  | 		if postInit != nil { | ||||||
|  | 			postInit(port) // the net.Listener a few lines above is at this point actually already buffering requests | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		errChan <- httpserver.Serve(ln) | ||||||
|  | 	}() | ||||||
|  |  | ||||||
|  | 	return errChan, httpserver | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *GinWrapper) DebugPrintRoutes() { | ||||||
|  | 	if !w.ginDebug { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	lines := make([][4]string, 0) | ||||||
|  |  | ||||||
|  | 	pad := [4]int{0, 0, 0, 0} | ||||||
|  |  | ||||||
|  | 	for _, spec := range w.routeSpecs { | ||||||
|  |  | ||||||
|  | 		line := [4]string{ | ||||||
|  | 			spec.Method, | ||||||
|  | 			spec.URL, | ||||||
|  | 			strings.Join(spec.Middlewares, " -> "), | ||||||
|  | 			spec.Handler, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		lines = append(lines, line) | ||||||
|  |  | ||||||
|  | 		pad[0] = mathext.Max(pad[0], len(line[0])) | ||||||
|  | 		pad[1] = mathext.Max(pad[1], len(line[1])) | ||||||
|  | 		pad[2] = mathext.Max(pad[2], len(line[2])) | ||||||
|  | 		pad[3] = mathext.Max(pad[3], len(line[3])) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, line := range lines { | ||||||
|  |  | ||||||
|  | 		fmt.Printf("Gin-Route: %s  %s  -->  %s  -->  %s\n", | ||||||
|  | 			langext.StrPadRight("["+line[0]+"]", " ", pad[0]+2), | ||||||
|  | 			langext.StrPadRight(line[1], " ", pad[1]), | ||||||
|  | 			langext.StrPadRight(line[2], " ", pad[2]), | ||||||
|  | 			langext.StrPadRight(line[3], " ", pad[3])) | ||||||
|  | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -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 = Error(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if g.Writer.Written() { | 		if g.Writer.Written() { | ||||||
|   | |||||||
| @@ -2,10 +2,13 @@ package ginext | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/gin-gonic/gin/binding" | 	"github.com/gin-gonic/gin/binding" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"runtime/debug" | 	"runtime/debug" | ||||||
|  | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type PreContext struct { | type PreContext struct { | ||||||
| @@ -15,6 +18,8 @@ type PreContext struct { | |||||||
| 	query   any | 	query   any | ||||||
| 	body    any | 	body    any | ||||||
| 	form    any | 	form    any | ||||||
|  | 	header  any | ||||||
|  | 	timeout *time.Duration | ||||||
| } | } | ||||||
|  |  | ||||||
| func (pctx *PreContext) URI(uri any) *PreContext { | func (pctx *PreContext) URI(uri any) *PreContext { | ||||||
| @@ -37,40 +42,90 @@ func (pctx *PreContext) Form(form any) *PreContext { | |||||||
| 	return pctx | 	return pctx | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (pctx *PreContext) Header(header any) *PreContext { | ||||||
|  | 	pctx.header = header | ||||||
|  | 	return pctx | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (pctx *PreContext) WithTimeout(to time.Duration) *PreContext { | ||||||
|  | 	pctx.timeout = &to | ||||||
|  | 	return pctx | ||||||
|  | } | ||||||
|  |  | ||||||
| func (pctx PreContext) Start() (*AppContext, *gin.Context, *HTTPResponse) { | func (pctx PreContext) Start() (*AppContext, *gin.Context, *HTTPResponse) { | ||||||
| 	if pctx.uri != nil { | 	if pctx.uri != nil { | ||||||
| 		if err := pctx.ginCtx.ShouldBindUri(pctx.uri); err != nil { | 		if err := pctx.ginCtx.ShouldBindUri(pctx.uri); err != nil { | ||||||
| 			return nil, nil, langext.Ptr(APIError(pctx.ginCtx, commonApiErr.BindFailURI, "Failed to read uri", err)) | 			err = exerr.Wrap(err, "Failed to read uri"). | ||||||
|  | 				WithType(exerr.TypeBindFailURI). | ||||||
|  | 				Str("struct_type", fmt.Sprintf("%T", pctx.uri)). | ||||||
|  | 				Build() | ||||||
|  | 			return nil, nil, langext.Ptr(Error(err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if pctx.query != nil { | 	if pctx.query != nil { | ||||||
| 		if err := pctx.ginCtx.ShouldBindQuery(pctx.query); err != nil { | 		if err := pctx.ginCtx.ShouldBindQuery(pctx.query); err != nil { | ||||||
| 			return nil, nil, langext.Ptr(APIError(pctx.ginCtx, commonApiErr.BindFailQuery, "Failed to read query", err)) | 			err = exerr.Wrap(err, "Failed to read query"). | ||||||
|  | 				WithType(exerr.TypeBindFailQuery). | ||||||
|  | 				Str("struct_type", fmt.Sprintf("%T", pctx.query)). | ||||||
|  | 				Build() | ||||||
|  | 			return nil, nil, langext.Ptr(Error(err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if pctx.body != nil { | 	if pctx.body != nil { | ||||||
| 		if pctx.ginCtx.ContentType() == "application/json" { | 		if pctx.ginCtx.ContentType() == "application/json" { | ||||||
| 			if err := pctx.ginCtx.ShouldBindJSON(pctx.body); err != nil { | 			if err := pctx.ginCtx.ShouldBindJSON(pctx.body); err != nil { | ||||||
| 				return nil, nil, langext.Ptr(APIError(pctx.ginCtx, commonApiErr.BindFailJSON, "Failed to read body", err)) | 				err = exerr.Wrap(err, "Failed to read json-body"). | ||||||
|  | 					WithType(exerr.TypeBindFailJSON). | ||||||
|  | 					Str("struct_type", fmt.Sprintf("%T", pctx.body)). | ||||||
|  | 					Build() | ||||||
|  | 				return nil, nil, langext.Ptr(Error(err)) | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			return nil, nil, langext.Ptr(APIError(pctx.ginCtx, commonApiErr.BindFailJSON, "missing JSON body", nil)) | 			err := exerr.New(exerr.TypeBindFailJSON, "missing JSON body"). | ||||||
|  | 				Str("struct_type", fmt.Sprintf("%T", pctx.body)). | ||||||
|  | 				Build() | ||||||
|  | 			return nil, nil, langext.Ptr(Error(err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if pctx.form != nil { | 	if pctx.form != nil { | ||||||
| 		if pctx.ginCtx.ContentType() == "multipart/form-data" { | 		if pctx.ginCtx.ContentType() == "multipart/form-data" { | ||||||
| 			if err := pctx.ginCtx.ShouldBindWith(pctx.form, binding.Form); err != nil { | 			if err := pctx.ginCtx.ShouldBindWith(pctx.form, binding.Form); err != nil { | ||||||
| 				return nil, nil, langext.Ptr(APIError(pctx.ginCtx, commonApiErr.BindFailFormData, "Failed to read multipart-form", err)) | 				err = exerr.Wrap(err, "Failed to read multipart-form"). | ||||||
|  | 					WithType(exerr.TypeBindFailFormData). | ||||||
|  | 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | ||||||
|  | 					Build() | ||||||
|  | 				return nil, nil, langext.Ptr(Error(err)) | ||||||
|  | 			} | ||||||
|  | 		} else if pctx.ginCtx.ContentType() == "application/x-www-form-urlencoded" { | ||||||
|  | 			if err := pctx.ginCtx.ShouldBindWith(pctx.form, binding.Form); err != nil { | ||||||
|  | 				err = exerr.Wrap(err, "Failed to read urlencoded-form"). | ||||||
|  | 					WithType(exerr.TypeBindFailFormData). | ||||||
|  | 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | ||||||
|  | 					Build() | ||||||
|  | 				return nil, nil, langext.Ptr(Error(err)) | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			return nil, nil, langext.Ptr(APIError(pctx.ginCtx, commonApiErr.BindFailJSON, "missing form body", nil)) | 			err := exerr.New(exerr.TypeBindFailFormData, "missing form body"). | ||||||
|  | 				Str("struct_type", fmt.Sprintf("%T", pctx.form)). | ||||||
|  | 				Build() | ||||||
|  | 			return nil, nil, langext.Ptr(Error(err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ictx, cancel := context.WithTimeout(context.Background(), pctx.wrapper.requestTimeout) | 	if pctx.header != nil { | ||||||
|  | 		if err := pctx.ginCtx.ShouldBindHeader(pctx.query); err != nil { | ||||||
|  | 			err = exerr.Wrap(err, "Failed to read header"). | ||||||
|  | 				WithType(exerr.TypeBindFailHeader). | ||||||
|  | 				Str("struct_type", fmt.Sprintf("%T", pctx.query)). | ||||||
|  | 				Build() | ||||||
|  | 			return nil, nil, langext.Ptr(Error(err)) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ictx, cancel := context.WithTimeout(context.Background(), langext.Coalesce(pctx.timeout, pctx.wrapper.requestTimeout)) | ||||||
| 	actx := CreateAppContext(pctx.ginCtx, ictx, cancel) | 	actx := CreateAppContext(pctx.ginCtx, ictx, cancel) | ||||||
|  |  | ||||||
| 	return actx, pctx.ginCtx, nil | 	return actx, pctx.ginCtx, nil | ||||||
|   | |||||||
| @@ -3,57 +3,97 @@ 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 headerval struct { | ||||||
|  | 	Key string | ||||||
|  | 	Val string | ||||||
|  | } | ||||||
|  |  | ||||||
| type HTTPResponse interface { | type HTTPResponse interface { | ||||||
| 	Write(g *gin.Context) | 	Write(g *gin.Context) | ||||||
|  | 	WithHeader(k string, v string) HTTPResponse | ||||||
| } | } | ||||||
|  |  | ||||||
| type jsonHTTPResponse struct { | type jsonHTTPResponse struct { | ||||||
| 	statusCode int | 	statusCode int | ||||||
| 	data       any | 	data       any | ||||||
|  | 	headers    []headerval | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j jsonHTTPResponse) Write(g *gin.Context) { | func (j jsonHTTPResponse) Write(g *gin.Context) { | ||||||
|  | 	for _, v := range j.headers { | ||||||
|  | 		g.Header(v.Key, v.Val) | ||||||
|  | 	} | ||||||
| 	g.Render(j.statusCode, json.GoJsonRender{Data: j.data, NilSafeSlices: true, NilSafeMaps: true}) | 	g.Render(j.statusCode, json.GoJsonRender{Data: j.data, NilSafeSlices: true, NilSafeMaps: true}) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (j jsonHTTPResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
| type emptyHTTPResponse struct { | type emptyHTTPResponse struct { | ||||||
| 	statusCode int | 	statusCode int | ||||||
|  | 	headers    []headerval | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j emptyHTTPResponse) Write(g *gin.Context) { | func (j emptyHTTPResponse) Write(g *gin.Context) { | ||||||
|  | 	for _, v := range j.headers { | ||||||
|  | 		g.Header(v.Key, v.Val) | ||||||
|  | 	} | ||||||
| 	g.Status(j.statusCode) | 	g.Status(j.statusCode) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (j emptyHTTPResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
| type textHTTPResponse struct { | type textHTTPResponse struct { | ||||||
| 	statusCode int | 	statusCode int | ||||||
| 	data       string | 	data       string | ||||||
|  | 	headers    []headerval | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j textHTTPResponse) Write(g *gin.Context) { | func (j textHTTPResponse) Write(g *gin.Context) { | ||||||
|  | 	for _, v := range j.headers { | ||||||
|  | 		g.Header(v.Key, v.Val) | ||||||
|  | 	} | ||||||
| 	g.String(j.statusCode, "%s", j.data) | 	g.String(j.statusCode, "%s", j.data) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (j textHTTPResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
| type dataHTTPResponse struct { | type dataHTTPResponse struct { | ||||||
| 	statusCode  int | 	statusCode  int | ||||||
| 	data        []byte | 	data        []byte | ||||||
| 	contentType string | 	contentType string | ||||||
|  | 	headers     []headerval | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j dataHTTPResponse) Write(g *gin.Context) { | func (j dataHTTPResponse) Write(g *gin.Context) { | ||||||
|  | 	for _, v := range j.headers { | ||||||
|  | 		g.Header(v.Key, v.Val) | ||||||
|  | 	} | ||||||
| 	g.Data(j.statusCode, j.contentType, j.data) | 	g.Data(j.statusCode, j.contentType, j.data) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (j dataHTTPResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
| type fileHTTPResponse struct { | type fileHTTPResponse struct { | ||||||
| 	mimetype string | 	mimetype string | ||||||
| 	filepath string | 	filepath string | ||||||
| 	filename *string | 	filename *string | ||||||
|  | 	headers  []headerval | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j fileHTTPResponse) Write(g *gin.Context) { | func (j fileHTTPResponse) Write(g *gin.Context) { | ||||||
| @@ -62,18 +102,71 @@ func (j fileHTTPResponse) Write(g *gin.Context) { | |||||||
| 		g.Header("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", *j.filename)) | 		g.Header("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", *j.filename)) | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
|  | 	for _, v := range j.headers { | ||||||
|  | 		g.Header(v.Key, v.Val) | ||||||
|  | 	} | ||||||
| 	g.File(j.filepath) | 	g.File(j.filepath) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (j fileHTTPResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type downloadDataHTTPResponse struct { | ||||||
|  | 	statusCode int | ||||||
|  | 	mimetype   string | ||||||
|  | 	data       []byte | ||||||
|  | 	filename   *string | ||||||
|  | 	headers    []headerval | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (j downloadDataHTTPResponse) Write(g *gin.Context) { | ||||||
|  | 	g.Header("Content-Type", j.mimetype) // if we don't set it here gin does weird file-sniffing later... | ||||||
|  | 	if j.filename != nil { | ||||||
|  | 		g.Header("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", *j.filename)) | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  | 	for _, v := range j.headers { | ||||||
|  | 		g.Header(v.Key, v.Val) | ||||||
|  | 	} | ||||||
|  | 	g.Data(j.statusCode, j.mimetype, j.data) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (j downloadDataHTTPResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
| type redirectHTTPResponse struct { | type redirectHTTPResponse struct { | ||||||
| 	statusCode int | 	statusCode int | ||||||
| 	url        string | 	url        string | ||||||
|  | 	headers    []headerval | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j redirectHTTPResponse) Write(g *gin.Context) { | func (j redirectHTTPResponse) Write(g *gin.Context) { | ||||||
| 	g.Redirect(j.statusCode, j.url) | 	g.Redirect(j.statusCode, j.url) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (j redirectHTTPResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type jsonAPIErrResponse struct { | ||||||
|  | 	err     *exerr.ExErr | ||||||
|  | 	headers []headerval | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (j jsonAPIErrResponse) Write(g *gin.Context) { | ||||||
|  | 	j.err.Output(g) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (j jsonAPIErrResponse) WithHeader(k string, v string) HTTPResponse { | ||||||
|  | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
|  | } | ||||||
|  |  | ||||||
| func Status(sc int) HTTPResponse { | func Status(sc int) HTTPResponse { | ||||||
| 	return &emptyHTTPResponse{statusCode: sc} | 	return &emptyHTTPResponse{statusCode: sc} | ||||||
| } | } | ||||||
| @@ -98,56 +191,26 @@ func Download(mimetype string, filepath string, filename string) HTTPResponse { | |||||||
| 	return &fileHTTPResponse{mimetype: mimetype, filepath: filepath, filename: &filename} | 	return &fileHTTPResponse{mimetype: mimetype, filepath: filepath, filename: &filename} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func DownloadData(status int, mimetype string, filename string, data []byte) HTTPResponse { | ||||||
|  | 	return &downloadDataHTTPResponse{statusCode: status, mimetype: mimetype, data: data, filename: &filename} | ||||||
|  | } | ||||||
|  |  | ||||||
| func Redirect(sc int, newURL string) HTTPResponse { | 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 Error(e error) HTTPResponse { | ||||||
| 	return createApiError(g, errcode, msg, e) | 	return &jsonAPIErrResponse{ | ||||||
|  | 		err: exerr.FromError(e), | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func NotImplemented(g *gin.Context) HTTPResponse { | func ErrWrap(e error, errorType exerr.ErrorType, msg string) HTTPResponse { | ||||||
| 	return createApiError(g, commonApiErr.NotImplemented, "", nil) | 	return &jsonAPIErrResponse{ | ||||||
|  | 		err: exerr.FromError(exerr.Wrap(e, msg).WithType(errorType).Build()), | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func createApiError(g *gin.Context, errcode commonApiErr.APIErrorCode, msg string, e error) HTTPResponse { | func NotImplemented() HTTPResponse { | ||||||
| 	reqUri := "" | 	return Error(exerr.New(exerr.TypeNotImplemented, "").Build()) | ||||||
| 	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, |  | ||||||
| 			}, |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										158
									
								
								ginext/routes.go
									
									
									
									
									
								
							
							
						
						
									
										158
									
								
								ginext/routes.go
									
									
									
									
									
								
							| @@ -2,7 +2,14 @@ package ginext | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/rext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"path" | ||||||
|  | 	"reflect" | ||||||
|  | 	"regexp" | ||||||
|  | 	"runtime" | ||||||
|  | 	"strings" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var anyMethods = []string{ | var anyMethods = []string{ | ||||||
| @@ -14,58 +21,87 @@ var anyMethods = []string{ | |||||||
| type GinRoutesWrapper struct { | type GinRoutesWrapper struct { | ||||||
| 	wrapper        *GinWrapper | 	wrapper        *GinWrapper | ||||||
| 	routes         gin.IRouter | 	routes         gin.IRouter | ||||||
|  | 	absPath        string | ||||||
|  | 	defaultHandler []gin.HandlerFunc | ||||||
| } | } | ||||||
|  |  | ||||||
| type GinRouteBuilder struct { | type GinRouteBuilder struct { | ||||||
| 	routes *GinRoutesWrapper | 	routes *GinRoutesWrapper | ||||||
|  |  | ||||||
| 	methods  []string | 	method   string | ||||||
| 	relPath  string | 	relPath  string | ||||||
|  | 	absPath  string | ||||||
| 	handlers []gin.HandlerFunc | 	handlers []gin.HandlerFunc | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinWrapper) Routes() *GinRoutesWrapper { | func (w *GinWrapper) Routes() *GinRoutesWrapper { | ||||||
| 	return &GinRoutesWrapper{wrapper: w, routes: w.engine} | 	return &GinRoutesWrapper{ | ||||||
|  | 		wrapper:        w, | ||||||
|  | 		routes:         w.engine, | ||||||
|  | 		absPath:        "", | ||||||
|  | 		defaultHandler: make([]gin.HandlerFunc, 0), | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) Group(relativePath string) *GinRoutesWrapper { | func (w *GinRoutesWrapper) Group(relativePath string) *GinRoutesWrapper { | ||||||
| 	return &GinRoutesWrapper{wrapper: w.wrapper, routes: w.routes.Group(relativePath)} | 	return &GinRoutesWrapper{ | ||||||
|  | 		wrapper:        w.wrapper, | ||||||
|  | 		routes:         w.routes.Group(relativePath), | ||||||
|  | 		defaultHandler: langext.ArrCopy(w.defaultHandler), | ||||||
|  | 		absPath:        joinPaths(w.absPath, relativePath), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *GinRoutesWrapper) Use(middleware ...gin.HandlerFunc) *GinRoutesWrapper { | ||||||
|  | 	defHandler := langext.ArrCopy(w.defaultHandler) | ||||||
|  | 	defHandler = append(defHandler, middleware...) | ||||||
|  | 	return &GinRoutesWrapper{wrapper: w.wrapper, routes: w.routes, defaultHandler: defHandler} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) GET(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) GET(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{http.MethodGet}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route(http.MethodGet, relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) POST(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) POST(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{http.MethodPost}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route(http.MethodPost, relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) DELETE(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) DELETE(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{http.MethodDelete}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route(http.MethodDelete, relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) PATCH(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) PATCH(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{http.MethodPatch}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route(http.MethodPatch, relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) PUT(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) PUT(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{http.MethodPut}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route(http.MethodPut, relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) OPTIONS(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) OPTIONS(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{http.MethodOptions}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route(http.MethodOptions, relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) HEAD(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) HEAD(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{http.MethodHead}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route(http.MethodHead, relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) COUNT(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) COUNT(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: []string{"COUNT"}, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route("COUNT", relativePath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRoutesWrapper) Any(relativePath string) *GinRouteBuilder { | func (w *GinRoutesWrapper) Any(relativePath string) *GinRouteBuilder { | ||||||
| 	return &GinRouteBuilder{routes: w, methods: anyMethods, relPath: relativePath, handlers: make([]gin.HandlerFunc, 0)} | 	return w._route("*", relativePath) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *GinRoutesWrapper) _route(method string, relativePath string) *GinRouteBuilder { | ||||||
|  | 	return &GinRouteBuilder{ | ||||||
|  | 		routes:   w, | ||||||
|  | 		method:   method, | ||||||
|  | 		relPath:  relativePath, | ||||||
|  | 		absPath:  joinPaths(w.absPath, relativePath), | ||||||
|  | 		handlers: langext.ArrCopy(w.defaultHandler), | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRouteBuilder) Use(middleware ...gin.HandlerFunc) *GinRouteBuilder { | func (w *GinRouteBuilder) Use(middleware ...gin.HandlerFunc) *GinRouteBuilder { | ||||||
| @@ -74,12 +110,102 @@ func (w *GinRouteBuilder) Use(middleware ...gin.HandlerFunc) *GinRouteBuilder { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinRouteBuilder) Handle(handler WHandlerFunc) { | func (w *GinRouteBuilder) Handle(handler WHandlerFunc) { | ||||||
| 	w.handlers = append(w.handlers, Wrap(w.routes.wrapper, handler)) |  | ||||||
| 	for _, m := range w.methods { | 	if w.routes.wrapper.bufferBody { | ||||||
| 		w.routes.routes.Handle(m, w.relPath, w.handlers...) | 		arr := make([]gin.HandlerFunc, 0, len(w.handlers)+1) | ||||||
|  | 		arr = append(arr, BodyBuffer) | ||||||
|  | 		arr = append(arr, w.handlers...) | ||||||
|  | 		w.handlers = arr | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	middlewareNames := langext.ArrMap(w.handlers, func(v gin.HandlerFunc) string { return nameOfFunction(v) }) | ||||||
|  | 	handlerName := nameOfFunction(handler) | ||||||
|  |  | ||||||
|  | 	w.handlers = append(w.handlers, Wrap(w.routes.wrapper, handler)) | ||||||
|  |  | ||||||
|  | 	methodName := w.method | ||||||
|  |  | ||||||
|  | 	if w.method == "*" { | ||||||
|  | 		methodName = "ANY" | ||||||
|  | 		for _, method := range anyMethods { | ||||||
|  | 			w.routes.routes.Handle(method, w.relPath, w.handlers...) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		w.routes.routes.Handle(w.method, w.relPath, w.handlers...) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	w.routes.wrapper.routeSpecs = append(w.routes.wrapper.routeSpecs, ginRouteSpec{ | ||||||
|  | 		Method:      methodName, | ||||||
|  | 		URL:         w.absPath, | ||||||
|  | 		Middlewares: middlewareNames, | ||||||
|  | 		Handler:     handlerName, | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (w *GinWrapper) NoRoute(handler WHandlerFunc) { | func (w *GinWrapper) NoRoute(handler WHandlerFunc) { | ||||||
| 	w.engine.NoRoute(Wrap(w, handler)) |  | ||||||
|  | 	handlers := make([]gin.HandlerFunc, 0) | ||||||
|  |  | ||||||
|  | 	if w.bufferBody { | ||||||
|  | 		handlers = append(handlers, BodyBuffer) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	middlewareNames := langext.ArrMap(handlers, func(v gin.HandlerFunc) string { return nameOfFunction(v) }) | ||||||
|  | 	handlerName := nameOfFunction(handler) | ||||||
|  |  | ||||||
|  | 	handlers = append(handlers, Wrap(w, handler)) | ||||||
|  |  | ||||||
|  | 	w.engine.NoRoute(handlers...) | ||||||
|  |  | ||||||
|  | 	w.routeSpecs = append(w.routeSpecs, ginRouteSpec{ | ||||||
|  | 		Method:      "ANY", | ||||||
|  | 		URL:         "[NO_ROUTE]", | ||||||
|  | 		Middlewares: middlewareNames, | ||||||
|  | 		Handler:     handlerName, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func nameOfFunction(f any) string { | ||||||
|  |  | ||||||
|  | 	fname := runtime.FuncForPC(reflect.ValueOf(f).Pointer()).Name() | ||||||
|  |  | ||||||
|  | 	split := strings.Split(fname, "/") | ||||||
|  | 	if len(split) == 0 { | ||||||
|  | 		return "" | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fname = split[len(split)-1] | ||||||
|  |  | ||||||
|  | 	// https://stackoverflow.com/a/32925345/1761622 | ||||||
|  | 	if strings.HasSuffix(fname, "-fm") { | ||||||
|  | 		fname = fname[:len(fname)-len("-fm")] | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	suffix := rext.W(regexp.MustCompile(`\.func[0-9]+(?:\.[0-9]+)*$`)) | ||||||
|  |  | ||||||
|  | 	if match, ok := suffix.MatchFirst(fname); ok { | ||||||
|  | 		fname = fname[:len(fname)-match.FullMatch().Length()] | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return fname | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // joinPaths is copied verbatim from gin@v1.9.1/gin.go | ||||||
|  | func joinPaths(absolutePath, relativePath string) string { | ||||||
|  | 	if relativePath == "" { | ||||||
|  | 		return absolutePath | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	finalPath := path.Join(absolutePath, relativePath) | ||||||
|  | 	if lastChar(relativePath) == '/' && lastChar(finalPath) != '/' { | ||||||
|  | 		return finalPath + "/" | ||||||
|  | 	} | ||||||
|  | 	return finalPath | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func lastChar(str string) uint8 { | ||||||
|  | 	if str == "" { | ||||||
|  | 		panic("The length of the string can't be 0") | ||||||
|  | 	} | ||||||
|  | 	return str[len(str)-1] | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								go.mod
									
									
									
									
									
								
							| @@ -5,22 +5,23 @@ 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/zerolog v1.29.1 | 	github.com/rs/xid v1.5.0 | ||||||
| 	go.mongodb.org/mongo-driver v1.12.0 | 	github.com/rs/zerolog v1.30.0 | ||||||
| 	golang.org/x/crypto v0.11.0 | 	go.mongodb.org/mongo-driver v1.12.1 | ||||||
| 	golang.org/x/sys v0.10.0 | 	golang.org/x/crypto v0.12.0 | ||||||
| 	golang.org/x/term v0.10.0 | 	golang.org/x/sys v0.11.0 | ||||||
|  | 	golang.org/x/term v0.11.0 | ||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/bytedance/sonic v1.10.0-rc2 // indirect | 	github.com/bytedance/sonic v1.10.0 // indirect | ||||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | ||||||
| 	github.com/chenzhuoyu/iasm v0.9.0 // indirect | 	github.com/chenzhuoyu/iasm v0.9.0 // indirect | ||||||
| 	github.com/gabriel-vasile/mimetype v1.4.2 // indirect | 	github.com/gabriel-vasile/mimetype v1.4.2 // indirect | ||||||
| 	github.com/gin-contrib/sse v0.1.0 // indirect | 	github.com/gin-contrib/sse v0.1.0 // indirect | ||||||
| 	github.com/go-playground/locales v0.14.1 // indirect | 	github.com/go-playground/locales v0.14.1 // indirect | ||||||
| 	github.com/go-playground/universal-translator v0.18.1 // indirect | 	github.com/go-playground/universal-translator v0.18.1 // indirect | ||||||
| 	github.com/go-playground/validator/v10 v10.14.1 // indirect | 	github.com/go-playground/validator/v10 v10.15.0 // indirect | ||||||
| 	github.com/goccy/go-json v0.10.2 // indirect | 	github.com/goccy/go-json v0.10.2 // indirect | ||||||
| 	github.com/golang/snappy v0.0.4 // indirect | 	github.com/golang/snappy v0.0.4 // indirect | ||||||
| 	github.com/json-iterator/go v1.1.12 // indirect | 	github.com/json-iterator/go v1.1.12 // indirect | ||||||
| @@ -41,9 +42,9 @@ require ( | |||||||
| 	github.com/xdg-go/stringprep v1.0.4 // indirect | 	github.com/xdg-go/stringprep v1.0.4 // indirect | ||||||
| 	github.com/youmark/pkcs8 v0.0.0-20201027041543-1326539a0a0a // indirect | 	github.com/youmark/pkcs8 v0.0.0-20201027041543-1326539a0a0a // indirect | ||||||
| 	golang.org/x/arch v0.4.0 // indirect | 	golang.org/x/arch v0.4.0 // indirect | ||||||
| 	golang.org/x/net v0.12.0 // indirect | 	golang.org/x/net v0.14.0 // indirect | ||||||
| 	golang.org/x/sync v0.3.0 // indirect | 	golang.org/x/sync v0.3.0 // indirect | ||||||
| 	golang.org/x/text v0.11.0 // indirect | 	golang.org/x/text v0.12.0 // indirect | ||||||
| 	google.golang.org/protobuf v1.31.0 // indirect | 	google.golang.org/protobuf v1.31.0 // indirect | ||||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								go.sum
									
									
									
									
									
								
							| @@ -2,6 +2,10 @@ github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1 | |||||||
| github.com/bytedance/sonic v1.10.0-rc/go.mod h1:ElCzW+ufi8qKqNW0FY314xriJhyJhuoJ3gFZdAHF7NM= | github.com/bytedance/sonic v1.10.0-rc/go.mod h1:ElCzW+ufi8qKqNW0FY314xriJhyJhuoJ3gFZdAHF7NM= | ||||||
| github.com/bytedance/sonic v1.10.0-rc2 h1:oDfRZ+4m6AYCOC0GFeOCeYqvBmucy1isvouS2K0cPzo= | github.com/bytedance/sonic v1.10.0-rc2 h1:oDfRZ+4m6AYCOC0GFeOCeYqvBmucy1isvouS2K0cPzo= | ||||||
| github.com/bytedance/sonic v1.10.0-rc2/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | github.com/bytedance/sonic v1.10.0-rc2/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | ||||||
|  | github.com/bytedance/sonic v1.10.0-rc3 h1:uNSnscRapXTwUgTyOF0GVljYD08p9X/Lbr9MweSV3V0= | ||||||
|  | github.com/bytedance/sonic v1.10.0-rc3/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | ||||||
|  | github.com/bytedance/sonic v1.10.0 h1:qtNZduETEIWJVIyDl01BeNxur2rW9OwTQ/yBqFRkKEk= | ||||||
|  | github.com/bytedance/sonic v1.10.0/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= | ||||||
| github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= | github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= | ||||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= | github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= | ||||||
| github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d h1:77cEq6EriyTZ0g/qfRdp61a3Uu/AWrgIq2s0ClJV1g0= | github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d h1:77cEq6EriyTZ0g/qfRdp61a3Uu/AWrgIq2s0ClJV1g0= | ||||||
| @@ -24,6 +28,8 @@ github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJn | |||||||
| github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | ||||||
| github.com/go-playground/validator/v10 v10.14.1 h1:9c50NUPC30zyuKprjL3vNZ0m5oG+jU0zvx4AqHGnv4k= | github.com/go-playground/validator/v10 v10.14.1 h1:9c50NUPC30zyuKprjL3vNZ0m5oG+jU0zvx4AqHGnv4k= | ||||||
| github.com/go-playground/validator/v10 v10.14.1/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU= | github.com/go-playground/validator/v10 v10.14.1/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU= | ||||||
|  | github.com/go-playground/validator/v10 v10.15.0 h1:nDU5XeOKtB3GEa+uB7GNYwhVKsgjAR7VgKoNB6ryXfw= | ||||||
|  | github.com/go-playground/validator/v10 v10.15.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU= | ||||||
| github.com/go-sql-driver/mysql v1.6.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= | github.com/go-sql-driver/mysql v1.6.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= | ||||||
| github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU= | github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU= | ||||||
| github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= | github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= | ||||||
| @@ -79,10 +85,14 @@ 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= | ||||||
| github.com/rs/zerolog v1.29.1/go.mod h1:Le6ESbR7hc+DP6Lt1THiV8CQSdkkNrd3R0XbEgp3ZBU= | github.com/rs/zerolog v1.29.1/go.mod h1:Le6ESbR7hc+DP6Lt1THiV8CQSdkkNrd3R0XbEgp3ZBU= | ||||||
|  | github.com/rs/zerolog v1.30.0 h1:SymVODrcRsaRaSInD9yQtKbtWqwsfoPcRff/oRXLj4c= | ||||||
|  | github.com/rs/zerolog v1.30.0/go.mod h1:/tk+P47gFdPXq4QYjvCmT5/Gsug2nagsFWBWhAiSi1w= | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
| github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | ||||||
| github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= | github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= | ||||||
| @@ -118,6 +128,8 @@ go.mongodb.org/mongo-driver v1.11.2 h1:+1v2rDQUWNcGW7/7E0Jvdz51V38XXxJfhzbV17aNH | |||||||
| go.mongodb.org/mongo-driver v1.11.2/go.mod h1:s7p5vEtfbeR1gYi6pnj3c3/urpbLv2T5Sfd6Rp2HBB8= | go.mongodb.org/mongo-driver v1.11.2/go.mod h1:s7p5vEtfbeR1gYi6pnj3c3/urpbLv2T5Sfd6Rp2HBB8= | ||||||
| go.mongodb.org/mongo-driver v1.12.0 h1:aPx33jmn/rQuJXPQLZQ8NtfPQG8CaqgLThFtqRb0PiE= | go.mongodb.org/mongo-driver v1.12.0 h1:aPx33jmn/rQuJXPQLZQ8NtfPQG8CaqgLThFtqRb0PiE= | ||||||
| go.mongodb.org/mongo-driver v1.12.0/go.mod h1:AZkxhPnFJUoH7kZlFkVKucV20K387miPfm7oimrSmK0= | go.mongodb.org/mongo-driver v1.12.0/go.mod h1:AZkxhPnFJUoH7kZlFkVKucV20K387miPfm7oimrSmK0= | ||||||
|  | go.mongodb.org/mongo-driver v1.12.1 h1:nLkghSU8fQNaK7oUmDhQFsnrtcoNy7Z6LVFKsEecqgE= | ||||||
|  | go.mongodb.org/mongo-driver v1.12.1/go.mod h1:/rGBTebI3XYboVmgz+Wv3Bcbl3aD0QF9zl6kDDw18rQ= | ||||||
| golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||||
| golang.org/x/arch v0.4.0 h1:A8WCeEWhLwPBKNbFi5Wv5UTCBx5zzubnXDlMOFAzFMc= | golang.org/x/arch v0.4.0 h1:A8WCeEWhLwPBKNbFi5Wv5UTCBx5zzubnXDlMOFAzFMc= | ||||||
| golang.org/x/arch v0.4.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | golang.org/x/arch v0.4.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||||
| @@ -129,6 +141,8 @@ golang.org/x/crypto v0.4.0 h1:UVQgzMY87xqpKNgb+kDsll2Igd33HszWHFLmpaRMq/8= | |||||||
| golang.org/x/crypto v0.4.0/go.mod h1:3quD/ATkf6oY+rnes5c3ExXTbLc8mueNue5/DoinL80= | golang.org/x/crypto v0.4.0/go.mod h1:3quD/ATkf6oY+rnes5c3ExXTbLc8mueNue5/DoinL80= | ||||||
| golang.org/x/crypto v0.11.0 h1:6Ewdq3tDic1mg5xRO4milcWCfMVQhI4NkqWWvqejpuA= | golang.org/x/crypto v0.11.0 h1:6Ewdq3tDic1mg5xRO4milcWCfMVQhI4NkqWWvqejpuA= | ||||||
| golang.org/x/crypto v0.11.0/go.mod h1:xgJhtzW8F9jGdVFWZESrid1U1bjeNy4zgy5cRr/CIio= | golang.org/x/crypto v0.11.0/go.mod h1:xgJhtzW8F9jGdVFWZESrid1U1bjeNy4zgy5cRr/CIio= | ||||||
|  | golang.org/x/crypto v0.12.0 h1:tFM/ta59kqch6LlvYnPa0yx5a83cL2nHflFhYKvv9Yk= | ||||||
|  | golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw= | ||||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | ||||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||||
| @@ -137,6 +151,10 @@ golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qx | |||||||
| golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= | golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= | ||||||
| golang.org/x/net v0.12.0 h1:cfawfvKITfUsFCeJIHJrbSxpeu/E81khclypR0GVT50= | golang.org/x/net v0.12.0 h1:cfawfvKITfUsFCeJIHJrbSxpeu/E81khclypR0GVT50= | ||||||
| golang.org/x/net v0.12.0/go.mod h1:zEVYFnQC7m/vmpQFELhcD1EWkZlX69l4oqgmer6hfKA= | golang.org/x/net v0.12.0/go.mod h1:zEVYFnQC7m/vmpQFELhcD1EWkZlX69l4oqgmer6hfKA= | ||||||
|  | golang.org/x/net v0.13.0 h1:Nvo8UFsZ8X3BhAC9699Z1j7XQ3rsZnUUm7jfBEk1ueY= | ||||||
|  | golang.org/x/net v0.13.0/go.mod h1:zEVYFnQC7m/vmpQFELhcD1EWkZlX69l4oqgmer6hfKA= | ||||||
|  | golang.org/x/net v0.14.0 h1:BONx9s002vGdD9umnlX1Po8vOZmrgH34qlHcD1MfK14= | ||||||
|  | golang.org/x/net v0.14.0/go.mod h1:PpSgVXXLK0OxS0F31C1/tv6XNguvCrnXIDrFMspZIUI= | ||||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= | golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= | ||||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| @@ -162,6 +180,8 @@ golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | |||||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.10.0 h1:SqMFp9UcQJZa+pmYuAKjd9xq1f0j5rLcDIk0mj4qAsA= | golang.org/x/sys v0.10.0 h1:SqMFp9UcQJZa+pmYuAKjd9xq1f0j5rLcDIk0mj4qAsA= | ||||||
| golang.org/x/sys v0.10.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.10.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
|  | golang.org/x/sys v0.11.0 h1:eG7RXZHdqOJ1i+0lgLgCpSXAp6M3LYlAo6osgSi0xOM= | ||||||
|  | golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||||
| golang.org/x/term v0.1.0 h1:g6Z6vPFA9dYBAF7DWcH6sCcOntplXsDKcliusYijMlw= | golang.org/x/term v0.1.0 h1:g6Z6vPFA9dYBAF7DWcH6sCcOntplXsDKcliusYijMlw= | ||||||
| @@ -170,6 +190,8 @@ golang.org/x/term v0.3.0 h1:qoo4akIqOcDME5bhc/NgxUdovd6BSS2uMsVjB56q1xI= | |||||||
| golang.org/x/term v0.3.0/go.mod h1:q750SLmJuPmVoN1blW3UFBPREJfb1KmY3vwxfr+nFDA= | golang.org/x/term v0.3.0/go.mod h1:q750SLmJuPmVoN1blW3UFBPREJfb1KmY3vwxfr+nFDA= | ||||||
| golang.org/x/term v0.10.0 h1:3R7pNqamzBraeqj/Tj8qt1aQ2HpmlC+Cx/qL/7hn4/c= | golang.org/x/term v0.10.0 h1:3R7pNqamzBraeqj/Tj8qt1aQ2HpmlC+Cx/qL/7hn4/c= | ||||||
| golang.org/x/term v0.10.0/go.mod h1:lpqdcUyK/oCiQxvxVrppt5ggO2KCZ5QblwqPnfZ6d5o= | golang.org/x/term v0.10.0/go.mod h1:lpqdcUyK/oCiQxvxVrppt5ggO2KCZ5QblwqPnfZ6d5o= | ||||||
|  | golang.org/x/term v0.11.0 h1:F9tnn/DA/Im8nCwm+fX+1/eBwi4qFjRT++MhtVC4ZX0= | ||||||
|  | golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU= | ||||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| @@ -180,6 +202,8 @@ golang.org/x/text v0.5.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | |||||||
| golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | ||||||
| golang.org/x/text v0.11.0 h1:LAntKIrcmeSKERyiOh0XMV39LXS8IE9UL2yP7+f5ij4= | golang.org/x/text v0.11.0 h1:LAntKIrcmeSKERyiOh0XMV39LXS8IE9UL2yP7+f5ij4= | ||||||
| golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= | golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= | ||||||
|  | golang.org/x/text v0.12.0 h1:k+n5B8goJNdU7hSvEtMUz3d1Q6D/XW4COJSJR6fN0mc= | ||||||
|  | golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= | ||||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||||
| golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| package goext | package goext | ||||||
|  |  | ||||||
| const GoextVersion = "0.0.184" | const GoextVersion = "0.0.240" | ||||||
|  |  | ||||||
| const GoextVersionTimestamp = "2023-07-19T19:29:59+0200" | const GoextVersionTimestamp = "2023-08-14T15:36:12+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 | ||||||
|  | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import ( | |||||||
|  |  | ||||||
| func TestGroupByNameOrEmpty1(t *testing.T) { | func TestGroupByNameOrEmpty1(t *testing.T) { | ||||||
|  |  | ||||||
| 	regex1 := W(regexp.MustCompile("0(?P<group1>A+)B(?P<group2>C+)0")) | 	regex1 := W(regexp.MustCompile(`0(?P<group1>A+)B(?P<group2>C+)0`)) | ||||||
|  |  | ||||||
| 	match1, ok1 := regex1.MatchFirst("Hello 0AAAABCCC0 Bye.") | 	match1, ok1 := regex1.MatchFirst("Hello 0AAAABCCC0 Bye.") | ||||||
|  |  | ||||||
| @@ -26,7 +26,7 @@ func TestGroupByNameOrEmpty1(t *testing.T) { | |||||||
|  |  | ||||||
| func TestGroupByNameOrEmpty2(t *testing.T) { | func TestGroupByNameOrEmpty2(t *testing.T) { | ||||||
|  |  | ||||||
| 	regex1 := W(regexp.MustCompile("0(?P<group1>A+)B(?P<group2>C+)(?P<group3>C+)?0")) | 	regex1 := W(regexp.MustCompile(`0(?P<group1>A+)B(?P<group2>C+)(?P<group3>C+)?0`)) | ||||||
|  |  | ||||||
| 	match1, ok1 := regex1.MatchFirst("Hello 0AAAABCCC0 Bye.") | 	match1, ok1 := regex1.MatchFirst("Hello 0AAAABCCC0 Bye.") | ||||||
|  |  | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ func HashSqliteSchema(ctx context.Context, schemaStr string) (string, error) { | |||||||
| 	return HashSqliteDatabase(ctx, db) | 	return HashSqliteDatabase(ctx, db) | ||||||
| } | } | ||||||
|  |  | ||||||
| func HashSqliteDatabase(ctx context.Context, db DB) (string, error) { | func HashSqliteDatabase(ctx context.Context, db Queryable) (string, error) { | ||||||
| 	ss, err := CreateSqliteDatabaseSchemaString(ctx, db) | 	ss, err := CreateSqliteDatabaseSchemaString(ctx, db) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| @@ -50,7 +50,7 @@ func HashSqliteDatabase(ctx context.Context, db DB) (string, error) { | |||||||
| 	return hex.EncodeToString(cs[:]), nil | 	return hex.EncodeToString(cs[:]), nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func CreateSqliteDatabaseSchemaString(ctx context.Context, db DB) (string, error) { | func CreateSqliteDatabaseSchemaString(ctx context.Context, db Queryable) (string, error) { | ||||||
|  |  | ||||||
| 	type colInfo struct { | 	type colInfo struct { | ||||||
| 		Name       string  `db:"name"` | 		Name       string  `db:"name"` | ||||||
|   | |||||||
| @@ -7,9 +7,19 @@ import ( | |||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | type TxStatus string | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	TxStatusInitial  TxStatus = "INITIAL" | ||||||
|  | 	TxStatusActive   TxStatus = "ACTIVE" | ||||||
|  | 	TxStatusComitted TxStatus = "COMMITTED" | ||||||
|  | 	TxStatusRollback TxStatus = "ROLLBACK" | ||||||
|  | ) | ||||||
|  |  | ||||||
| type Tx interface { | type Tx interface { | ||||||
| 	Rollback() error | 	Rollback() error | ||||||
| 	Commit() error | 	Commit() error | ||||||
|  | 	Status() TxStatus | ||||||
| 	Exec(ctx context.Context, sql string, prep PP) (sql.Result, error) | 	Exec(ctx context.Context, sql string, prep PP) (sql.Result, error) | ||||||
| 	Query(ctx context.Context, sql string, prep PP) (*sqlx.Rows, error) | 	Query(ctx context.Context, sql string, prep PP) (*sqlx.Rows, error) | ||||||
| } | } | ||||||
| @@ -18,6 +28,9 @@ type transaction struct { | |||||||
| 	tx       *sqlx.Tx | 	tx       *sqlx.Tx | ||||||
| 	id       uint16 | 	id       uint16 | ||||||
| 	lstr     []Listener | 	lstr     []Listener | ||||||
|  | 	status   TxStatus | ||||||
|  | 	execCtr  int | ||||||
|  | 	queryCtr int | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewTransaction(xtx *sqlx.Tx, txid uint16, lstr []Listener) Tx { | func NewTransaction(xtx *sqlx.Tx, txid uint16, lstr []Listener) Tx { | ||||||
| @@ -25,6 +38,9 @@ func NewTransaction(xtx *sqlx.Tx, txid uint16, lstr []Listener) Tx { | |||||||
| 		tx:       xtx, | 		tx:       xtx, | ||||||
| 		id:       txid, | 		id:       txid, | ||||||
| 		lstr:     lstr, | 		lstr:     lstr, | ||||||
|  | 		status:   TxStatusInitial, | ||||||
|  | 		execCtr:  0, | ||||||
|  | 		queryCtr: 0, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -38,6 +54,10 @@ func (tx *transaction) Rollback() error { | |||||||
|  |  | ||||||
| 	result := tx.tx.Rollback() | 	result := tx.tx.Rollback() | ||||||
|  |  | ||||||
|  | 	if result == nil { | ||||||
|  | 		tx.status = TxStatusRollback | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	for _, v := range tx.lstr { | 	for _, v := range tx.lstr { | ||||||
| 		v.PostTxRollback(tx.id, result) | 		v.PostTxRollback(tx.id, result) | ||||||
| 	} | 	} | ||||||
| @@ -55,6 +75,10 @@ func (tx *transaction) Commit() error { | |||||||
|  |  | ||||||
| 	result := tx.tx.Commit() | 	result := tx.tx.Commit() | ||||||
|  |  | ||||||
|  | 	if result == nil { | ||||||
|  | 		tx.status = TxStatusComitted | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	for _, v := range tx.lstr { | 	for _, v := range tx.lstr { | ||||||
| 		v.PostTxRollback(tx.id, result) | 		v.PostTxRollback(tx.id, result) | ||||||
| 	} | 	} | ||||||
| @@ -73,6 +97,10 @@ func (tx *transaction) Exec(ctx context.Context, sqlstr string, prep PP) (sql.Re | |||||||
|  |  | ||||||
| 	res, err := tx.tx.NamedExecContext(ctx, sqlstr, prep) | 	res, err := tx.tx.NamedExecContext(ctx, sqlstr, prep) | ||||||
|  |  | ||||||
|  | 	if tx.status == TxStatusInitial && err == nil { | ||||||
|  | 		tx.status = TxStatusActive | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	for _, v := range tx.lstr { | 	for _, v := range tx.lstr { | ||||||
| 		v.PostExec(langext.Ptr(tx.id), origsql, sqlstr, prep) | 		v.PostExec(langext.Ptr(tx.id), origsql, sqlstr, prep) | ||||||
| 	} | 	} | ||||||
| @@ -94,6 +122,10 @@ func (tx *transaction) Query(ctx context.Context, sqlstr string, prep PP) (*sqlx | |||||||
|  |  | ||||||
| 	rows, err := sqlx.NamedQueryContext(ctx, tx.tx, sqlstr, prep) | 	rows, err := sqlx.NamedQueryContext(ctx, tx.tx, sqlstr, prep) | ||||||
|  |  | ||||||
|  | 	if tx.status == TxStatusInitial && err == nil { | ||||||
|  | 		tx.status = TxStatusActive | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	for _, v := range tx.lstr { | 	for _, v := range tx.lstr { | ||||||
| 		v.PostQuery(langext.Ptr(tx.id), origsql, sqlstr, prep) | 		v.PostQuery(langext.Ptr(tx.id), origsql, sqlstr, prep) | ||||||
| 	} | 	} | ||||||
| @@ -103,3 +135,11 @@ func (tx *transaction) Query(ctx context.Context, sqlstr string, prep PP) (*sqlx | |||||||
| 	} | 	} | ||||||
| 	return rows, nil | 	return rows, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (tx *transaction) Status() TxStatus { | ||||||
|  | 	return tx.status | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (tx *transaction) Traffic() (int, int) { | ||||||
|  | 	return tx.execCtr, tx.queryCtr | ||||||
|  | } | ||||||
|   | |||||||
| @@ -71,12 +71,12 @@ func SupportsColors() bool { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var term256Regex = regexp.MustCompile("(?i)-256(color)?$") | 	var term256Regex = regexp.MustCompile(`(?i)-256(color)?$`) | ||||||
| 	if term256Regex.MatchString(termenv) { | 	if term256Regex.MatchString(termenv) { | ||||||
| 		return true | 		return true | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var termBasicRegex = regexp.MustCompile("(?i)^screen|^xterm|^vt100|^vt220|^rxvt|color|ansi|cygwin|linux") | 	var termBasicRegex = regexp.MustCompile(`(?i)^screen|^xterm|^vt100|^vt220|^rxvt|color|ansi|cygwin|linux`) | ||||||
|  |  | ||||||
| 	if termBasicRegex.MatchString(termenv) { | 	if termBasicRegex.MatchString(termenv) { | ||||||
| 		return true | 		return true | ||||||
|   | |||||||
| @@ -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