forked from inetaf/netaddr
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnetaddr.go
320 lines (289 loc) · 8.38 KB
/
netaddr.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
// Copyright 2020 The Inet.Af AUTHORS. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Package netaddr contains an IP address type.
//
// This is a work in progress. See https://github.com/inetaf/netaddr for background.
package netaddr // import "inet.af/netaddr"
import (
"bytes"
"errors"
"fmt"
"net"
"strings"
)
// Sizes: (64-bit)
// net.IP: 24 byte slice header + {4, 16} = 28 to 40 bytes
// net.IPAddr: 40 byte slice header + {4, 16} = 44 to 56 bytes + zone length
// netaddr.IP: 16 byte interface + {4, 16, 24} = 20, 32, 40 bytes + zone length
// IP represents an IPv4 or IPv6 address (with or without a scoped
// addressing zone), similar to Go's net.IP or net.IPAddr.
//
// Unlike net.IP or net.IPAddr, the netaddr.IP is a comparable value
// type (it supports == and can be a map key) and is immutable.
// Its memory representation ranges from 20 to 40 bytes, depending on
// whether the underlying adddress is IPv4, IPv6, or IPv6 with a
// zone. (This is smaller than the standard library's 28 to 56 bytes)
type IP struct {
ipImpl
}
// ipImpl is the interface representing either a v4addr, v6addr, v6ZoneAddr.
type ipImpl interface {
is4() bool
is6() bool
is4in6() bool
as16() [16]byte
String() string
}
type v4Addr [4]byte
func (v4Addr) is4() bool { return true }
func (v4Addr) is6() bool { return false }
func (v4Addr) is4in6() bool { return false }
func (ip v4Addr) as16() [16]byte {
return [16]byte{
10: 0xff,
11: 0xff,
12: ip[0],
13: ip[1],
14: ip[2],
15: ip[3],
}
}
func (ip v4Addr) String() string { return fmt.Sprintf("%d.%d.%d.%d", ip[0], ip[1], ip[2], ip[3]) }
// mapped4Prefix are the 12 leading bytes in a IPv4-mapped IPv6 address.
const mapped4Prefix = "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xff\xff"
type v6Addr [16]byte
func (v6Addr) is4() bool { return false }
func (v6Addr) is6() bool { return true }
func (ip v6Addr) is4in6() bool { return string(ip[:len(mapped4Prefix)]) == mapped4Prefix }
func (ip v6Addr) as16() [16]byte { return ip }
func (ip v6Addr) String() string {
// TODO: better implementation; don't jump through these hoops
// and pay these allocs just to share a bit of code with
// std. Just copy & modify it as needed.
if ip.is4in6() {
mod := ip
mod[10] = 0xfe // change to arbitrary byte that's not 0xff to hide it from Go
s := net.IP(mod[:]).String()
return strings.Replace(s, "::feff:", "::ffff:", 1)
}
return (&net.IPAddr{IP: net.IP(ip[:])}).String()
}
type v6AddrZone struct {
v6Addr
zone string
}
func (ip v6AddrZone) String() string {
// TODO: better implementation
return (&net.IPAddr{IP: net.IP(ip.v6Addr[:]), Zone: ip.zone}).String()
}
// ParseIP parses s as an IP address, returning the result. The string
// s can be in dotted decimal ("192.0.2.1"), IPv6 ("2001:db8::68"),
// or IPv6 with a scoped addressing zone ("fe80::1cc0:3e8c:119f:c2e1%ens18").
func ParseIP(s string) (IP, error) {
// TODO: do our own parsing to save some allocs? For now,
// while showing off new API & representation, just use the
// standard library's parsing.
ipa, err := net.ResolveIPAddr("ip", s)
if err != nil {
return IP{}, err
}
if !strings.Contains(s, ":") {
if ip4 := ipa.IP.To4(); ip4 != nil {
var v4 v4Addr
copy(v4[:], ip4)
return IP{v4}, nil
}
}
var v6 v6Addr
copy(v6[:], ipa.IP.To16())
if ipa.Zone != "" {
return IP{v6AddrZone{v6, ipa.Zone}}, nil
}
return IP{v6}, nil
}
// Zone returns ip's IPv6 scoped addressing zone, if any.
func (ip IP) Zone() string {
if v6z, ok := ip.ipImpl.(v6AddrZone); ok {
return v6z.zone
}
return ""
}
// Less reports whether ip sorts before ip2.
// IP addresses sort first by length, then their address.
// IPv6 addresses with zones sort just after the same address without a zone.
func (ip IP) Less(ip2 IP) bool {
a, b := ip, ip2
// Zero value sorts first.
if a.ipImpl == nil {
return b.ipImpl != nil
}
if b.ipImpl == nil {
return false
}
// At this point, a and b are both either v4 or v6.
if a4, ok := a.ipImpl.(v4Addr); ok {
if b4, ok := b.ipImpl.(v4Addr); ok {
return bytes.Compare(a4[:], b4[:]) < 0
}
// v4 sorts before v6.
return true
}
// At this point, a and b are both v6 or v6+zone.
a16 := a.ipImpl.as16()
b16 := b.ipImpl.as16()
switch bytes.Compare(a16[:], b16[:]) {
case -1:
return true
default:
return a.Zone() < b.Zone()
case 1:
return false
}
}
func (ip IP) ipZone() (stdIP net.IP, zone string) {
switch ip := ip.ipImpl.(type) {
case nil:
return nil, ""
case v4Addr:
return net.IP{ip[0], ip[1], ip[2], ip[3]}, ""
case v6Addr:
stdIP = make(net.IP, net.IPv6len)
copy(stdIP, ip[:])
return stdIP, ""
case v6AddrZone:
stdIP = make(net.IP, net.IPv6len)
copy(stdIP, ip.v6Addr[:])
return stdIP, ip.zone
default:
panic("netaddr: unhandled ipImpl representation")
}
}
// IPAddr returns the net.IPAddr representation of an IP. The returned value is
// always non-nil, but the IPAddr.IP will be nil if ip is the zero value.
// If ip contains a zone identifier, IPAddr.Zone is populated.
func (ip IP) IPAddr() *net.IPAddr {
stdIP, zone := ip.ipZone()
return &net.IPAddr{IP: stdIP, Zone: zone}
}
// Is4 reports whether ip is an IPv4 address.
//
// It returns false for IP4-mapped IPv6 addresses. See IP.Unmap.
func (ip IP) Is4() bool {
if ip.ipImpl == nil {
return false
}
return ip.ipImpl.is4()
}
// Is4in6 reports whether ip is an IPv4-mapped IPv6 address.
func (ip IP) Is4in6() bool {
if ip.ipImpl == nil {
return false
}
return ip.ipImpl.is4in6()
}
// Is6 reports whether ip is an IPv6 address, including IPv4-mapped
// IPv6 addresses.
func (ip IP) Is6() bool {
if ip.ipImpl == nil {
return false
}
return ip.ipImpl.is6()
}
// Unmap returns ip with any IPv4-mapped IPv6 address prefix removed.
//
// That is, if ip is an IPv6 address wrapping an IPv4 adddress, it
// returns the wrapped IPv4 address. Otherwise it returns ip, regardless
// of its type.
func (ip IP) Unmap() IP {
if !ip.Is4in6() {
return ip
}
a := ip.ipImpl.as16()
return IP{v4Addr{a[12], a[13], a[14], a[15]}}
}
// IsMulticast reports whether ip is a multicast address. If ip is the zero
// value, it will return false.
func (ip IP) IsMulticast() bool {
// See: https://en.wikipedia.org/wiki/Multicast_address.
switch ip := ip.ipImpl.(type) {
case nil:
return false
case v4Addr:
return ip[0]&0xf0 == 0xe0
case v6Addr:
return ip[0] == 0xff
case v6AddrZone:
return ip.v6Addr[0] == 0xff
default:
panic("netaddr: unhandled ipImpl representation")
}
}
// String returns the string form of the IP address ip.
// It returns one of 4 forms:
//
// - "invalid IP", if ip is the zero value
// - IPv4 dotted decimal ("192.0.2.1")
// - IPv6 ("2001:db8::1")
// - IPv6 with zone ("fe80:db8::1%eth0")
//
// Note that unlike the Go standard library's IP.String method,
// IP4-mapped IPv6 addresses do not format as dotted decimals.
func (ip IP) String() string {
if ip.ipImpl == nil {
return "invalid IP"
}
return ip.ipImpl.String()
}
// MarshalText implements the encoding.TextMarshaler interface,
// The encoding is the same as returned by String, with one exception:
// If ip is the zero value, the encoding is the empty string.
func (ip IP) MarshalText() ([]byte, error) {
if ip.ipImpl == nil {
return []byte(""), nil
}
return []byte(ip.String()), nil
}
// UnmarshalText implements the encoding.TextUnmarshaler interface.
// The IP address is expected in a form accepted by ParseIP.
// It returns an error if *ip is not the IP zero value.
func (ip *IP) UnmarshalText(text []byte) error {
if ip.ipImpl != nil {
return errors.New("netaddr: refusing to Unmarshal into non-zero IP")
}
if len(text) == 0 {
return nil
}
var err error
*ip, err = ParseIP(string(text))
return err
}
// IPPort is an IP & port number.
//
// It's meant to be used as a value type.
type IPPort struct {
IP IP
Port uint16
}
// UDPAddr returns a standard library net.UDPAddr from p.
// The returned value is always non-nil. If p.IP is the zero
// value, then UDPAddr.IP is nil.
func (p IPPort) UDPAddr() *net.UDPAddr {
ip, zone := p.IP.ipZone()
return &net.UDPAddr{
IP: ip,
Port: int(p.Port),
Zone: zone,
}
}
// TCPAddr returns a standard library net.UDPAddr from p.
// The returned value is always non-nil. If p.IP is the zero
// value, then TCPAddr.IP is nil.
func (p IPPort) TCPAddr() *net.TCPAddr {
ip, zone := p.IP.ipZone()
return &net.TCPAddr{
IP: ip,
Port: int(p.Port),
Zone: zone,
}
}