mirror of
https://git.mirrors.martin98.com/https://github.com/SigNoz/signoz
synced 2025-06-04 11:25:52 +08:00

* docs(readme): add docs/contributing/go/readme * docs(readme): add docs/contributing/go/readme * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * docs(readme): add errors package * Update docs/contributing/go/errors.md Co-authored-by: ellipsis-dev[bot] <65095814+ellipsis-dev[bot]@users.noreply.github.com> * Update docs/contributing/go/errors.md Co-authored-by: ellipsis-dev[bot] <65095814+ellipsis-dev[bot]@users.noreply.github.com> --------- Co-authored-by: ellipsis-dev[bot] <65095814+ellipsis-dev[bot]@users.noreply.github.com>
104 lines
4.3 KiB
Markdown
104 lines
4.3 KiB
Markdown
# Errors
|
|
|
|
SigNoz includes its own structured [errors](/pkg/errors/errors.go) package. It's built on top of Go's `error` interface, extending it to add additional context that helps provide more meaningful error messages throughout the application.
|
|
|
|
## How to use it?
|
|
|
|
To use the SigNoz structured errors package, use these functions instead of the standard library alternatives:
|
|
|
|
```go
|
|
// Instead of errors.New()
|
|
errors.New(typ, code, message)
|
|
|
|
// Instead of fmt.Errorf()
|
|
errors.Newf(typ, code, message, args...)
|
|
```
|
|
|
|
### Typ
|
|
The Typ (read as Type, defined as `typ`) is used to categorize errors across the codebase and is loosely coupled with HTTP/GRPC status codes. All predefined types can be found in [pkg/errors/type.go](/pkg/errors/type.go). For example:
|
|
|
|
- `TypeInvalidInput` - Indicates invalid input was provided
|
|
- `TypeNotFound` - Indicates a resource was not found
|
|
|
|
By design, `typ` is unexported and cannot be declared outside of [errors](/pkg/errors/errors.go) package. This ensures that it is consistent across the codebase and is used in a way that is meaningful.
|
|
|
|
### Code
|
|
Codes are used to provide more granular categorization within types. For instance, a type of `TypeInvalidInput` might have codes like `CodeInvalidEmail` or `CodeInvalidPassword`.
|
|
|
|
To create new error codes, use the `errors.MustNewCode` function:
|
|
|
|
```go
|
|
var (
|
|
CodeThingAlreadyExists = errors.MustNewCode("thing_already_exists")
|
|
CodeThingNotFound = errors.MustNewCode("thing_not_found")
|
|
)
|
|
```
|
|
|
|
> 💡 **Note**: Error codes must match the regex `^[a-z_]+$` otherwise the code will panic.
|
|
|
|
## Show me some examples
|
|
|
|
### Using the error
|
|
A basic example of using the error:
|
|
|
|
```go
|
|
var (
|
|
CodeThingAlreadyExists = errors.MustNewCode("thing_already_exists")
|
|
)
|
|
|
|
func CreateThing(id string) error {
|
|
t, err := thing.GetFromStore(id)
|
|
if err != nil {
|
|
if errors.As(err, errors.TypeNotFound) {
|
|
// thing was not found, create it
|
|
return thing.Create(id)
|
|
}
|
|
|
|
// something else went wrong, wrap the error with more context
|
|
return errors.Wrapf(err, errors.TypeInternal, errors.CodeUnknown, "failed to get thing from store")
|
|
}
|
|
|
|
return errors.Newf(errors.TypeAlreadyExists, CodeThingAlreadyExists, "thing with id %s already exists", id)
|
|
}
|
|
```
|
|
|
|
### Changing the error
|
|
Sometimes you may want to change the error while preserving the message:
|
|
|
|
```go
|
|
func GetUserSecurely(id string) (*User, error) {
|
|
user, err := repository.GetUser(id)
|
|
if err != nil {
|
|
if errors.Ast(err, errors.TypeNotFound) {
|
|
// Convert NotFound to Forbidden for security reasons
|
|
return nil, errors.New(errors.TypeForbidden, errors.CodeAccessDenied, "access denied to requested resource")
|
|
}
|
|
return nil, err
|
|
}
|
|
return user, nil
|
|
}
|
|
```
|
|
|
|
## Why do we need this?
|
|
|
|
In a large codebase like SigNoz, error handling is critical for maintaining reliability, debuggability, and a good user experience. We believe that it is the **responsibility of a function** to return **well-defined** errors that **accurately describe what went wrong**. With our structured error system:
|
|
|
|
- Functions can create precise errors with appropriate additional context
|
|
- Callers can make informed decisions based on the additional context
|
|
- Error context is preserved and enhanced as it moves up the call stack
|
|
|
|
The caller (which can be another function or a HTTP/gRPC handler or something else entirely), can then choose to use this error to take appropriate actions such as:
|
|
|
|
- A function can branch into different paths based on the context
|
|
- An HTTP/gRPC handler can derive the correct status code and message from the error and send it to the client
|
|
- Logging systems can capture structured error information for better diagnostics
|
|
|
|
Although there might be cases where this might seem too verbose, it makes the code more maintainable and consistent. A little verbose code is better than clever code that doesn't provide enough context.
|
|
|
|
## What should I remember?
|
|
|
|
- Think about error handling as you write your code, not as an afterthought.
|
|
- Always use the [errors](/pkg/errors/errors.go) package instead of the standard library's `errors.New()` or `fmt.Errorf()`.
|
|
- Always assign appropriate codes to errors when creating them instead of using the "catch all" error codes defined in [pkg/errors/code.go](/pkg/errors/code.go).
|
|
- Use `errors.Wrapf()` to add context to errors while preserving the original when appropriate.
|