Add IPNetSlice and unit tests (#170)

diff --git a/ipnet_slice.go b/ipnet_slice.go
new file mode 100644
index 0000000..6b541aa
--- /dev/null
+++ b/ipnet_slice.go
@@ -0,0 +1,147 @@
+package pflag
+
+import (
+	"fmt"
+	"io"
+	"net"
+	"strings"
+)
+
+// -- ipNetSlice Value
+type ipNetSliceValue struct {
+	value   *[]net.IPNet
+	changed bool
+}
+
+func newIPNetSliceValue(val []net.IPNet, p *[]net.IPNet) *ipNetSliceValue {
+	ipnsv := new(ipNetSliceValue)
+	ipnsv.value = p
+	*ipnsv.value = val
+	return ipnsv
+}
+
+// Set converts, and assigns, the comma-separated IPNet argument string representation as the []net.IPNet value of this flag.
+// If Set is called on a flag that already has a []net.IPNet assigned, the newly converted values will be appended.
+func (s *ipNetSliceValue) Set(val string) error {
+
+	// remove all quote characters
+	rmQuote := strings.NewReplacer(`"`, "", `'`, "", "`", "")
+
+	// read flag arguments with CSV parser
+	ipNetStrSlice, err := readAsCSV(rmQuote.Replace(val))
+	if err != nil && err != io.EOF {
+		return err
+	}
+
+	// parse ip values into slice
+	out := make([]net.IPNet, 0, len(ipNetStrSlice))
+	for _, ipNetStr := range ipNetStrSlice {
+		_, n, err := net.ParseCIDR(strings.TrimSpace(ipNetStr))
+		if err != nil {
+			return fmt.Errorf("invalid string being converted to CIDR: %s", ipNetStr)
+		}
+		out = append(out, *n)
+	}
+
+	if !s.changed {
+		*s.value = out
+	} else {
+		*s.value = append(*s.value, out...)
+	}
+
+	s.changed = true
+
+	return nil
+}
+
+// Type returns a string that uniquely represents this flag's type.
+func (s *ipNetSliceValue) Type() string {
+	return "ipNetSlice"
+}
+
+// String defines a "native" format for this net.IPNet slice flag value.
+func (s *ipNetSliceValue) String() string {
+
+	ipNetStrSlice := make([]string, len(*s.value))
+	for i, n := range *s.value {
+		ipNetStrSlice[i] = n.String()
+	}
+
+	out, _ := writeAsCSV(ipNetStrSlice)
+	return "[" + out + "]"
+}
+
+func ipNetSliceConv(val string) (interface{}, error) {
+	val = strings.Trim(val, "[]")
+	// Emtpy string would cause a slice with one (empty) entry
+	if len(val) == 0 {
+		return []net.IPNet{}, nil
+	}
+	ss := strings.Split(val, ",")
+	out := make([]net.IPNet, len(ss))
+	for i, sval := range ss {
+		_, n, err := net.ParseCIDR(strings.TrimSpace(sval))
+		if err != nil {
+			return nil, fmt.Errorf("invalid string being converted to CIDR: %s", sval)
+		}
+		out[i] = *n
+	}
+	return out, nil
+}
+
+// GetIPNetSlice returns the []net.IPNet value of a flag with the given name
+func (f *FlagSet) GetIPNetSlice(name string) ([]net.IPNet, error) {
+	val, err := f.getFlagType(name, "ipNetSlice", ipNetSliceConv)
+	if err != nil {
+		return []net.IPNet{}, err
+	}
+	return val.([]net.IPNet), nil
+}
+
+// IPNetSliceVar defines a ipNetSlice flag with specified name, default value, and usage string.
+// The argument p points to a []net.IPNet variable in which to store the value of the flag.
+func (f *FlagSet) IPNetSliceVar(p *[]net.IPNet, name string, value []net.IPNet, usage string) {
+	f.VarP(newIPNetSliceValue(value, p), name, "", usage)
+}
+
+// IPNetSliceVarP is like IPNetSliceVar, but accepts a shorthand letter that can be used after a single dash.
+func (f *FlagSet) IPNetSliceVarP(p *[]net.IPNet, name, shorthand string, value []net.IPNet, usage string) {
+	f.VarP(newIPNetSliceValue(value, p), name, shorthand, usage)
+}
+
+// IPNetSliceVar defines a []net.IPNet flag with specified name, default value, and usage string.
+// The argument p points to a []net.IPNet variable in which to store the value of the flag.
+func IPNetSliceVar(p *[]net.IPNet, name string, value []net.IPNet, usage string) {
+	CommandLine.VarP(newIPNetSliceValue(value, p), name, "", usage)
+}
+
+// IPNetSliceVarP is like IPNetSliceVar, but accepts a shorthand letter that can be used after a single dash.
+func IPNetSliceVarP(p *[]net.IPNet, name, shorthand string, value []net.IPNet, usage string) {
+	CommandLine.VarP(newIPNetSliceValue(value, p), name, shorthand, usage)
+}
+
+// IPNetSlice defines a []net.IPNet flag with specified name, default value, and usage string.
+// The return value is the address of a []net.IPNet variable that stores the value of that flag.
+func (f *FlagSet) IPNetSlice(name string, value []net.IPNet, usage string) *[]net.IPNet {
+	p := []net.IPNet{}
+	f.IPNetSliceVarP(&p, name, "", value, usage)
+	return &p
+}
+
+// IPNetSliceP is like IPNetSlice, but accepts a shorthand letter that can be used after a single dash.
+func (f *FlagSet) IPNetSliceP(name, shorthand string, value []net.IPNet, usage string) *[]net.IPNet {
+	p := []net.IPNet{}
+	f.IPNetSliceVarP(&p, name, shorthand, value, usage)
+	return &p
+}
+
+// IPNetSlice defines a []net.IPNet flag with specified name, default value, and usage string.
+// The return value is the address of a []net.IP variable that stores the value of the flag.
+func IPNetSlice(name string, value []net.IPNet, usage string) *[]net.IPNet {
+	return CommandLine.IPNetSliceP(name, "", value, usage)
+}
+
+// IPNetSliceP is like IPNetSlice, but accepts a shorthand letter that can be used after a single dash.
+func IPNetSliceP(name, shorthand string, value []net.IPNet, usage string) *[]net.IPNet {
+	return CommandLine.IPNetSliceP(name, shorthand, value, usage)
+}
diff --git a/ipnet_slice_test.go b/ipnet_slice_test.go
new file mode 100644
index 0000000..11644c5
--- /dev/null
+++ b/ipnet_slice_test.go
@@ -0,0 +1,239 @@
+package pflag
+
+import (
+	"fmt"
+	"net"
+	"strings"
+	"testing"
+)
+
+// Helper function to set static slices
+func getCIDR(ip net.IP, cidr *net.IPNet, err error) net.IPNet {
+	return *cidr
+}
+
+func equalCIDR(c1 net.IPNet, c2 net.IPNet) bool {
+	if c1.String() == c2.String() {
+		return true
+	}
+	return false
+}
+
+func setUpIPNetFlagSet(ipsp *[]net.IPNet) *FlagSet {
+	f := NewFlagSet("test", ContinueOnError)
+	f.IPNetSliceVar(ipsp, "cidrs", []net.IPNet{}, "Command separated list!")
+	return f
+}
+
+func setUpIPNetFlagSetWithDefault(ipsp *[]net.IPNet) *FlagSet {
+	f := NewFlagSet("test", ContinueOnError)
+	f.IPNetSliceVar(ipsp, "cidrs",
+		[]net.IPNet{
+			getCIDR(net.ParseCIDR("192.168.1.1/16")),
+			getCIDR(net.ParseCIDR("fd00::/64")),
+		},
+		"Command separated list!")
+	return f
+}
+
+func TestEmptyIPNet(t *testing.T) {
+	var cidrs []net.IPNet
+	f := setUpIPNetFlagSet(&cidrs)
+	err := f.Parse([]string{})
+	if err != nil {
+		t.Fatal("expected no error; got", err)
+	}
+
+	getIPNet, err := f.GetIPNetSlice("cidrs")
+	if err != nil {
+		t.Fatal("got an error from GetIPNetSlice():", err)
+	}
+	if len(getIPNet) != 0 {
+		t.Fatalf("got ips %v with len=%d but expected length=0", getIPNet, len(getIPNet))
+	}
+}
+
+func TestIPNets(t *testing.T) {
+	var ips []net.IPNet
+	f := setUpIPNetFlagSet(&ips)
+
+	vals := []string{"192.168.1.1/24", "10.0.0.1/16", "fd00:0:0:0:0:0:0:2/64"}
+	arg := fmt.Sprintf("--cidrs=%s", strings.Join(vals, ","))
+	err := f.Parse([]string{arg})
+	if err != nil {
+		t.Fatal("expected no error; got", err)
+	}
+	for i, v := range ips {
+		if _, cidr, _ := net.ParseCIDR(vals[i]); cidr == nil {
+			t.Fatalf("invalid string being converted to CIDR: %s", vals[i])
+		} else if !equalCIDR(*cidr, v) {
+			t.Fatalf("expected ips[%d] to be %s but got: %s from GetIPSlice", i, vals[i], v)
+		}
+	}
+}
+
+func TestIPNetDefault(t *testing.T) {
+	var cidrs []net.IPNet
+	f := setUpIPNetFlagSetWithDefault(&cidrs)
+
+	vals := []string{"192.168.1.1/16", "fd00::/64"}
+	err := f.Parse([]string{})
+	if err != nil {
+		t.Fatal("expected no error; got", err)
+	}
+	for i, v := range cidrs {
+		if _, cidr, _ := net.ParseCIDR(vals[i]); cidr == nil {
+			t.Fatalf("invalid string being converted to CIDR: %s", vals[i])
+		} else if !equalCIDR(*cidr, v) {
+			t.Fatalf("expected cidrs[%d] to be %s but got: %s", i, vals[i], v)
+		}
+	}
+
+	getIPNet, err := f.GetIPNetSlice("cidrs")
+	if err != nil {
+		t.Fatal("got an error from GetIPNetSlice")
+	}
+	for i, v := range getIPNet {
+		if _, cidr, _ := net.ParseCIDR(vals[i]); cidr == nil {
+			t.Fatalf("invalid string being converted to CIDR: %s", vals[i])
+		} else if !equalCIDR(*cidr, v) {
+			t.Fatalf("expected cidrs[%d] to be %s but got: %s", i, vals[i], v)
+		}
+	}
+}
+
+func TestIPNetWithDefault(t *testing.T) {
+	var cidrs []net.IPNet
+	f := setUpIPNetFlagSetWithDefault(&cidrs)
+
+	vals := []string{"192.168.1.1/16", "fd00::/64"}
+	arg := fmt.Sprintf("--cidrs=%s", strings.Join(vals, ","))
+	err := f.Parse([]string{arg})
+	if err != nil {
+		t.Fatal("expected no error; got", err)
+	}
+	for i, v := range cidrs {
+		if _, cidr, _ := net.ParseCIDR(vals[i]); cidr == nil {
+			t.Fatalf("invalid string being converted to CIDR: %s", vals[i])
+		} else if !equalCIDR(*cidr, v) {
+			t.Fatalf("expected cidrs[%d] to be %s but got: %s", i, vals[i], v)
+		}
+	}
+
+	getIPNet, err := f.GetIPNetSlice("cidrs")
+	if err != nil {
+		t.Fatal("got an error from GetIPNetSlice")
+	}
+	for i, v := range getIPNet {
+		if _, cidr, _ := net.ParseCIDR(vals[i]); cidr == nil {
+			t.Fatalf("invalid string being converted to CIDR: %s", vals[i])
+		} else if !equalCIDR(*cidr, v) {
+			t.Fatalf("expected cidrs[%d] to be %s but got: %s", i, vals[i], v)
+		}
+	}
+}
+
+func TestIPNetCalledTwice(t *testing.T) {
+	var cidrs []net.IPNet
+	f := setUpIPNetFlagSet(&cidrs)
+
+	in := []string{"192.168.1.2/16,fd00::/64", "10.0.0.1/24"}
+
+	expected := []net.IPNet{
+		getCIDR(net.ParseCIDR("192.168.1.2/16")),
+		getCIDR(net.ParseCIDR("fd00::/64")),
+		getCIDR(net.ParseCIDR("10.0.0.1/24")),
+	}
+	argfmt := "--cidrs=%s"
+	arg1 := fmt.Sprintf(argfmt, in[0])
+	arg2 := fmt.Sprintf(argfmt, in[1])
+	err := f.Parse([]string{arg1, arg2})
+	if err != nil {
+		t.Fatal("expected no error; got", err)
+	}
+	for i, v := range cidrs {
+		if !equalCIDR(expected[i], v) {
+			t.Fatalf("expected cidrs[%d] to be %s but got: %s", i, expected[i], v)
+		}
+	}
+}
+
+func TestIPNetBadQuoting(t *testing.T) {
+
+	tests := []struct {
+		Want    []net.IPNet
+		FlagArg []string
+	}{
+		{
+			Want: []net.IPNet{
+				getCIDR(net.ParseCIDR("a4ab:61d:f03e:5d7d:fad7:d4c2:a1a5:568/128")),
+				getCIDR(net.ParseCIDR("203.107.49.208/32")),
+				getCIDR(net.ParseCIDR("14.57.204.90/32")),
+			},
+			FlagArg: []string{
+				"a4ab:61d:f03e:5d7d:fad7:d4c2:a1a5:568/128",
+				"203.107.49.208/32",
+				"14.57.204.90/32",
+			},
+		},
+		{
+			Want: []net.IPNet{
+				getCIDR(net.ParseCIDR("204.228.73.195/32")),
+				getCIDR(net.ParseCIDR("86.141.15.94/32")),
+			},
+			FlagArg: []string{
+				"204.228.73.195/32",
+				"86.141.15.94/32",
+			},
+		},
+		{
+			Want: []net.IPNet{
+				getCIDR(net.ParseCIDR("c70c:db36:3001:890f:c6ea:3f9b:7a39:cc3f/128")),
+				getCIDR(net.ParseCIDR("4d17:1d6e:e699:bd7a:88c5:5e7e:ac6a:4472/128")),
+			},
+			FlagArg: []string{
+				"c70c:db36:3001:890f:c6ea:3f9b:7a39:cc3f/128",
+				"4d17:1d6e:e699:bd7a:88c5:5e7e:ac6a:4472/128",
+			},
+		},
+		{
+			Want: []net.IPNet{
+				getCIDR(net.ParseCIDR("5170:f971:cfac:7be3:512a:af37:952c:bc33/128")),
+				getCIDR(net.ParseCIDR("93.21.145.140/32")),
+				getCIDR(net.ParseCIDR("2cac:61d3:c5ff:6caf:73e0:1b1a:c336:c1ca/128")),
+			},
+			FlagArg: []string{
+				" 5170:f971:cfac:7be3:512a:af37:952c:bc33/128  , 93.21.145.140/32     ",
+				"2cac:61d3:c5ff:6caf:73e0:1b1a:c336:c1ca/128",
+			},
+		},
+		{
+			Want: []net.IPNet{
+				getCIDR(net.ParseCIDR("2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128")),
+				getCIDR(net.ParseCIDR("2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128")),
+				getCIDR(net.ParseCIDR("2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128")),
+				getCIDR(net.ParseCIDR("2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128")),
+			},
+			FlagArg: []string{
+				`"2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128,        2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128,2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128     "`,
+				" 2e5e:66b2:6441:848:5b74:76ea:574c:3a7b/128"},
+		},
+	}
+
+	for i, test := range tests {
+
+		var cidrs []net.IPNet
+		f := setUpIPNetFlagSet(&cidrs)
+
+		if err := f.Parse([]string{fmt.Sprintf("--cidrs=%s", strings.Join(test.FlagArg, ","))}); err != nil {
+			t.Fatalf("flag parsing failed with error: %s\nparsing:\t%#v\nwant:\t\t%s",
+				err, test.FlagArg, test.Want[i])
+		}
+
+		for j, b := range cidrs {
+			if !equalCIDR(b, test.Want[j]) {
+				t.Fatalf("bad value parsed for test %d on net.IP %d:\nwant:\t%s\ngot:\t%s", i, j, test.Want[j], b)
+			}
+		}
+	}
+}