Compare commits
12 commits
Author | SHA1 | Date | |
---|---|---|---|
55ed5598e0 | |||
0570e1925b | |||
616835f641 | |||
1154fd39a2 | |||
c66ba0e169 | |||
bc020b2830 | |||
16d41de1ae | |||
776917b9c0 | |||
ec5422f335 | |||
6fb67141fe | |||
ebf3a9aa38 | |||
35249137c5 |
2 changed files with 151 additions and 35 deletions
44
README.md
44
README.md
|
@ -2,25 +2,45 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
// External
|
||||
werr "git.gibonuddevalla.se/go/wrappederror"
|
||||
// External
|
||||
werr "git.gibonuddevalla.se/go/wrappederror"
|
||||
|
||||
// Standard
|
||||
"errors"
|
||||
"fmt"
|
||||
// Standard
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"os"
|
||||
)
|
||||
|
||||
func errorHandler(err werr.Error) {
|
||||
// For example print or log error to file
|
||||
fmt.Printf("ERROR - %s\n", err)
|
||||
// For example print or log error to file
|
||||
fmt.Printf("\x1b[31;1m%s\x1b[0m\n", err)
|
||||
fmt.Printf("\x1b[33;1m%s\x1b[0m\n", err.NoTrace())
|
||||
|
||||
j, _ := json.MarshalIndent(err, "", " ")
|
||||
fmt.Printf("%s\n\n", j)
|
||||
}
|
||||
|
||||
func foo() {
|
||||
werr.SetLogCallback(errorHandler)
|
||||
func main() {
|
||||
// Make file paths relative to this file.
|
||||
werr.Init()
|
||||
|
||||
err := errors.New("foobar 1")
|
||||
err1 := werr.Wrap(err)
|
||||
// Handler to call when using Log().
|
||||
werr.SetLogCallback(errorHandler)
|
||||
|
||||
err2 := werr.New("foobar 2")
|
||||
// Wrap an existing error.
|
||||
err := errors.New("foobar 1")
|
||||
err1 := werr.Wrap(err)
|
||||
|
||||
// Create a new error with extra information.
|
||||
err2 := werr.New("foobar 2").WithCode("FOO-100").WithData(137)
|
||||
|
||||
// The os error contains more information.
|
||||
_, errOS := os.ReadFile("/tmp/does_not_exist")
|
||||
werr.Wrap(errOS).Log()
|
||||
|
||||
// Log the previously wrapped errors.
|
||||
werr.Wrap(err1).Log()
|
||||
werr.Wrap(err2).Log()
|
||||
}
|
||||
```
|
||||
|
|
142
pkg.go
142
pkg.go
|
@ -1,23 +1,81 @@
|
|||
/*
|
||||
wrappederror provides traceable errors:
|
||||
|
||||
package main
|
||||
|
||||
import (
|
||||
// External
|
||||
werr "git.gibonuddevalla.se/go/wrappederror"
|
||||
|
||||
// Standard
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"os"
|
||||
)
|
||||
|
||||
func errorHandler(err werr.Error) {
|
||||
// For example print or log error to file
|
||||
fmt.Printf("\x1b[31;1m%s\x1b[0m\n", err)
|
||||
fmt.Printf("\x1b[33;1m%s\x1b[0m\n", err.NoTrace())
|
||||
|
||||
j, _ := json.MarshalIndent(err, "", " ")
|
||||
fmt.Printf("%s\n\n", j)
|
||||
}
|
||||
|
||||
func main() {
|
||||
// Make file paths relative to this file.
|
||||
werr.Init()
|
||||
|
||||
// Handler to call when using Log().
|
||||
werr.SetLogCallback(errorHandler)
|
||||
|
||||
// Wrap an existing error.
|
||||
err := errors.New("foobar 1")
|
||||
err1 := werr.Wrap(err)
|
||||
|
||||
// Create a new error with extra information.
|
||||
err2 := werr.New("foobar 2").WithCode("FOO-100").WithData(137)
|
||||
|
||||
// The os error contains more information.
|
||||
_, errOS := os.ReadFile("/tmp/does_not_exist")
|
||||
werr.Wrap(errOS).Log()
|
||||
|
||||
// Log the previously wrapped errors.
|
||||
werr.Wrap(err1).Log()
|
||||
werr.Wrap(err2).Log()
|
||||
}
|
||||
*/
|
||||
package WrappedError
|
||||
|
||||
import (
|
||||
// Standard
|
||||
"errors"
|
||||
"fmt"
|
||||
"path"
|
||||
"regexp"
|
||||
"runtime"
|
||||
)
|
||||
|
||||
type Error struct {
|
||||
err error
|
||||
file string
|
||||
line int
|
||||
Wrapped error
|
||||
ErrStr string // wrapped error isn't necessarily json encodable
|
||||
Code string
|
||||
File string
|
||||
Line int
|
||||
Data any
|
||||
}
|
||||
|
||||
type CodableError interface {
|
||||
error
|
||||
WithCode(string) CodableError
|
||||
WithData(any) CodableError
|
||||
Log() CodableError
|
||||
}
|
||||
|
||||
type LogCallback func(Error)
|
||||
|
||||
var (
|
||||
logCallback LogCallback
|
||||
logCallback LogCallback
|
||||
baseDirLength int
|
||||
)
|
||||
|
||||
|
@ -42,36 +100,74 @@ func callback(wrapped Error) {
|
|||
|
||||
// Error implements the error inteface and adds filename and line to the error.
|
||||
func (wrapped Error) Error() string {
|
||||
var code string
|
||||
if wrapped.Code != "" {
|
||||
code = wrapped.Code + ":"
|
||||
}
|
||||
return fmt.Sprintf(
|
||||
"[%s:%d] %s",
|
||||
wrapped.file,
|
||||
wrapped.line,
|
||||
wrapped.err.Error(),
|
||||
"[%s%s:%d] %s",
|
||||
code,
|
||||
wrapped.File,
|
||||
wrapped.Line,
|
||||
wrapped.Wrapped.Error(),
|
||||
)
|
||||
}
|
||||
|
||||
func create(err error) error {
|
||||
_, file, line, _ := runtime.Caller(2)
|
||||
file = file[baseDirLength+1:]
|
||||
|
||||
wrapped := Error{
|
||||
err: err,
|
||||
file: file,
|
||||
line: line,
|
||||
func create(err error, data interface{}) *Error {
|
||||
if err == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
callback(wrapped)
|
||||
_, file, line, _ := runtime.Caller(2)
|
||||
file = file[baseDirLength+1:]
|
||||
wrapped := Error{
|
||||
Wrapped: err,
|
||||
ErrStr: err.Error(),
|
||||
File: file,
|
||||
Line: line,
|
||||
Data: data,
|
||||
}
|
||||
|
||||
return wrapped
|
||||
return &wrapped
|
||||
}
|
||||
|
||||
// Wrap wraps an existing error with file and line.
|
||||
func Wrap(err error) error {
|
||||
return create(err)
|
||||
func Wrap(err error) *Error {
|
||||
return create(err, nil)
|
||||
}
|
||||
|
||||
// New creates a new wrapped error with file and line.
|
||||
func New(msg string) error {
|
||||
wrapped := create(errors.New(msg))
|
||||
func New(msg string, params ...any) *Error {
|
||||
err := fmt.Errorf(msg, params...)
|
||||
wrapped := create(err, nil)
|
||||
return wrapped
|
||||
}
|
||||
|
||||
// WithCode associates a string code with the error.
|
||||
func (e *Error) WithCode(code string) CodableError {
|
||||
e.Code = code
|
||||
return e
|
||||
}
|
||||
|
||||
// WithData associates any data with the error to make troubleshooting easier.
|
||||
func (e *Error) WithData(data any) CodableError {
|
||||
e.Data = data
|
||||
return e
|
||||
}
|
||||
|
||||
// Log calls the log callback.
|
||||
func (e *Error) Log() CodableError {
|
||||
callback(*e)
|
||||
return e
|
||||
}
|
||||
|
||||
// NoTrace returns the Error() string without the trace.
|
||||
func (e *Error) NoTrace() string {
|
||||
rxp := regexp.MustCompile(`^(\[[^\]]+\.go:\d+\]\s*)*`)
|
||||
return string(
|
||||
rxp.ReplaceAll(
|
||||
[]byte(e.Error()),
|
||||
[]byte(""),
|
||||
),
|
||||
)
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue