How to write MeshKit compatible errors
Meshery pervasively uses MeshKit as a golang and service mesh management-specific library in all of its components. MeshKit helps populate error messages with a uniform and useful set of informative attributes.
To help with creating error codes, MeshKit contains a tool that analyzes, verifies and updates error codes in Meshery source code trees. It extracts error details into a file that can be used for publishing all error code references on the Meshery error codes reference page. The objective to create this was to avoid centralized handling of error codes and automating everything
In order to create a Meshery error object, you will need to create a custom wrapper object for the native golang error. This can be done from the MeshKit Error package.
This tool will create a couple of files, one of them is designed to be used to generate the error reference on the Meshery Documentation website. The file errorutil_analyze_summary.json contains a summary of the analysis, notably lists of duplicates etc.
Conventions to follow while creating errors
- Errors names and codes are namespaced to components, i.e. they need to be unique within a component, which is verified by this tool.
- Errors are not to be reused across components and modules.
- Error codes are not to be set as integer. CI will take care of updating Error codes from a string to an integer.
- Running
make error
analyzes your code and returns any warnings to be aware of. - Capitalize the first letter of the every error description.
- Using errors.NewDefault(…) is deprecated. This tool emits a warning if its use is detected.
- Use errors.New(…) from MeshKit to create actual errors with all the details. This is often done in a factory function. It is important that the error code variable is used here, not a literal. Specify detailed descriptions, probable causes, and remedies. They need to be string literals, call expressions are ignored. This tool extracts this information from the code and exports it. For the Code argument in the errors.New use the same Error name and append a “Code” after it. e.g error name : ErrApplyManifest then the error code is ErrApplyManifestCode
- Set the value to any string, like “replace_me” (no convention here), e.g. ErrApplyManifestCode = “replace_me”.
- By convention, error codes and the factory functions live in files called error.go. The tool checks all files, but updates only error.go files.
Use the errors.New()
function to create a new instance of the error object and pass situation-specific attributes as function arguments. These attributes are:
- Code
- Short Description
- Long Description
- Probable Cause
- Suggested Remediation
Syntax
errors.New(ErrExampleCode, errors.Alert, []string{"<short-description>"}, []string{"<long-description>"}, []string{"<probable-cause>"}, []string{"<suggested remediation>"})
Example
In this example we are creating an Error for being unable to marshal JSON
var (
// Error code
ErrMarshalCode= "replace_me"
//Static errors (for example)
ErrExample = errors.New(ErrExampleCode, errors.Alert, []string{"<short-description>"}, []string{"<long-description>"}, []string{"<probable-cause>"}, []string{"<suggested remediation>"})
)
// Dynamic errors
//Error Name
func ErrMarshal(err error, obj string) error {
return errors.New(ErrMarshalCode, errors.Alert, []string{"Unable to marshal the : ", obj}, []string{err.Error()}, []string{}, []string{})
}
Replacing old Error Codes
Old
bd, err := json.Marshal(providers)
if err != nil {
http.Error(w, "unable to marshal the providers", http.StatusInternalServerError)
return
}
New
bd, err := json.Marshal(providers)
if err != nil {
obj := "provider"
http.Error(w, ErrMarshal(err, obj).Error(), http.StatusInternalServerError)
return
}
Replacing logrus
There already exists an interface for logger in MeshKit.
WARNING
To enforce the use of meshkit errors, meshkit logger was designed such that it only works with meshkit errors. If a non-meshkit error is logged through the logger, it would panic and kill the process. See: meshkit#119 for more insight.Defining a Logger
type Logger struct {
log logger.Handler
}
Debug
Old
logrus.Debugf("meshLocationURL: %s", meshLocationURL)
New
l.log.Debug("meshLocationURL: ", meshLocationURL)
Error
Old
logrus.Errorf("error marshaling data: %v.", err)
New
l.log.Error(ErrMarshal(err, obj))
A small program using meshkit errors and logger
package main
import (
"fmt"
"os"
meshkitErrors "github.com/layer5io/meshkit/errors"
"github.com/layer5io/meshkit/logger"
)
var (
// CI will replace `test_code` with new error code
ErrOpeningFileCode = "test_code"
)
func main() {
log, err := logger.New("test", logger.Options{
Format: logger.SyslogLogFormat,
DebugLevel: true,
})
if err != nil {
fmt.Println(err)
os.Exit(1)
}
// logging meshkit error
err = openFileWithMeshkitError("some.txt")
if err != nil {
log.Error(err)
}
// OUTPUT
// ERRO[2021-11-10T17:31:28+05:30] open some.txt: no such file or directory app=test code=1001 probable-cause="empty string passed as argument .file with this name doesn't exist" severity=2 short-description="unable to open file" suggested-remediation="pass a non-empty string as filename .create file before opening it"
// logging non meshkit error
err = openFile("some.txt")
if err != nil {
log.Error(err)
}
// OUTPUT
// panic: interface conversion: error is *fs.PathError, not *errors.Error
// goroutine 1 [running]:
// github.com/layer5io/meshkit/errors.GetCode({0x50dfc0, 0xc000068450})
// /home/rudraksh/go/pkg/mod/github.com/layer5io/meshkit@v0.2.33/errors/errors.go:90 +0x90
// github.com/layer5io/meshkit/logger.(*Logger).Error(0xc00000e040, {0x50dfc0, 0xc000068450})
// /home/rudraksh/go/pkg/mod/github.com/layer5io/meshkit@v0.2.33/logger/logger.go:57 +0xbb
// main.main()
// /home/rudraksh/trash/meshkitplay/main.go:32 +0xe2
// exit status 2
}
// this returns a non meshkit error
func openFile(name string) error {
_, err := os.Open(name)
return err
}
// this returns a meshkit error
func openFileWithMeshkitError(name string) error {
_, err := os.Open(name)
return ErrOpeningFile(err)
}
func ErrOpeningFile(err error) error {
return meshkitErrors.New(ErrOpeningFileCode, meshkitErrors.Alert, []string{"unable to open file"},
[]string{err.Error()},
[]string{"empty string passed as argument ", "file with this name doesn't exist"},
[]string{"pass a non-empty string as filename ", "create file before opening it"})
}