LEFT | RIGHT |
1 // Copyright 2009 The Go Authors. All rights reserved. | 1 // Copyright 2009 The Go Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style | 2 // Use of this source code is governed by a BSD-style |
3 // license that can be found in the LICENSE file. | 3 // license that can be found in the LICENSE file. |
4 | 4 |
5 // Internet protocol family sockets | 5 // Internet protocol family sockets |
6 | 6 |
7 package net | 7 package net |
8 | 8 |
9 import "time" | 9 import "time" |
10 | 10 |
(...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
62 return nil | 62 return nil |
63 } | 63 } |
64 | 64 |
65 type InvalidAddrError string | 65 type InvalidAddrError string |
66 | 66 |
67 func (e InvalidAddrError) Error() string { return string(e) } | 67 func (e InvalidAddrError) Error() string { return string(e) } |
68 func (e InvalidAddrError) Timeout() bool { return false } | 68 func (e InvalidAddrError) Timeout() bool { return false } |
69 func (e InvalidAddrError) Temporary() bool { return false } | 69 func (e InvalidAddrError) Temporary() bool { return false } |
70 | 70 |
71 // SplitHostPort splits a network address of the form "host:port", | 71 // SplitHostPort splits a network address of the form "host:port", |
72 // "host%ipv6zoneid:port", "[host]:port" or "[host%ipv6zoneid]:port" | 72 // "[host]:port" or "[ipv6-host%zone]:port" into host or |
73 // into host and port. A literal IPv6 host address must be enclosed | 73 // ipv6-host%zone and port. A literal address or host name for IPv6 |
74 // in square brackets, as in "[::]:80" or "[fe80::1%en0]:80". | 74 // must be enclosed in square brackets, as in "[::1]:80", |
| 75 // "[ipv6-host]:http" or "[ipv6-host%zone]:80". |
75 func SplitHostPort(hostport string) (host, port string, err error) { | 76 func SplitHostPort(hostport string) (host, port string, err error) { |
76 » host, _, port, err = splitHostPort(hostport) | 77 » j, k := 0, 0 |
77 » return | 78 |
78 } | |
79 | |
80 func splitHostPort(hostport string) (host, id, port string, err error) { | |
81 // The port starts after the last colon. | 79 // The port starts after the last colon. |
82 i := last(hostport, ':') | 80 i := last(hostport, ':') |
83 if i < 0 { | 81 if i < 0 { |
84 » » err = &AddrError{"missing port in address", hostport} | 82 » » goto missingPort |
| 83 » } |
| 84 |
| 85 » if hostport[0] == '[' { |
| 86 » » // Expect the first ']' just before the last ':'. |
| 87 » » end := byteIndex(hostport, ']') |
| 88 » » if end < 0 { |
| 89 » » » err = &AddrError{"missing ']' in address", hostport} |
| 90 » » » return |
| 91 » » } |
| 92 » » switch end + 1 { |
| 93 » » case len(hostport): |
| 94 » » » // There can't be a ':' behind the ']' now. |
| 95 » » » goto missingPort |
| 96 » » case i: |
| 97 » » » // The expected result. |
| 98 » » default: |
| 99 » » » // Either ']' isn't followed by a colon, or it is |
| 100 » » » // followed by a colon that is not the last one. |
| 101 » » » if hostport[end+1] == ':' { |
| 102 » » » » goto tooManyColons |
| 103 » » » } |
| 104 » » » goto missingPort |
| 105 » » } |
| 106 » » host = hostport[1:end] |
| 107 » » j, k = 1, end+1 // there can't be a '[' resp. ']' before these p
ositions |
| 108 » } else { |
| 109 » » host = hostport[:i] |
| 110 » » if byteIndex(host, ':') >= 0 { |
| 111 » » » goto tooManyColons |
| 112 » » } |
| 113 » » if byteIndex(host, '%') >= 0 { |
| 114 » » » goto missingBrackets |
| 115 » » } |
| 116 » } |
| 117 » if byteIndex(hostport[j:], '[') >= 0 { |
| 118 » » err = &AddrError{"unexpected '[' in address", hostport} |
85 return | 119 return |
86 } | 120 } |
87 » host, port = hostport[:i], hostport[i+1:] | 121 » if byteIndex(hostport[k:], ']') >= 0 { |
88 » // Can put brackets around host ... | 122 » » err = &AddrError{"unexpected ']' in address", hostport} |
89 » if len(host) > 0 && host[0] == '[' && host[len(host)-1] == ']' { | 123 » » return |
90 » » host = host[1 : len(host)-1] | 124 » } |
91 » } else { | 125 |
92 » » // ... but if there are no brackets, no colons. | 126 » port = hostport[i+1:] |
93 » » if byteIndex(host, ':') >= 0 { | 127 » return |
94 » » » err = &AddrError{"too many colons in address", hostport} | 128 |
95 » » » return | 129 missingPort: |
96 » » } | 130 » err = &AddrError{"missing port in address", hostport} |
97 » } | 131 » return |
| 132 |
| 133 tooManyColons: |
| 134 » err = &AddrError{"too many colons in address", hostport} |
| 135 » return |
| 136 |
| 137 missingBrackets: |
| 138 » err = &AddrError{"missing brackets in address", hostport} |
| 139 » return |
| 140 } |
| 141 |
| 142 func splitHostZone(s string) (host, zone string) { |
98 // The IPv6 scoped addressing zone identifer starts after the | 143 // The IPv6 scoped addressing zone identifer starts after the |
99 // last percent sign. | 144 // last percent sign. |
100 » if i := last(host, '%'); i > 0 { | 145 » if i := last(s, '%'); i > 0 { |
101 » » host, id = host[:i], host[i+1:] | 146 » » host, zone = s[:i], s[i+1:] |
| 147 » } else { |
| 148 » » host = s |
102 } | 149 } |
103 return | 150 return |
104 } | 151 } |
105 | 152 |
106 // JoinHostPort combines host and port into a network address of the | 153 // JoinHostPort combines host and port into a network address of the |
107 // form "host:port". A literal IPv6 host address must be enclosed in | 154 // form "host:port" or, if host contains a colon or a percent sign, |
108 // square brackets, as in "[::]:80". | 155 // "[host]:port". |
109 func JoinHostPort(host, port string) string { | 156 func JoinHostPort(host, port string) string { |
110 » // If host has colons, have to bracket it. | 157 » // If host has colons or a percent sign, have to bracket it. |
111 » if byteIndex(host, ':') >= 0 { | 158 » if byteIndex(host, ':') >= 0 || byteIndex(host, '%') >= 0 { |
112 return "[" + host + "]:" + port | 159 return "[" + host + "]:" + port |
113 } | 160 } |
114 return host + ":" + port | 161 return host + ":" + port |
115 } | 162 } |
116 | 163 |
117 func resolveInternetAddr(net, litAddr string, deadline time.Time) (Addr, error)
{ | 164 func resolveInternetAddr(net, addr string, deadline time.Time) (Addr, error) { |
118 var ( | 165 var ( |
119 » » err error | 166 » » err error |
120 » » host, id, port string | 167 » » host, port, zone string |
121 » » portnum int | 168 » » portnum int |
122 ) | 169 ) |
123 switch net { | 170 switch net { |
124 case "tcp", "tcp4", "tcp6", "udp", "udp4", "udp6": | 171 case "tcp", "tcp4", "tcp6", "udp", "udp4", "udp6": |
125 » » if litAddr != "" { | 172 » » if addr != "" { |
126 » » » if host, id, port, err = splitHostPort(litAddr); err !=
nil { | 173 » » » if host, port, err = SplitHostPort(addr); err != nil { |
127 return nil, err | 174 return nil, err |
128 } | 175 } |
129 if portnum, err = parsePort(net, port); err != nil { | 176 if portnum, err = parsePort(net, port); err != nil { |
130 return nil, err | 177 return nil, err |
131 } | 178 } |
132 } | 179 } |
133 case "ip", "ip4", "ip6": | 180 case "ip", "ip4", "ip6": |
134 » » if litAddr != "" { | 181 » » if addr != "" { |
135 » » » // The IPv6 scoped addressing zone identifer | 182 » » » host = addr |
136 » » » // starts after the last percent sign. | |
137 » » » if i := last(litAddr, '%'); i > 0 { | |
138 » » » » host, id = litAddr[:i], litAddr[i+1:] | |
139 » » » } else { | |
140 » » » » host = litAddr | |
141 » » » } | |
142 } | 183 } |
143 default: | 184 default: |
144 return nil, UnknownNetworkError(net) | 185 return nil, UnknownNetworkError(net) |
145 } | 186 } |
146 » inetaddr := func(net string, ip IP, port, id int) Addr { | 187 » inetaddr := func(net string, ip IP, port int, zone string) Addr { |
147 switch net { | 188 switch net { |
148 case "tcp", "tcp4", "tcp6": | 189 case "tcp", "tcp4", "tcp6": |
149 » » » return &TCPAddr{IP: ip, Port: port, zoneId: id} | 190 » » » return &TCPAddr{IP: ip, Port: port, Zone: zone} |
150 case "udp", "udp4", "udp6": | 191 case "udp", "udp4", "udp6": |
151 » » » return &UDPAddr{IP: ip, Port: port, zoneId: id} | 192 » » » return &UDPAddr{IP: ip, Port: port, Zone: zone} |
152 case "ip", "ip4", "ip6": | 193 case "ip", "ip4", "ip6": |
153 » » » return &IPAddr{IP: ip, zoneId: id} | 194 » » » return &IPAddr{IP: ip, Zone: zone} |
154 } | 195 } |
155 return nil | 196 return nil |
156 } | 197 } |
157 if host == "" { | 198 if host == "" { |
158 » » return inetaddr(net, nil, portnum, parseZoneId(id)), nil | 199 » » return inetaddr(net, nil, portnum, zone), nil |
159 } | 200 } |
160 // Try as an IP address. | 201 // Try as an IP address. |
161 » if ip := ParseIP(host); ip != nil { | 202 » if ip := parseIPv4(host); ip != nil { |
162 » » return inetaddr(net, ip, portnum, parseZoneId(id)), nil | 203 » » return inetaddr(net, ip, portnum, zone), nil |
| 204 » } |
| 205 » if ip, zone := parseIPv6(host, true); ip != nil { |
| 206 » » return inetaddr(net, ip, portnum, zone), nil |
| 207 » } |
| 208 » // Try as a domain name. |
| 209 » host, zone = splitHostZone(host) |
| 210 » addrs, err := lookupHostDeadline(host, deadline) |
| 211 » if err != nil { |
| 212 » » return nil, err |
163 } | 213 } |
164 var filter func(IP) IP | 214 var filter func(IP) IP |
165 if net != "" && net[len(net)-1] == '4' { | 215 if net != "" && net[len(net)-1] == '4' { |
166 filter = ipv4only | 216 filter = ipv4only |
167 } | 217 } |
168 » if net != "" && net[len(net)-1] == '6' { | 218 » if net != "" && net[len(net)-1] == '6' || zone != "" { |
169 filter = ipv6only | 219 filter = ipv6only |
170 } | |
171 // Try as a DNS name. | |
172 addrs, err := lookupHostDeadline(host, deadline) | |
173 if err != nil { | |
174 return nil, err | |
175 } | 220 } |
176 ip := firstFavoriteAddr(filter, addrs) | 221 ip := firstFavoriteAddr(filter, addrs) |
177 if ip == nil { | 222 if ip == nil { |
178 // should not happen | 223 // should not happen |
179 return nil, &AddrError{"LookupHost returned no suitable address"
, addrs[0]} | 224 return nil, &AddrError{"LookupHost returned no suitable address"
, addrs[0]} |
180 } | 225 } |
181 » return inetaddr(net, ip, portnum, parseZoneId(id)), nil | 226 » return inetaddr(net, ip, portnum, zone), nil |
182 } | 227 } |
183 | 228 |
184 func parseZoneId(id string) int { | 229 func zoneToString(zone int) string { |
185 » if id == "" { | 230 » if zone == 0 { |
| 231 » » return "" |
| 232 » } |
| 233 » if ifi, err := InterfaceByIndex(zone); err == nil { |
| 234 » » return ifi.Name |
| 235 » } |
| 236 » return itod(uint(zone)) |
| 237 } |
| 238 |
| 239 func zoneToInt(zone string) int { |
| 240 » if zone == "" { |
186 return 0 | 241 return 0 |
187 } | 242 } |
188 » if ifi, err := InterfaceByName(id); err == nil { | 243 » if ifi, err := InterfaceByName(zone); err == nil { |
189 return ifi.Index | 244 return ifi.Index |
190 } | 245 } |
191 » n, _, _ := dtoi(id, 0) | 246 » n, _, _ := dtoi(zone, 0) |
192 return n | 247 return n |
193 } | 248 } |
LEFT | RIGHT |