This is Go error handling example based on Ben Johnson's Failure is your Domain. According to the article error consumers have different expectations:
- end user requires human-readable message. API undefined errors must not be shown, e.g., Postgres error can reveal db schema.
- app itself can recover from error states using error codes.
- operator should be able to debug and see all errors including stack trace.
Let's start the API server and see that in action,
but before beginning make sure to generate gRPC code, e.g., buf generate
.
$ go run ./cmd/server/
Domain errors (API errors) should have Code
and Message
. For instance, "duplicate username" error
account.Error{
Code: account.EConflict,
Message: "Username is already in use. Please choose a different username.",
}
is shown to the API consumer
$ curl -i -X POST -d '{"username":"bob"}' http://localhost:8000/v1/users
HTTP/1.1 400 Bad Request
{"error":{"code":"conflict","message":"Username is already in use. Please choose a different username."}}
and also logged for operators
{
"caller": "middleware.go:44",
"err": "conflict: Username is already in use. Please choose a different username.",
"method": "CreateUser",
"took": "2.257µs",
"ts": "2018-12-20T13:49:10.379131Z",
"user": {
"ID": "",
"Username": "bob"
}
}
Non-domain errors are considered internal and not shown to API consumers. For example, db connection error
fmt.Errorf(
"UserStorage.CreateUser: %w",
fmt.Errorf(
"insertUser: %w",
fmt.Errorf("db connection failed"),
),
)
is suppressed on API level
$ curl -i -X POST -d '{"username":"alice"}' http://localhost:8000/v1/users
HTTP/1.1 500 Internal Server Error
{"error":{"code":"internal","message":"An internal error has occurred."}}
but logged for operators
{
"caller": "middleware.go:44",
"err": "UserStorage.CreateUser: insertUser: db connection failed",
"method": "CreateUser",
"took": "7.585µs",
"ts": "2018-12-20T13:44:52.491775Z",
"user": {
"ID": "",
"Username": "alice"
}
}
Errors returned from Go kit's endpoint.Endpoint
can be propagated to the end user (requests throttling)
or shown as internal errors (JSON serialization errors, e.g., EOF):
$ curl -i -X POST -d '{"username":"bob"}' http://localhost:8000/v1/users
HTTP/1.1 429 Too Many Requests
{"error":{"code":"rate_limit","message":"API rate limit exceeded."}}
The error was also logged for operators:
{
"caller": "server.go:112",
"component": "HTTP",
"err": "rate limit exceeded",
"ts": "2018-12-20T13:49:12.333333Z"
}
Note, it's possible to wrap domain errors, for example,
err := account.Error{
Code: account.ENotFound,
Message: "User not found.",
Inner: sql.ErrNoRows,
}
fmt.Errorf("user (id %s) not found: %w", userID, err)
shows the domain error
$ curl -i http://localhost:8000/v1/users/87553f14-4c0f-4bd8-8be1-1b6ff5bd8eef
HTTP/1.1 404 Not Found
{"error":{"code":"not_found","message":"User not found."}}
and logs the full error for operators
{
"caller": "middleware.go:29",
"err": "user (id 87553f14-4c0f-4bd8-8be1-1b6ff5bd8eef) not found: not_found: User not found.: sql: no rows in result set",
"method": "FindUserByID",
"output": null,
"took": "26.001µs",
"ts": "2019-09-24T14:19:04.632203Z",
"user_id": "87553f14-4c0f-4bd8-8be1-1b6ff5bd8eef"
}
To run tests you will need Postgres and test env variables set up. Also make sure to generate gRPC code.
$ make docker_run_postgres
$ make test
If tests fail with "connection refused", point TEST_PGHOST
to a default docker machine.
$ make test TEST_PGHOST=$(docker-machine ip default)