mirror of
https://github.com/tailscale/tailscale.git
synced 2024-11-25 19:15:34 +00:00
util/netconv: add package to convert between netip and netaddr types
Signed-off-by: Josh Bleecher Snyder <josh@tailscale.com>
This commit is contained in:
parent
5cb9999be3
commit
463728a885
63
util/netconv/netconv.go
Normal file
63
util/netconv/netconv.go
Normal file
@ -0,0 +1,63 @@
|
||||
// Copyright (c) 2022 Tailscale Inc & 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 netconv provides utilities to convert between netaddr and netip.
|
||||
// To convert from a net.IP, use the netaddr/netip API.
|
||||
package netconv
|
||||
|
||||
import (
|
||||
"net/netip"
|
||||
|
||||
"inet.af/netaddr"
|
||||
)
|
||||
|
||||
// AsIP returns a as a netaddr.IP.
|
||||
func AsIP(a netip.Addr) netaddr.IP {
|
||||
switch {
|
||||
case a.Is4():
|
||||
return netaddr.IPFrom4(a.As4())
|
||||
case a.Is6():
|
||||
return netaddr.IPv6Raw(a.As16()).WithZone(a.Zone())
|
||||
}
|
||||
return netaddr.IP{}
|
||||
}
|
||||
|
||||
// AsAddr returns a as a netip.IP.
|
||||
func AsAddr(a netaddr.IP) netip.Addr {
|
||||
switch {
|
||||
case a.Is4():
|
||||
return netip.AddrFrom4(a.As4())
|
||||
case a.Is6():
|
||||
return netip.AddrFrom16(a.As16()).WithZone(a.Zone())
|
||||
}
|
||||
return netip.Addr{}
|
||||
}
|
||||
|
||||
// AsIPPrefix returns a as a netaddr.IPPrefix.
|
||||
// If a has Bits of -1, indicating an invalid bits,
|
||||
// the returned IPPrefix will have Bits of 255.
|
||||
// AsIPPrefix and AsPrefix do not
|
||||
// round trip for invalid Bits values.
|
||||
func AsIPPrefix(a netip.Prefix) netaddr.IPPrefix {
|
||||
return netaddr.IPPrefixFrom(AsIP(a.Addr()), uint8(a.Bits()))
|
||||
}
|
||||
|
||||
// AsPrefix returns a as a netip.Prefix.
|
||||
// If a has an invalid Bits value,
|
||||
// the returned Prefix will have Bits of -1.
|
||||
// AsIPPrefix and AsPrefix do not
|
||||
// round trip for invalid Bits values.
|
||||
func AsPrefix(a netaddr.IPPrefix) netip.Prefix {
|
||||
return netip.PrefixFrom(AsAddr(a.IP()), int(a.Bits()))
|
||||
}
|
||||
|
||||
// AsIPPort returns a as a netaddr.IPPort.
|
||||
func AsIPPort(a netip.AddrPort) netaddr.IPPort {
|
||||
return netaddr.IPPortFrom(AsIP(a.Addr()), a.Port())
|
||||
}
|
||||
|
||||
// AsAddrPort returns a as a netip.AddrPort.
|
||||
func AsAddrPort(a netaddr.IPPort) netip.AddrPort {
|
||||
return netip.AddrPortFrom(AsAddr(a.IP()), a.Port())
|
||||
}
|
76
util/netconv/netconv_test.go
Normal file
76
util/netconv/netconv_test.go
Normal file
@ -0,0 +1,76 @@
|
||||
// Copyright (c) 2022 Tailscale Inc & 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 netconv
|
||||
|
||||
import (
|
||||
"net/netip"
|
||||
"testing"
|
||||
|
||||
qt "github.com/frankban/quicktest"
|
||||
"inet.af/netaddr"
|
||||
)
|
||||
|
||||
func TestAddr(t *testing.T) {
|
||||
c := qt.New(t)
|
||||
|
||||
c.Assert(netip.Addr{}, qt.Equals, AsAddr(netaddr.IP{}))
|
||||
c.Assert(netaddr.IP{}, qt.Equals, AsIP(netip.Addr{}))
|
||||
|
||||
// Cover IPv4, IPv6, 4in6, and zones.
|
||||
addrStrs := []string{
|
||||
"0.0.0.0",
|
||||
"123.45.67.89",
|
||||
"::",
|
||||
"fd7a:115c:a1e0:ab12:4843:cd96:626b:430b",
|
||||
"fd7a:115c:a1e0:ab12:4843:cd96:626b:430b%eth0",
|
||||
"::ffff:192.0.2.128",
|
||||
"::ffff:192.0.2.128%eth0",
|
||||
}
|
||||
for _, s := range addrStrs {
|
||||
ip := netaddr.MustParseIP(s)
|
||||
addr := netip.MustParseAddr(s)
|
||||
c.Assert(addr, qt.Equals, AsAddr(ip))
|
||||
c.Assert(ip, qt.Equals, AsIP(addr))
|
||||
}
|
||||
}
|
||||
|
||||
func TestAddrPort(t *testing.T) {
|
||||
c := qt.New(t)
|
||||
|
||||
c.Assert(netip.AddrPort{}, qt.Equals, AsAddrPort(netaddr.IPPort{}))
|
||||
c.Assert(netaddr.IPPort{}, qt.Equals, AsIPPort(netip.AddrPort{}))
|
||||
|
||||
// Test just a single AddrPort conversion;
|
||||
// there's almost nothing happening in the code.
|
||||
portStr := "1.2.4.5:8"
|
||||
ipPort := netaddr.MustParseIPPort(portStr)
|
||||
ap := netip.MustParseAddrPort(portStr)
|
||||
c.Assert(ipPort, qt.Equals, AsIPPort(ap))
|
||||
c.Assert(ap, qt.Equals, AsAddrPort(ipPort))
|
||||
}
|
||||
|
||||
func TestPrefix(t *testing.T) {
|
||||
c := qt.New(t)
|
||||
|
||||
// The interesting Prefix cases are invalid bits.
|
||||
addr := netip.MustParseAddr("1.2.3.4")
|
||||
ip := AsIP(addr)
|
||||
|
||||
tests := []struct {
|
||||
ipp netaddr.IPPrefix // input IPPrefix, output from converting pfx
|
||||
pfx netip.Prefix // input Prefix, output from converting ipp
|
||||
out netaddr.IPPrefix // output from converting pfx
|
||||
}{
|
||||
{netaddr.IPPrefix{}, netip.Prefix{}, netaddr.IPPrefix{}},
|
||||
{netaddr.IPPrefixFrom(ip, 24), netip.PrefixFrom(addr, 24), netaddr.IPPrefixFrom(ip, 24)},
|
||||
{netaddr.IPPrefixFrom(ip, 255), netip.PrefixFrom(addr, -1), netaddr.IPPrefixFrom(ip, 255)},
|
||||
{netaddr.IPPrefixFrom(ip, 204), netip.PrefixFrom(addr, -1), netaddr.IPPrefixFrom(ip, 255)},
|
||||
}
|
||||
|
||||
for _, test := range tests {
|
||||
c.Assert(test.out, qt.Equals, AsIPPrefix(test.pfx))
|
||||
c.Assert(test.pfx, qt.Equals, AsPrefix(test.ipp))
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user