Skip to content

Commit 0402086

Browse files
committed
Attractor ipv4/ipv6-prefix also accept IP range
Example: ipv4-prefix: 169.254.100.0-169.254.100.100/24 ipv6-prefix: 100:100::0-100:100::100/64;100:100::a:0-100:100::a:aaa9/64
1 parent 8ce909e commit 0402086

File tree

4 files changed

+61
-8
lines changed

4 files changed

+61
-8
lines changed

api/v1alpha1/attractor_types.go

+8-2
Original file line numberDiff line numberDiff line change
@@ -48,10 +48,16 @@ type InterfaceSpec struct {
4848
// name of the interface
4949
Name string `json:"name"`
5050

51-
// (immutable) ipv4 prefix of the interface, which is used for frontend to set up communication with the ipv4 gateways
51+
// (immutable) ipv4 prefix or range of the interface, which is used for frontend to set up communication with the ipv4 gateways
52+
// For example, '192.168.100.0/24', '192.168.100.1-192.168.100.100/24'.
53+
// Multiple IP ranges can be combined using semicolon delimiter, but they must belong to the same network.
54+
// For example, '192.168.100.3-192.168.100.100/24;192.168.100.200-192.168.100.250/24'.
5255
PrefixIPv4 string `json:"ipv4-prefix"`
5356

54-
// (immutable) ipv6 prefix of the interface, which is used for frontend to set up communication with the ipv6 gateways
57+
// (immutable) ipv6 prefix or range of the interface, which is used for frontend to set up communication with the ipv6 gateways
58+
// For example, '100:100::/64', '100:100::bbbb-100:100::cccc/64'.
59+
// Multiple IP ranges can be combined using semicolon delimiter, but they must belong to the same network.
60+
// For example, '100:100::2-100:100::ffff/64;100:100::a:2-100:100::a:f/64;100:100::e:2-100:100::e:f/64'.
5561
PrefixIPv6 string `json:"ipv6-prefix"`
5662

5763
// interface choice.

api/v1alpha1/attractor_webhook.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -92,12 +92,12 @@ func (r *Attractor) validateAttractor() error {
9292
if err := r.validateLabels(); err != nil {
9393
allErrs = append(allErrs, field.Invalid(field.NewPath("metadata").Child("labels"), r.ObjectMeta.Labels, err.Error()))
9494
}
95-
_, err := validatePrefix(r.Spec.Interface.PrefixIPv4)
95+
err := validatePrefixAndRange(r.Spec.Interface.PrefixIPv4)
9696
if err != nil {
9797
allErrs = append(allErrs, field.Invalid(field.NewPath("spec").Child("ipv4-prefix"), r.Spec.Interface.PrefixIPv4, err.Error()))
9898
}
9999

100-
_, err = validatePrefix(r.Spec.Interface.PrefixIPv6)
100+
err = validatePrefixAndRange(r.Spec.Interface.PrefixIPv6)
101101
if err != nil {
102102
allErrs = append(allErrs, field.Invalid(field.NewPath("spec").Child("ipv6-prefix"), r.Spec.Interface.PrefixIPv6, err.Error()))
103103
}

api/v1alpha1/types.go

+41
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,47 @@ func validatePrefix(p string) (*net.IPNet, error) {
9898
return n, nil
9999
}
100100

101+
func validatePrefixAndRange(p string) error {
102+
var err error
103+
var network *net.IPNet
104+
composite := strings.Split(p, ";")
105+
for _, c := range composite {
106+
if r := strings.SplitN(c, "-", 2); len(r) == 2 {
107+
// validate IP range
108+
firstip := net.ParseIP(r[0])
109+
if firstip == nil {
110+
return fmt.Errorf("'%s' is not a valid IP range start", r[0])
111+
}
112+
_, ipNet, err := net.ParseCIDR(r[1])
113+
if err != nil {
114+
return fmt.Errorf("'%s' is not a valid CIDR: %s", r[1], err)
115+
}
116+
if !ipNet.Contains(firstip) {
117+
return fmt.Errorf("'%s' is not a valid IP range start for CIDR %s", ipNet.String(), firstip)
118+
}
119+
if network == nil {
120+
network = ipNet
121+
} else {
122+
if !network.IP.Equal(ipNet.IP) {
123+
return fmt.Errorf("network mismatch: '%s' != '%s'", network, ipNet)
124+
}
125+
netOnes, netBits := network.Mask.Size()
126+
ones, bits := ipNet.Mask.Size()
127+
if netOnes != ones || netBits != bits {
128+
return fmt.Errorf("network mask mismatch: %v != %v", network.Mask, ipNet.Mask)
129+
}
130+
}
131+
} else {
132+
// validate prefix
133+
if len(composite) > 1 {
134+
return fmt.Errorf("%s composite subnet config is invalid", composite)
135+
}
136+
_, err = validatePrefix(c)
137+
}
138+
}
139+
return err
140+
}
141+
101142
type InterfaceType string
102143

103144
const (

config/crd/bases/meridio.nordix.org_attractors.yaml

+10-4
Original file line numberDiff line numberDiff line change
@@ -74,12 +74,18 @@ spec:
7474
description: defines the interface information that attractor use
7575
properties:
7676
ipv4-prefix:
77-
description: (immutable) ipv4 prefix of the interface, which is
78-
used for frontend to set up communication with the ipv4 gateways
77+
description: (immutable) ipv4 prefix or range of the interface,
78+
which is used for frontend to set up communication with the
79+
ipv4 gateways For example, '192.168.100.0/24', '192.168.100.1-192.168.100.100/24'.
80+
Multiple IP ranges can be combined using semicolon delimiter,
81+
but they must belong to the same network. For example, '192.168.100.3-192.168.100.100/24;192.168.100.200-192.168.100.250/24'.
7982
type: string
8083
ipv6-prefix:
81-
description: (immutable) ipv6 prefix of the interface, which is
82-
used for frontend to set up communication with the ipv6 gateways
84+
description: (immutable) ipv6 prefix or range of the interface,
85+
which is used for frontend to set up communication with the
86+
ipv6 gateways For example, '100:100::/64', '100:100::bbbb-100:100::cccc/64'.
87+
Multiple IP ranges can be combined using semicolon delimiter,
88+
but they must belong to the same network. For example, '100:100::2-100:100::ffff/64;100:100::a:2-100:100::a:f/64;100:100::e:2-100:100::e:f/64'.
8389
type: string
8490
name:
8591
description: name of the interface

0 commit comments

Comments
 (0)