use netipx.IPSet for matcher
Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com>
This commit is contained in:
parent
ccedf276ab
commit
56b63c6e10
1 changed files with 41 additions and 35 deletions
68
acls.go
68
acls.go
|
@ -809,8 +809,9 @@ func (pol *ACLPolicy) getIPsFromIPPrefix(
|
||||||
return lo.Uniq(val), nil
|
return lo.Uniq(val), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is borrowed from
|
// This is borrowed from, and updated to use IPSet
|
||||||
// https://github.com/tailscale/tailscale/blob/71029cea2ddf82007b80f465b256d027eab0f02d/wgengine/filter/tailcfg.go#L97-L162
|
// https://github.com/tailscale/tailscale/blob/71029cea2ddf82007b80f465b256d027eab0f02d/wgengine/filter/tailcfg.go#L97-L162
|
||||||
|
// TODO(kradalby): contribute upstream and make public
|
||||||
var (
|
var (
|
||||||
zeroIP4 = netip.AddrFrom4([4]byte{})
|
zeroIP4 = netip.AddrFrom4([4]byte{})
|
||||||
zeroIP6 = netip.AddrFrom16([16]byte{})
|
zeroIP6 = netip.AddrFrom16([16]byte{})
|
||||||
|
@ -825,19 +826,14 @@ var (
|
||||||
//
|
//
|
||||||
// bits, if non-nil, is the legacy SrcBits CIDR length to make a IP
|
// bits, if non-nil, is the legacy SrcBits CIDR length to make a IP
|
||||||
// address (without a slash) treated as a CIDR of *bits length.
|
// address (without a slash) treated as a CIDR of *bits length.
|
||||||
//
|
|
||||||
// TODO(bradfitz): make this return an IPSet and plumb that all
|
|
||||||
// around, and ultimately use a new version of IPSet.ContainsFunc like
|
|
||||||
// Contains16Func that works in [16]byte address, so we we can match
|
|
||||||
// at runtime without allocating?
|
|
||||||
// nolint
|
// nolint
|
||||||
func parseIPSet(arg string, bits *int) ([]netip.Prefix, error) {
|
func parseIPSet(arg string, bits *int) (*netipx.IPSet, error) {
|
||||||
|
var ipSet netipx.IPSetBuilder
|
||||||
if arg == "*" {
|
if arg == "*" {
|
||||||
// User explicitly requested wildcard.
|
ipSet.AddPrefix(netip.PrefixFrom(zeroIP4, 0))
|
||||||
return []netip.Prefix{
|
ipSet.AddPrefix(netip.PrefixFrom(zeroIP6, 0))
|
||||||
netip.PrefixFrom(zeroIP4, 0),
|
|
||||||
netip.PrefixFrom(zeroIP6, 0),
|
return ipSet.IPSet()
|
||||||
}, nil
|
|
||||||
}
|
}
|
||||||
if strings.Contains(arg, "/") {
|
if strings.Contains(arg, "/") {
|
||||||
pfx, err := netip.ParsePrefix(arg)
|
pfx, err := netip.ParsePrefix(arg)
|
||||||
|
@ -848,7 +844,9 @@ func parseIPSet(arg string, bits *int) ([]netip.Prefix, error) {
|
||||||
return nil, fmt.Errorf("%v contains non-network bits set", pfx)
|
return nil, fmt.Errorf("%v contains non-network bits set", pfx)
|
||||||
}
|
}
|
||||||
|
|
||||||
return []netip.Prefix{pfx}, nil
|
ipSet.AddPrefix(pfx)
|
||||||
|
|
||||||
|
return ipSet.IPSet()
|
||||||
}
|
}
|
||||||
if strings.Count(arg, "-") == 1 {
|
if strings.Count(arg, "-") == 1 {
|
||||||
ip1s, ip2s, _ := strings.Cut(arg, "-")
|
ip1s, ip2s, _ := strings.Cut(arg, "-")
|
||||||
|
@ -868,7 +866,11 @@ func parseIPSet(arg string, bits *int) ([]netip.Prefix, error) {
|
||||||
return nil, fmt.Errorf("invalid IP range %q", arg)
|
return nil, fmt.Errorf("invalid IP range %q", arg)
|
||||||
}
|
}
|
||||||
|
|
||||||
return r.Prefixes(), nil
|
for _, prefix := range r.Prefixes() {
|
||||||
|
ipSet.AddPrefix(prefix)
|
||||||
|
}
|
||||||
|
|
||||||
|
return ipSet.IPSet()
|
||||||
}
|
}
|
||||||
ip, err := netip.ParseAddr(arg)
|
ip, err := netip.ParseAddr(arg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -882,55 +884,59 @@ func parseIPSet(arg string, bits *int) ([]netip.Prefix, error) {
|
||||||
bits8 = uint8(*bits)
|
bits8 = uint8(*bits)
|
||||||
}
|
}
|
||||||
|
|
||||||
return []netip.Prefix{netip.PrefixFrom(ip, int(bits8))}, nil
|
ipSet.AddPrefix(netip.PrefixFrom(ip, int(bits8)))
|
||||||
|
|
||||||
|
return ipSet.IPSet()
|
||||||
}
|
}
|
||||||
|
|
||||||
type Match struct {
|
type Match struct {
|
||||||
Srcs []netip.Prefix
|
Srcs *netipx.IPSet
|
||||||
Dests []netip.Prefix
|
Dests *netipx.IPSet
|
||||||
}
|
}
|
||||||
|
|
||||||
func MatchFromFilterRule(rule tailcfg.FilterRule) Match {
|
func MatchFromFilterRule(rule tailcfg.FilterRule) Match {
|
||||||
match := Match{
|
srcs := new(netipx.IPSetBuilder)
|
||||||
Srcs: []netip.Prefix{},
|
dests := new(netipx.IPSetBuilder)
|
||||||
Dests: []netip.Prefix{},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, srcIP := range rule.SrcIPs {
|
for _, srcIP := range rule.SrcIPs {
|
||||||
prefix, _ := parseIPSet(srcIP, nil)
|
set, _ := parseIPSet(srcIP, nil)
|
||||||
|
|
||||||
match.Srcs = append(match.Srcs, prefix...)
|
srcs.AddSet(set)
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, dest := range rule.DstPorts {
|
for _, dest := range rule.DstPorts {
|
||||||
prefix, _ := parseIPSet(dest.IP, nil)
|
set, _ := parseIPSet(dest.IP, nil)
|
||||||
|
|
||||||
match.Dests = append(match.Dests, prefix...)
|
dests.AddSet(set)
|
||||||
|
}
|
||||||
|
|
||||||
|
srcsSet, _ := srcs.IPSet()
|
||||||
|
destsSet, _ := dests.IPSet()
|
||||||
|
|
||||||
|
match := Match{
|
||||||
|
Srcs: srcsSet,
|
||||||
|
Dests: destsSet,
|
||||||
}
|
}
|
||||||
|
|
||||||
return match
|
return match
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *Match) SrcsContainsIPs(ips []netip.Addr) bool {
|
func (m *Match) SrcsContainsIPs(ips []netip.Addr) bool {
|
||||||
for _, prefix := range m.Srcs {
|
|
||||||
for _, ip := range ips {
|
for _, ip := range ips {
|
||||||
if prefix.Contains(ip) {
|
if m.Srcs.Contains(ip) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *Match) DestsContainsIP(ips []netip.Addr) bool {
|
func (m *Match) DestsContainsIP(ips []netip.Addr) bool {
|
||||||
for _, prefix := range m.Dests {
|
|
||||||
for _, ip := range ips {
|
for _, ip := range ips {
|
||||||
if prefix.Contains(ip) {
|
if m.Dests.Contains(ip) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue