Compare commits

...

22 commits
v0.0.1 ... main

Author SHA1 Message Date
55ed5598e0 Added godoc 2024-09-27 09:41:07 +02:00
0570e1925b Better example 2024-09-27 07:43:50 +02:00
616835f641 Added NoTrace function and removed WrapData, NewData since WithData exist. 2024-09-27 07:42:30 +02:00
1154fd39a2 Explicit log callback 2024-04-05 08:30:30 +02:00
c66ba0e169 Different format of error string 2024-03-29 11:16:11 +01:00
bc020b2830 Added error codes 2024-03-29 11:14:01 +01:00
16d41de1ae Added error string and better data default 2024-03-29 09:38:32 +01:00
776917b9c0 Wrapping nil returns nil 2023-09-27 07:32:50 +02:00
ec5422f335 More flexible parameters to New functions 2023-09-26 15:01:29 +02:00
6fb67141fe gofmt 2023-09-26 09:37:09 +02:00
ebf3a9aa38 Made wrapped error accessible 2023-09-26 09:33:46 +02:00
35249137c5 Added data to wrapped error 2023-09-26 09:31:18 +02:00
b77f5b94c8 Inline documentation 2023-09-26 09:17:17 +02:00
a1602d21d9 Stack test 2023-09-26 09:12:57 +02:00
32fcdb62fc Stack test 2023-09-26 09:12:16 +02:00
c15e592827 Stack test 2023-09-26 09:11:10 +02:00
a2c18bcc50 Stack test 2023-09-26 09:05:27 +02:00
031e85ed62 Stack test 2023-09-26 09:04:40 +02:00
806f31b69d Stack test 2023-09-26 09:03:46 +02:00
1c0579d065 Added Init() 2023-09-26 08:54:26 +02:00
015167f4a3 caller test 2023-09-26 08:50:57 +02:00
c73070d69a caller test 2023-09-26 08:49:08 +02:00
2 changed files with 160 additions and 40 deletions

View file

@ -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()
}
```

156
pkg.go
View file

@ -1,34 +1,93 @@
/*
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
)
func init() {
_, file, _, _ := runtime.Caller(0)
fmt.Printf("--\nWRAPPED_ERROR: %s\n--\n", file)
dir := path.Dir(file)
dirBase := path.Dir(dir)
// Init only works if called from the main package and sets the length of code base path
// to later be removed in file paths to receive relative paths.
func Init() {
_, file, _, _ := runtime.Caller(1)
dirBase := path.Dir(file)
baseDirLength = len(dirBase)
}
// SetLogCallback gives a possibility to automatically run code to handle any errors.
func SetLogCallback(cbk LogCallback) {
logCallback = cbk
}
@ -39,35 +98,76 @@ 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
}
// Wrap wraps an existing error with file and line.
func Wrap(err error) *Error {
return create(err, nil)
}
// New creates a new wrapped error with file and line.
func New(msg string, params ...any) *Error {
err := fmt.Errorf(msg, params...)
wrapped := create(err, nil)
return wrapped
}
func Wrap(err error) error {
return create(err)
// WithCode associates a string code with the error.
func (e *Error) WithCode(code string) CodableError {
e.Code = code
return e
}
func New(msg string) error {
wrapped := create(errors.New(msg))
return wrapped
// 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(""),
),
)
}