Description
The following example program:
package main
import (
"net"
)
func main() {
net.Listen("tcp", "localhost:8080")
select{}
}
Currently yields this result:
$ netstat -nl | grep 8080
tcp 0 0 127.0.0.1:8080 0.0.0.0:* LISTEN
While the following result would be optimal:
$ netstat -nl | grep 8080
tcp6 0 0 127.0.0.1:8080 :::* LISTEN
tcp6 0 0 ::1:8080 :::* LISTEN
(Note that the first socket is actually dualstack, and bound to ::ffff:127.0.0.1, but that's less critical than adding the second socket bound to ::1.)
More generally, when you call net.Listen()
on a hostname which resolves to multiple IPv4/IPv6 addresses, only the first IPv4 address is selected. An analogous problem occurs if you Listen("tcp", ":8080")
on an operating system that doesn't support dualstack sockets: instead of returning a pair of sockets bound to [::]:8080
and 0.0.0.0:80
, you only get IPv4.
The fundamental flaw is that Listen()
assumes a single socket, which is a leaky abstraction that's inappropriate for high-level things like example servers, e.g.:
http://golang.org/pkg/net/#pkg-overview
Go should either adapt the Listen()
API to support multiple sockets, or if that's not feasible, a new multi-socket API should be introduced, which deprecates Listen()
for all cases except simple non-wildcard addresses.