21

For example, I am using one Go standard library function as:

func Dial(network, address string) (*Client, error)

This function may return errors, and I just care about errors which report "connection lost" or "connection refused", then do some code to fix these.
It seems like:

client, err := rpc.Dial("tcp", ":1234")  
if err == KindOf(ConnectionRefused) {
  // do something
}

What's more, how to get all the errors a specific standard library function may return?

Jonathan Hall
  • 75,165
  • 16
  • 143
  • 189
fding
  • 424
  • 1
  • 5
  • 18
  • https://tour.golang.org/methods/15 https://tour.golang.org/methods/16 – zerkms Apr 25 '17 at 08:58
  • There is no direct / exact answer, partly blamed on the `net` package authors. Related / useful questions (about error handling): [Specifically check for timeout error](http://stackoverflow.com/questions/23494950/specifically-check-for-timeout-error); and [Does go have standard Err variables?](http://stackoverflow.com/questions/30177860/does-go-have-standard-err-variables/30178766#30178766); and [Go. Best practice to handle error from multiple abstract level](http://stackoverflow.com/questions/37346694/go-best-practice-to-handle-error-from-multiple-abstract-level/37347531#37347531). – icza Apr 25 '17 at 11:41

2 Answers2

15

In modern Go, the best general way to do this is with the errors.Is and errors.As functions in the standard library.

For the example in the original question, this is quite straight forward:

if errors.Is(err, syscall.ECONNREFUSED) {
    // Connection refused error
} else {
    // Some other kind of error
}

However, this only works for errors which are properly wrapped/created to use these capabilities. Sometimes you may need to mix-and-match the use of errors.Is and errors.As with string comparison. I've made a video that goes into some of these details for those interested.


My old answer:

There's no standard way to do this.

The most obvious way, which should only be used if no other method is available, is to compare the error string against what you expect:

if err.Error() == "connection lost" { ... }

Or perhaps more robust in some situations:

if strings.HasSuffix(err.Error(), ": connection lost") { ... }

But many libraries will return specific error types, which makes this much easier.

In your case, what's relevant are the various error types exported by the net package: AddrError, DNSConfigError, DNSError, Error, etc.

You probably care most about net.Error, which is used for network errors. So you could check thusly:

if _, ok := err.(net.Error); ok {
    // You know it's a net.Error instance
    if err.Error() == "connection lost" { ... }
}

What's more, how to get all the errors a specific standard library function may return?

The only fool-proof way to do this is to read the source for the library. Before going to that extreme, a first step is simply to read the godoc, as in the case of the net package, the errors are pretty well documented.

Jonathan Hall
  • 75,165
  • 16
  • 143
  • 189
4

You can now use the errors.Is() function to compare some standard errors:

client, err := net.Dial("tcp", ":1234")
if errors.Is(err, net.ErrClosed) {
    fmt.Println("connection has been closed.")
}

A common file opening scenario:

file, err := os.Open("non-existing");
if err != nil {
    if errors.Is(err, fs.ErrNotExist) {
        fmt.Println("file does not exist")
    } else {
        fmt.Println(err)
    }
}

UPDATE
You can also use errors.As() if you'd like to check the error type:

client, err := net.Dial("tcp", ":1234")
var errC = net.ErrClosed
if errors.As(err, &errC) {
    fmt.Printf("connection has been closed: %s", errC)
}
client.Close()

A more detailed explanation can be found in the Go Blog.

17xande
  • 2,430
  • 1
  • 24
  • 33