Survive under the Crap Go Error System
===
<!-- .slide: data-background-color="pink" -->
<!-- .slide: data-transition="zoom" -->
> [name=郭學聰 Hsueh-Tsung Kuo]
> [time=Sat, 14 Nov 2020] [color=red]
###### CC BY-SA 4.0
---
<!-- .slide: data-transition="convex" -->
## Who am I?
![fieliapm](https://www.gravatar.com/avatar/2aef78f04240a6ac9ccd473ba1cbd1e3?size=2048 =384x384)
<small>Someone (who?) said:
a game programmer should be able to draw cute anime character(?)</small>
----
<!-- .slide: data-transition="convex" -->
* A programmer from game company in Taiwan
* Backend (and temporary frontend) engineer
* Usually develop something related to my work in Python, Ruby, ECMAScript, Golang, C#
* ECMAScript hater since **Netscape** is dead
* Built CDN-aware game asset update system
* Professional small vehicle driver
* Draw cute anime character in spare time
---
<!-- .slide: data-transition="convex" -->
## Outline
----
<!-- .slide: data-transition="convex" -->
4. Error Usage
5. Error Design Flaw
* Error 5 Ws
* Go Error System
6. Package
* errors
* golang.org/x/xerrors
* github.com/pkg/errors
* Integration
----
<!-- .slide: data-transition="convex" -->
7. Advice
* Wrapping
* Inspection
* Displaying
* Sum Up
8. One More Thing
9. Conclusion
10. Reference
11. Q&A
---
<!-- .slide: data-transition="convex" -->
## Error Usage
----
<!-- .slide: data-transition="convex" -->
### Typical Example
```go=
func MyFunc(param1 int, param2 string) (string, error) {
// do something...
f, err := os.Open("filename.ext")
if err != nil {
return "", err
}
// do something...
return result, nil
}
```
---
<!-- .slide: data-transition="convex" -->
## Error Design Flaw
----
<!-- .slide: data-transition="convex" -->
### Error 5 Ws
* What
* Error title
* Why
* Error description
* Which
* Error type
* When
* Line number
* Where
* Call stack
----
<!-- .slide: data-transition="convex" -->
#### Who Need to Know What
* End-User
* What
* Why
* Package Importer
* Which
* Package Developer & Maintainer
* When
* Where
----
<!-- .slide: data-transition="convex" -->
### :thumbsdown: Go Error System
* [x] Error title
* [x] Error description
* [ ] Error type
* [ ] Line number
* [ ] Call stack
==Why?== <!-- .element: class="fragment" data-fragment-index="1" -->
----
<!-- .slide: data-transition="convex" -->
#### Review Typical Example
```go=
func MyFunc(param1 int, param2 string) (string, error) {
// do something...
f, err := os.Open("filename.ext")
if err != nil {
return "", err
}
// do something...
return result, nil
}
```
----
<!-- .slide: data-transition="convex" -->
#### Pitfalls
```go=
ErrSomething = errors.New("error description")
return ErrSomething
err := fmt.Errorf("error %s", detail)
return err
```
----
<!-- .slide: data-transition="convex" -->
#### Darkness
* How to classify built-in errors before Go 1.13
* Unreliable error message
* Useless error type (in most cases)
* Nothing further :face_palm:
----
<!-- .slide: data-transition="convex" -->
#### Dawn?
* 3rd party error module
* Go 1.13 error system
---
<!-- .slide: data-transition="convex" -->
## Package
----
<!-- .slide: data-transition="convex" -->
### errors
* :no_entry_sign: Go 1.13 error system
----
<!-- .slide: data-transition="convex" -->
#### Wrap & Unwrap
```go=
// wrap error
err = fmt.Errorf("... %w ...", ..., cause, ...)
// unwrap and return NEXT error, or nil
cause = errors.Unwrap(err)
```
----
<!-- .slide: data-transition="convex" -->
#### Inspect Error Chain
```go=
errors.Is(err, ErrorInstance) // must be same address
var perr *ErrorType
errors.As(err, &perr)
```
----
<!-- .slide: data-transition="convex" -->
#### Example
```go=
package main
import (
"errors"
"fmt"
)
type MyError struct {
msg string
err error
}
func NewMyError(msg string, err error) *MyError {
if err != nil {
msg = fmt.Sprintf("%s: %s", msg, err.Error())
}
return &MyError{
msg: msg,
err: err,
}
}
func (e *MyError) Error() string {
return e.msg
}
func (e *MyError) Unwrap() error {
return e.err
}
func main() {
var err error
err = errors.New("err1")
err = fmt.Errorf("err2: %w", err)
myErr := NewMyError("my err3", err)
myErr2 := NewMyError("my err3", err)
err = myErr
err = fmt.Errorf("err4: %w", err)
err = fmt.Errorf("err5: %w", err)
if err != nil {
fmt.Println("error is:", err)
if errors.Is(err, myErr) {
fmt.Println("it is myErr!")
}
if !errors.Is(err, myErr2) {
fmt.Println("it is NOT myErr2!")
}
fmt.Println("")
var myError *MyError
if errors.As(err, &myError) {
fmt.Println("failed:", err.Error())
fmt.Println("MyError:", myError.Error())
}
}
}
```
----
<!-- .slide: data-transition="convex" -->
#### Result
```shell
error is: err5: err4: my err3: err2: err1
it is myErr!
it is NOT myErr2!
failed: err5: err4: my err3: err2: err1
MyError: my err3: err2: err1
```
----
<!-- .slide: data-transition="convex" -->
### golang.org/x/xerrors
* :no_entry_sign: golang.org/x/xerrors by Go team members
* https://github.com/golang/xerrors
* https://godoc.org/golang.org/x/xerrors
----
<!-- .slide: data-transition="convex" -->
#### Additional Features
* Frame
* New & Wrap
* Attach current function & line number
* Inspect
* Print current function & line number
----
<!-- .slide: data-transition="convex" -->
#### Usage
```go=
// new error
err = xerrors.New("...")
// wrap error
err = xerrors.Errorf("... %w ...", ..., cause, ...)
// print error chain and frames
fmt.Printf("%+v\n", err)
```
----
<!-- .slide: data-transition="convex" -->
#### Example
```go=
package main
import (
"fmt"
"golang.org/x/xerrors"
)
func MyFuncInner() error {
return xerrors.New("inner error")
}
func MyFuncMiddle() error {
err := MyFuncInner()
return xerrors.Errorf("middle error: %w", err)
}
func MyFuncOuter() error {
err := MyFuncMiddle()
return xerrors.Errorf("outer error: %w", err)
}
func main() {
err := MyFuncOuter()
fmt.Printf("%v\n\n", err)
fmt.Printf("%+v\n", err)
}
```
----
<!-- .slide: data-transition="convex" -->
#### Result
```shell
outer error: middle error: inner error
outer error:
main.MyFuncOuter
/.../go/src/survive_under_the_crap_go_error_system/xerrors_example.go:20
- middle error:
main.MyFuncMiddle
/.../go/src/survive_under_the_crap_go_error_system/xerrors_example.go:15
- inner error:
main.MyFuncInner
/.../go/src/survive_under_the_crap_go_error_system/xerrors_example.go:10
```
----
<!-- .slide: data-transition="convex" -->
### github.com/pkg/errors
* :thumbsup: github.com/pkg/errors by Dave Cheney
* https://github.com/pkg/errors
* https://godoc.org/github.com/pkg/errors
----
<!-- .slide: data-transition="convex" -->
#### Wrap & Unwrap
```go=
// new error
err = pkg_errors.New("...")
// wrap error
err = pkg_errors.Wrap(cause, "...")
err = pkg_errors.Wrapf(cause, "oh noes #%d", 2)
// unwrap the whole error chain and return the MOST INNER error, or return the error itself
cause = pkg_errors.Cause(err)
```
----
<!-- .slide: data-transition="convex" -->
#### Inspect Error Chain
```go=
fmt.Printf("%+v\n", err)
```
----
<!-- .slide: data-transition="convex" -->
#### Example
```go=
package main
import (
"fmt"
pkg_errors "github.com/pkg/errors"
)
func MyFuncInner() error {
return pkg_errors.New("inner error")
}
func MyFuncMiddle() error {
err := MyFuncInner()
return pkg_errors.Wrap(err, "middle error")
}
func MyFuncOuter() error {
err := MyFuncMiddle()
return pkg_errors.Wrapf(err, "outer error - %d", 2)
}
func main() {
err := MyFuncOuter()
fmt.Printf("%v\n\n", err)
fmt.Printf("%+v\n", err)
}
```
----
<!-- .slide: data-transition="convex" -->
#### Result
```shell
outer error - 2: middle error: inner error
inner error
main.MyFuncInner
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:10
main.MyFuncMiddle
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:14
main.MyFuncOuter
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:19
main.main
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:24
runtime.main
/usr/local/go/src/runtime/proc.go:204
runtime.goexit
/usr/local/go/src/runtime/asm_amd64.s:1374
middle error
main.MyFuncMiddle
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:15
main.MyFuncOuter
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:19
main.main
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:24
runtime.main
/usr/local/go/src/runtime/proc.go:204
runtime.goexit
/usr/local/go/src/runtime/asm_amd64.s:1374
outer error - 2
main.MyFuncOuter
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:20
main.main
/.../go/src/survive_under_the_crap_go_error_system/pkg_errors_example.go:24
runtime.main
/usr/local/go/src/runtime/proc.go:204
runtime.goexit
/usr/local/go/src/runtime/asm_amd64.s:1374
```
----
<!-- .slide: data-transition="convex" -->
### Integration
* for http server
* use the gin, luke
* <small>https://github.com/gin-gonic/gin#model-binding-and-validation</small>
```go=
// push error
c.Error(err)
// list error
c.Errors.ByType(errType)
```
---
<!-- .slide: data-transition="convex" -->
## Advice
----
<!-- .slide: data-transition="convex" -->
### Wrapping
Wrap as early as possible
```go=
import (
"errors"
"fmt"
"golang.org/x/xerrors"
pkg_errors "github.com/pkg/errors"
)
func MyFunc(param1 int, param2 string) (string, error) {
// do something...
r, err := foreign_package.ForeignFunc()
if err != nil {
err = fmt.Errorf("<error description>: %w", err)
err = xerrors.Errorf("<error description>: %w", err)
err = pkg_errors.Wrap(err, "<error description>")
return "", err
}
// do something...
return result, nil
}
```
----
<!-- .slide: data-transition="convex" -->
### Inspection
Collect and log error at entry function or middleware
```go=
// for Go 1.13 error system
func ExtractErrorMessageChain(err error) string {
var b strings.Builder
for e := err; e != nil; e = errors.Unwrap(e) {
b.WriteString(e.Error())
b.WriteString("\n")
}
return b.String()
}
func MyEntryFunc() {
// do something...
r, err := MyFunc(param1, param2)
// for Go 1.13 error system
fmt.Printf("%s", ExtractErrorMessageChain(err))
// for golang.org/x/xerrors & github.com/pkg/errors
fmt.Printf("%+v\n", err)
}
```
----
<!-- .slide: data-transition="convex" -->
### Displaying
Only show the most outer error message to end-user
```go=
// for Go 1.13 error system & golang.org/x/xerrors
func WrapError(msg string, err error) error {
return fmt.Errorf("%s: %w", msg, err)
}
// for all error frameworks
func GetErrorMessageTitle(err error) string {
return strings.SplitN(err.Error(), ":", 2)[0]
}
```
----
<!-- .slide: data-transition="convex" -->
### Sum Up
----
<!-- .slide: data-transition="convex" -->
#### Example
* Go 1.13 error system
* Go standard HTTP library
* github.com/julienschmidt/httprouter
```go=
package main
import (
"encoding/json"
"errors"
"fmt"
"io/ioutil"
"net/http"
"os"
"strings"
//"reflect"
"github.com/julienschmidt/httprouter"
)
// utility function
// for Go 1.13 error system
func ExtractErrorMessageChain(err error) string {
var b strings.Builder
for e := err; e != nil; e = errors.Unwrap(e) {
b.WriteString(e.Error())
b.WriteString("\n")
}
return b.String()
}
// for Go 1.13 error system & golang.org/x/xerrors
func WrapError(msg string, err error) error {
return fmt.Errorf("%s: %w", msg, err)
}
// for all error frameworks
func GetErrorMessageTitle(err error) string {
return strings.SplitN(err.Error(), ":", 2)[0]
}
// compare error using first part of message before colon, for example: "<first>: <rest>"
func IsErrorEqual(e1 error, e2 error) bool {
if e1 == nil || e2 == nil {
return e1 == e2
} else {
// CAUTION: error type is useless (in most cases)
// return reflect.TypeOf(e1) == reflect.TypeOf(e2) && GetErrorMessageTitle(e1) == GetErrorMessageTitle(e2)
return GetErrorMessageTitle(e1) == GetErrorMessageTitle(e2)
}
}
// ApiError
type ApiError struct {
statusCode int
msg string
err error
}
func NewApiError(statusCode int, msg string, err error) error {
if err != nil {
msg = fmt.Sprintf("%s: %s", msg, err.Error())
}
return &ApiError{statusCode: statusCode, msg: msg, err: err}
}
func (e *ApiError) Error() string {
return e.msg
}
func (e *ApiError) Unwrap() error {
return e.err
}
func (e *ApiError) StatusCode() int {
return e.statusCode
}
func WrapApiError(template error, err error) error {
apiErrorTemplate, ok := template.(*ApiError)
if ok {
err = NewApiError(apiErrorTemplate.StatusCode(), apiErrorTemplate.Error(), err)
} else {
err = WrapError(template.Error(), err)
}
return err
}
// handler
type HandleE func(http.ResponseWriter, *http.Request, httprouter.Params) error
func ErrAwareHandle(h HandleE) httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, ps httprouter.Params) {
err := h(w, r, ps)
if err != nil {
// respond error with simplified message
errorMessage := GetErrorMessageTitle(err)
var statusCode int
switch e := err.(type) {
case *ApiError:
statusCode = e.StatusCode()
default:
statusCode = http.StatusInternalServerError
}
http.Error(w, errorMessage, statusCode)
// log detailed error
messageChain := ExtractErrorMessageChain(err)
fmt.Fprintf(os.Stderr, "error:\n%s====\n", messageChain)
}
}
}
// usage
var (
ErrEmptyMessage = errors.New("message is empty")
ErrDirtyMessage = errors.New("助兵衛")
ErrHentaiMessage = errors.New("変態")
)
func detectNonDirtyMessage(msg string) error {
if msg == "FQ" {
return ErrDirtyMessage
}
return nil
}
func processMessage(msg string) (string, error) {
if msg == "" {
return "", ErrEmptyMessage
}
err := detectNonDirtyMessage(msg)
if err != nil {
return "", WrapApiError(ErrHentaiMessage, err)
}
return msg, nil
}
var (
ErrMessageLost = NewApiError(http.StatusBadRequest, "message lost", nil)
ErrYouDirty = NewApiError(http.StatusForbidden, "you dirty", nil)
)
type Input struct {
Msg string `json:msg`
}
func example(w http.ResponseWriter, r *http.Request, ps httprouter.Params) error {
name := ps.ByName("name")
if name != "say" {
err := errors.New("must be /error_test/say")
return NewApiError(http.StatusNotFound, "not found", err)
}
body, err := ioutil.ReadAll(r.Body)
if err != nil {
// this statement won't wrap error, it just concatenates error message
return fmt.Errorf("body read failed: %v", err)
}
if len(body) < 2 {
return errors.New("body is not enough")
}
var input Input
if err = json.Unmarshal(body, &input); err != nil {
return NewApiError(http.StatusBadRequest, "bad request", err)
}
resp, err := processMessage(input.Msg)
if IsErrorEqual(err, ErrEmptyMessage) {
return WrapApiError(ErrMessageLost, err)
}
if IsErrorEqual(err, ErrHentaiMessage) {
return WrapApiError(ErrYouDirty, err)
}
if err != nil {
return err
}
w.WriteHeader(200)
w.Write([]byte("you sent:\n"))
w.Write([]byte(resp))
w.Write([]byte("\n"))
return nil
}
func main() {
router := httprouter.New()
router.POST("/error_test/:name", ErrAwareHandle(example))
err := http.ListenAndServe(":8000", router)
if err != nil {
panic(err)
}
}
// client example
// curl -v -X POST --data '{"msg": "FQ2"}' http://localhost:8000/error_test/say
```
----
<!-- .slide: data-transition="convex" -->
#### Integrate with Error Type from Foreign Packages
* If we use internal error types from
* golang.org/x/xerrors
* github.com/pkg/errors
* We could store error message like below
```go=
"403|forbidden: <error description>: <error description>: ..."
```
<!-- .element: class="fragment" data-fragment-index="1" -->
---
<!-- .slide: data-transition="convex" -->
## One More Thing
----
<!-- .slide: data-transition="convex" -->
### Fail Fast
* When error occurs:
* Stop immediately
* Report the error as early as possible
----
<!-- .slide: data-transition="convex" -->
### Idiom Error Handling
```go=
func MyFunc(param1 int, param2 string) (string, error) {
// do something...
f, err := os.Open("filename.ext")
if err != nil {
return "", err
}
// do something...
n, err := io.ReadFull(f, buf1)
if err != nil {
return "", err
}
// do something...
n, err = f.Read(buf2)
if err != nil {
return "", err
}
// do something...
return result, nil
}
```
----
<!-- .slide: data-transition="convex" -->
### Error Handling in Foreign Language
![rust](https://www.rust-lang.org/static/images/rust-logo-blk.svg)
----
<!-- .slide: data-transition="convex" -->
```rust=
panic!()
Result<T, E>
Option<T>
myobj.myfn()? // ? operator
```
<small>https://doc.rust-lang.org/book/ch09-00-error-handling.html</small>
----
<!-- .slide: data-transition="convex" -->
```rust=
e.source()
e.backtrace()
```
<small>https://doc.rust-lang.org/std/error/trait.Error.html</small>
----
<!-- .slide: data-transition="convex" -->
### Evolution?
```go=
func MyFunc(param1 int, param2 string) (string, error) {
// do something...
f := try(os.Open("filename.ext"))
// do something...
n := try(io.ReadFull(f, buf1))
// do something...
n = try(f.Read(buf2))
// do something...
return result, nil
}
```
----
<!-- .slide: data-transition="convex" -->
### Impossible!?
Because something mess up:
```go=
if err != nil {
if err != io.EOF {
return "", err
} else {
return result, nil
}
}
```
----
<!-- .slide: data-transition="convex" -->
人生オワタ \(\^o\^)/ I am done!
----
<!-- .slide: data-transition="convex" -->
### Evolution Again
Add handle:
```go=
func MyFunc(param1 int, param2 string) (string, error) {
handle err {
if err != io.EOF {
return "", err
} else {
return result, nil
}
}
// do something...
f := check os.Open("filename.ext")
// do something...
n := check io.ReadFull(f, buf1)
// do something...
n = check f.Read(buf2)
// do something...
return result, nil
}
```
<small>https://go.googlesource.com/proposal/+/master/design/go2draft-error-handling.md</small>
----
<!-- .slide: data-transition="convex" -->
![讓我們看下去](https://stickershop.line-scdn.net/stickershop/v1/sticker/16846578/iPhone/sticker@2x.png =531x495)
<small>https://store.line.me/stickershop/product/8601/zh-Hant</small>
---
<!-- .slide: data-transition="convex" -->
## Conclusion
----
<!-- .slide: data-transition="convex" -->
### Advice
* Wrapping
* Wrap as early as possible
* Inspection
* Collect and log error at entry function or middleware
* Displaying
* Only show the most outer error message to end-user
----
<!-- .slide: data-transition="convex" -->
### Promote
* :thumbsup: github.com/pkg/errors by Dave Cheney
* https://github.com/pkg/errors
* https://godoc.org/github.com/pkg/errors
----
<!-- .slide: data-transition="convex" -->
### Bless
:hash: {錯誤抓精光 碼農發大財|Errors get eliminated, coders get richer.}
> [name=郭學聰 Hsueh-Tsung Kuo] [time=2020_11_14] [color=red] :notebook:
---
<!-- .slide: data-transition="convex" -->
## Reference
----
<!-- .slide: data-transition="convex" -->
### Resources
* Go 2 Draft Designs
* <small>https://go.googlesource.com/proposal/+/master/design/go2draft.md</small>
* Proposal: Go 2 Error Inspection
* <small>https://go.googlesource.com/proposal/+/master/design/29934-error-values.md</small>
----
<!-- .slide: data-transition="convex" -->
### Resources
* Working with Errors in Go 1.13
* https://blog.golang.org/go1.13-errors
* golang.org/x/xerrors by Go team members
* https://github.com/golang/xerrors
* https://godoc.org/golang.org/x/xerrors
* github.com/pkg/errors by Dave Cheney
* https://github.com/pkg/errors
* https://godoc.org/github.com/pkg/errors
---
<!-- .slide: data-transition="zoom" -->
## Q&A
---
<style>
.reveal {
background: #FFDFEF;
color: black;
}
.reveal h2,
.reveal h3,
.reveal h4,
.reveal h5,
.reveal h6 {
color: black;
}
.reveal code {
font-size: 18px !important;
line-height: 1.2;
}
.progress div{
height:14px !important;
background: hotpink !important;
}
// original template
.rightpart{
float:right;
width:50%;
}
.leftpart{
margin-right: 50% !important;
height:50%;
}
.reveal section img { background:none; border:none; box-shadow:none; }
p.blo {
font-size: 50px !important;
background:#B6BDBB;
border:1px solid silver;
display:inline-block;
padding:0.5em 0.75em;
border-radius: 10px;
box-shadow: 5px 5px 5px #666;
}
p.blo1 {
background: #c7c2bb;
}
p.blo2 {
background: #b8c0c8;
}
p.blo3 {
background: #c7cedd;
}
p.bloT {
font-size: 60px !important;
background:#B6BDD3;
border:1px solid silver;
display:inline-block;
padding:0.5em 0.75em;
border-radius: 8px;
box-shadow: 1px 2px 5px #333;
}
p.bloA {
background: #B6BDE3;
}
p.bloB {
background: #E3BDB3;
}
/*.slide-number{
margin-bottom:10px !important;
width:100%;
text-align:center;
font-size:25px !important;
background-color:transparent !important;
}*/
iframe.myclass{
width:100px;
height:100px;
bottom:0;
left:0;
position:fixed;
border:none;
z-index:99999;
}
h1.raw {
color: #fff;
background-image: linear-gradient(90deg,#f35626,#feab3a);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
animation: hue 5s infinite linear;
}
@keyframes hue {
from {
filter: hue-rotate(0deg);
}
to {
filter: hue-rotate(360deg);
}
}
.progress{
height:14px !important;
}
.progress span{
height:14px !important;
background: url("") repeat-x !important;
}
.progress span:after,
.progress span.nyancat{
content: "";
background: url('') !important;
width: 34px !important;
height: 21px !important;
border: none !important;
float:right;
margin-top:-7px;
margin-right:-10px;
}
</style>
{"metaMigratedAt":"2023-06-15T15:10:47.958Z","metaMigratedFrom":"YAML","title":"Survive under the Crap Go Error System","breaks":true,"description":"View the slide with \"Slide Mode\".","slideOptions":"{\"spotlight\":{\"enabled\":false},\"allottedMinutes\":25}","contributors":"[{\"id\":\"ea27dcd7-a3f2-47c2-b25e-6760e7936c38\",\"add\":75912,\"del\":47990}]"}