Index: src/pkg/net/ipsock.go |
=================================================================== |
--- a/src/pkg/net/ipsock.go |
+++ b/src/pkg/net/ipsock.go |
@@ -68,15 +68,12 @@ |
func (e InvalidAddrError) Timeout() bool { return false } |
func (e InvalidAddrError) Temporary() bool { return false } |
-// SplitHostPort splits a network address of the form |
-// "host:port" or "[host]:port" into host and port. |
-// The latter form must be used when host contains a colon. |
+// SplitHostPort splits a network address of the form "host:port", |
+// "[host]:port" or "[ipv6-host%zone]:port" into host or |
+// ipv6-host%zone and port. A literal address or host name for IPv6 |
+// must be enclosed in square brackets, as in "[::1]:80", |
+// "[ipv6-host]:http" or "[ipv6-host%zone]:80". |
func SplitHostPort(hostport string) (host, port string, err error) { |
- host, port, _, err = splitHostPort(hostport) |
- return |
-} |
- |
-func splitHostPort(hostport string) (host, port, zone string, err error) { |
j, k := 0, 0 |
// The port starts after the last colon. |
@@ -110,10 +107,12 @@ |
j, k = 1, end+1 // there can't be a '[' resp. ']' before these positions |
} else { |
host = hostport[:i] |
- |
if byteIndex(host, ':') >= 0 { |
goto tooManyColons |
} |
+ if byteIndex(host, '%') >= 0 { |
+ goto missingBrackets |
+ } |
} |
if byteIndex(hostport[j:], '[') >= 0 { |
err = &AddrError{"unexpected '[' in address", hostport} |
@@ -134,13 +133,29 @@ |
tooManyColons: |
err = &AddrError{"too many colons in address", hostport} |
return |
+ |
+missingBrackets: |
+ err = &AddrError{"missing brackets in address", hostport} |
+ return |
} |
-// JoinHostPort combines host and port into a network address |
-// of the form "host:port" or, if host contains a colon, "[host]:port". |
+func splitHostZone(s string) (host, zone string) { |
+ // The IPv6 scoped addressing zone identifer starts after the |
+ // last percent sign. |
+ if i := last(s, '%'); i > 0 { |
+ host, zone = s[:i], s[i+1:] |
+ } else { |
+ host = s |
+ } |
+ return |
+} |
+ |
+// JoinHostPort combines host and port into a network address of the |
+// form "host:port" or, if host contains a colon or a percent sign, |
+// "[host]:port". |
func JoinHostPort(host, port string) string { |
- // If host has colons, have to bracket it. |
- if byteIndex(host, ':') >= 0 { |
+ // If host has colons or a percent sign, have to bracket it. |
+ if byteIndex(host, ':') >= 0 || byteIndex(host, '%') >= 0 { |
return "[" + host + "]:" + port |
} |
return host + ":" + port |
@@ -155,7 +170,7 @@ |
switch net { |
case "tcp", "tcp4", "tcp6", "udp", "udp4", "udp6": |
if addr != "" { |
- if host, port, zone, err = splitHostPort(addr); err != nil { |
+ if host, port, err = SplitHostPort(addr); err != nil { |
return nil, err |
} |
if portnum, err = parsePort(net, port); err != nil { |
@@ -184,21 +199,25 @@ |
return inetaddr(net, nil, portnum, zone), nil |
} |
// Try as an IP address. |
- if ip := ParseIP(host); ip != nil { |
+ if ip := parseIPv4(host); ip != nil { |
return inetaddr(net, ip, portnum, zone), nil |
} |
+ if ip, zone := parseIPv6(host, true); ip != nil { |
+ return inetaddr(net, ip, portnum, zone), nil |
+ } |
+ // Try as a domain name. |
+ host, zone = splitHostZone(host) |
+ addrs, err := lookupHostDeadline(host, deadline) |
+ if err != nil { |
+ return nil, err |
+ } |
var filter func(IP) IP |
if net != "" && net[len(net)-1] == '4' { |
filter = ipv4only |
} |
- if net != "" && net[len(net)-1] == '6' { |
+ if net != "" && net[len(net)-1] == '6' || zone != "" { |
filter = ipv6only |
} |
- // Try as a DNS name. |
- addrs, err := lookupHostDeadline(host, deadline) |
- if err != nil { |
- return nil, err |
- } |
ip := firstFavoriteAddr(filter, addrs) |
if ip == nil { |
// should not happen |