2020-11-09 21:22:36 -08:00
|
|
|
// Copyright (c) 2020 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 filter
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2020-12-14 08:21:41 -08:00
|
|
|
"strings"
|
2020-11-09 21:22:36 -08:00
|
|
|
|
|
|
|
"inet.af/netaddr"
|
|
|
|
"tailscale.com/tailcfg"
|
2021-03-19 21:05:51 -07:00
|
|
|
"tailscale.com/types/ipproto"
|
2020-11-09 21:22:36 -08:00
|
|
|
)
|
|
|
|
|
2021-03-19 21:05:51 -07:00
|
|
|
var defaultProtos = []ipproto.Proto{
|
2021-03-20 21:45:47 -07:00
|
|
|
ipproto.TCP,
|
|
|
|
ipproto.UDP,
|
|
|
|
ipproto.ICMPv4,
|
|
|
|
ipproto.ICMPv6,
|
2021-03-17 14:24:32 -07:00
|
|
|
}
|
|
|
|
|
2020-11-09 21:22:36 -08:00
|
|
|
// MatchesFromFilterRules converts tailcfg FilterRules into Matches.
|
|
|
|
// If an error is returned, the Matches result is still valid,
|
|
|
|
// containing the rules that were successfully converted.
|
2020-11-09 21:33:41 -08:00
|
|
|
func MatchesFromFilterRules(pf []tailcfg.FilterRule) ([]Match, error) {
|
2020-11-09 21:22:36 -08:00
|
|
|
mm := make([]Match, 0, len(pf))
|
|
|
|
var erracc error
|
|
|
|
|
|
|
|
for _, r := range pf {
|
|
|
|
m := Match{}
|
|
|
|
|
2021-03-17 14:24:32 -07:00
|
|
|
if len(r.IPProto) == 0 {
|
2021-03-19 21:05:51 -07:00
|
|
|
m.IPProto = append([]ipproto.Proto(nil), defaultProtos...)
|
2021-03-17 14:24:32 -07:00
|
|
|
} else {
|
2021-03-19 21:05:51 -07:00
|
|
|
m.IPProto = make([]ipproto.Proto, 0, len(r.IPProto))
|
2021-03-17 14:24:32 -07:00
|
|
|
for _, n := range r.IPProto {
|
|
|
|
if n >= 0 && n <= 0xff {
|
2021-03-19 21:05:51 -07:00
|
|
|
m.IPProto = append(m.IPProto, ipproto.Proto(n))
|
2021-03-17 14:24:32 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-09 21:22:36 -08:00
|
|
|
for i, s := range r.SrcIPs {
|
2020-12-14 08:21:41 -08:00
|
|
|
var bits *int
|
2020-11-09 21:22:36 -08:00
|
|
|
if len(r.SrcBits) > i {
|
2020-12-14 08:21:41 -08:00
|
|
|
bits = &r.SrcBits[i]
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
2020-12-14 08:21:41 -08:00
|
|
|
nets, err := parseIPSet(s, bits)
|
2020-11-09 21:22:36 -08:00
|
|
|
if err != nil && erracc == nil {
|
|
|
|
erracc = err
|
|
|
|
continue
|
|
|
|
}
|
2020-11-12 14:14:11 -08:00
|
|
|
m.Srcs = append(m.Srcs, nets...)
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, d := range r.DstPorts {
|
2020-12-14 08:21:41 -08:00
|
|
|
nets, err := parseIPSet(d.IP, d.Bits)
|
2020-11-09 21:22:36 -08:00
|
|
|
if err != nil && erracc == nil {
|
|
|
|
erracc = err
|
|
|
|
continue
|
|
|
|
}
|
2020-11-12 14:14:11 -08:00
|
|
|
for _, net := range nets {
|
|
|
|
m.Dsts = append(m.Dsts, NetPortRange{
|
|
|
|
Net: net,
|
|
|
|
Ports: PortRange{
|
|
|
|
First: d.Ports.First,
|
|
|
|
Last: d.Ports.Last,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
mm = append(mm, m)
|
|
|
|
}
|
|
|
|
return mm, erracc
|
|
|
|
}
|
|
|
|
|
2020-11-10 23:23:17 -08:00
|
|
|
var (
|
|
|
|
zeroIP4 = netaddr.IPv4(0, 0, 0, 0)
|
|
|
|
zeroIP6 = netaddr.IPFrom16([16]byte{})
|
|
|
|
)
|
|
|
|
|
2020-12-14 08:21:41 -08:00
|
|
|
// parseIPSet parses arg as one:
|
|
|
|
//
|
|
|
|
// * an IP address (IPv4 or IPv6)
|
|
|
|
// * the string "*" to match everything (both IPv4 & IPv6)
|
|
|
|
// * a CIDR (e.g. "192.168.0.0/16")
|
|
|
|
// * a range of two IPs, inclusive, separated by hyphen ("2eff::1-2eff::0800")
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
// 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?
|
|
|
|
func parseIPSet(arg string, bits *int) ([]netaddr.IPPrefix, error) {
|
|
|
|
if arg == "*" {
|
|
|
|
// User explicitly requested wildcard.
|
2020-11-12 14:14:11 -08:00
|
|
|
return []netaddr.IPPrefix{
|
2021-05-14 18:07:28 -07:00
|
|
|
netaddr.IPPrefixFrom(zeroIP4, 0),
|
|
|
|
netaddr.IPPrefixFrom(zeroIP6, 0),
|
2020-11-12 14:14:11 -08:00
|
|
|
}, nil
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
2020-12-14 08:21:41 -08:00
|
|
|
if strings.Contains(arg, "/") {
|
|
|
|
pfx, err := netaddr.ParseIPPrefix(arg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if pfx != pfx.Masked() {
|
|
|
|
return nil, fmt.Errorf("%v contains non-network bits set", pfx)
|
|
|
|
}
|
|
|
|
return []netaddr.IPPrefix{pfx}, nil
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
2020-12-14 08:21:41 -08:00
|
|
|
if strings.Count(arg, "-") == 1 {
|
2022-03-16 14:25:31 -07:00
|
|
|
ip1s, ip2s, _ := strings.Cut(arg, "-")
|
2020-12-14 08:21:41 -08:00
|
|
|
ip1, err := netaddr.ParseIP(ip1s)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
ip2, err := netaddr.ParseIP(ip2s)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-05-14 18:07:28 -07:00
|
|
|
r := netaddr.IPRangeFrom(ip1, ip2)
|
2020-12-14 08:21:41 -08:00
|
|
|
if !r.Valid() {
|
|
|
|
return nil, fmt.Errorf("invalid IP range %q", arg)
|
|
|
|
}
|
|
|
|
return r.Prefixes(), nil
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
2020-12-14 08:21:41 -08:00
|
|
|
ip, err := netaddr.ParseIP(arg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("invalid IP address %q", arg)
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
2020-12-15 12:12:56 -08:00
|
|
|
bits8 := ip.BitLen()
|
|
|
|
if bits != nil {
|
|
|
|
if *bits < 0 || *bits > int(bits8) {
|
|
|
|
return nil, fmt.Errorf("invalid CIDR size %d for IP %q", *bits, arg)
|
2020-12-14 08:21:41 -08:00
|
|
|
}
|
2020-12-15 12:12:56 -08:00
|
|
|
bits8 = uint8(*bits)
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|
2021-05-14 18:07:28 -07:00
|
|
|
return []netaddr.IPPrefix{netaddr.IPPrefixFrom(ip, bits8)}, nil
|
2020-11-09 21:22:36 -08:00
|
|
|
}
|