layout | title |
---|---|
article |
Handling Errors |
Almost all operations with database/sql
types return an error as the last
value. You should always check these errors, never ignore them.
There are a few places where error behavior is special-case, or there's something additional you might need to know.
Consider the following code:
for rows.Next() { // ... } if err = rows.Err(); err != nil { // handle the error here }
The error from rows.Err()
could be the result of a variety of errors in the
rows.Next()
loop. The loop
might exit for some reason other than finishing the loop normally, so you always
need to check whether the loop terminated normally or not. An abnormal
termination automatically calls rows.Close()
, although it's harmless to call it
multiple times.
You should always explicitly close a sql.Rows
if you exit the loop
prematurely, as previously mentioned. It's auto-closed if the loop exits
normally or through an error, but you might mistakenly do this:
for rows.Next() { // ... break; // whoops, rows is not closed! memory leak... } // do the usual "if err = rows.Err()" [omitted here]... // it's always safe to [re?]close here: if err = rows.Close(); err != nil { // but what should we do if there's an error? log.Println(err) }
The error returned by rows.Close()
is the only exception to the general rule
that it's best to capture and check for errors in all database operations. If
rows.Close()
returns an error, it's unclear what you should do.
Logging the error message or panicing might be the only sensible thing,
and if that's not sensible, then perhaps you should just ignore the error.
Consider the following code to fetch a single row:
var name string err = db.QueryRow("select name from users where id = ?", 1).Scan(&name) if err != nil { log.Fatal(err) } fmt.Println(name)
What if there was no user with id = 1
? Then there would be no row in the
result, and .Scan()
would not scan a value into name
. What happens then?
Go defines a special error constant, called sql.ErrNoRows
, which is returned
from QueryRow()
when the result is empty. This needs to be handled as a
special case in most circumstances. An empty result is often not considered an
error by application code, and if you don't check whether an error is this
special constant, you'll cause application-code errors you didn't expect.
Errors from the query are deferred until Scan()
is called, and then are
returned from that. The above code is better written like this instead:
var name string err = db.QueryRow("select name from users where id = ?", 1).Scan(&name) if err != nil { if err == sql.ErrNoRows { // there were no rows, but otherwise no error occurred } else { log.Fatal(err) } } fmt.Println(name)
One might ask why an empty result set is considered an error. There's nothing
erroneous about an empty set. The reason is that the QueryRow()
method needs
to use this special-case in order to let the caller distinguish whether
QueryRow()
in fact found a row; without it, Scan()
wouldn't do anything and
you might not realize that your variable didn't get any value from the database
after all.
You should not run into this error when you're not using QueryRow()
. If you
encounter this error elsewhere, you're doing something wrong.
It can be tempting to write code like the following:
rows, err := db.Query("SELECT someval FROM sometable") // err contains: // ERROR 1045 (28000): Access denied for user 'foo'@'::1' (using password: NO) if strings.Contains(err.Error(), "Access denied") { // Handle the permission-denied error }
This is not the best way to do it, though. For example, the string value might vary depending on what language the server uses to send error messages. It's much better to compare error numbers to identify what a specific error is.
The mechanism to do this varies between drivers, however, because this isn't
part of database/sql
itself. In the MySQL driver that this tutorial focuses
on, you could write the following code:
if driverErr, ok := err.(*mysql.MySQLError); ok { // Now the error number is accessible directly if driverErr.Number == 1045 { // Handle the permission-denied error } }
Again, the MySQLError
type here is provided by this specific driver, and the
.Number
field may differ between drivers. The value of the number, however,
is taken from MySQL's error message, and is therefore database specific, not
driver specific.
This code is still ugly. Comparing to 1045, a magic number, is a code smell.
Some drivers (though not the MySQL one, for reasons that are off-topic here)
provide a list of error identifiers. The libpq
driver does, for example, in
error.go. And there's an
external package of MySQL error numbers maintained by
VividCortex. Using such a list, the
above code is better written thus:
if driverErr, ok := err.(*mysql.MySQLError); ok { if driverErr.Number == mysqlerr.ER_ACCESS_DENIED_ERROR { // Handle the permission-denied error } }
What if your connection to the database is dropped, killed, or has an error?
You don't need to implement any logic to retry failed statements when this
happens. As part of the connection pooling in
database/sql
, handling failed connections is built-in. If you execute a query
or other statement and the underlying connection has a failure, Go will reopen a
new connection (or just get another from the connection pool) and retry, up to
10 times.
There can be some unintended consequences, however. Some types of errors may be
retried when other error conditions happen. This might also be driver-specific.
One example that has occurred with the MySQL driver is that using KILL
to
cancel an undesired statement (such as a long-running query) results in the
statement being retried up to 10 times.
Previous: Using Prepared Statements Next: Working with NULLs