mirror of
https://github.com/openshift/installer.git
synced 2026-02-05 06:46:36 +01:00
117 lines
2.5 KiB
Go
117 lines
2.5 KiB
Go
// Package ipnet wraps net.IPNet to get CIDR serialization.
|
|
package ipnet
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net"
|
|
"reflect"
|
|
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var nullString = "null"
|
|
var nullBytes = []byte(nullString)
|
|
var emptyIPNet = net.IPNet{}
|
|
|
|
// IPNet wraps net.IPNet to get CIDR serialization.
|
|
// +kubebuilder:validation:Type=string
|
|
type IPNet struct {
|
|
net.IPNet
|
|
}
|
|
|
|
// String returns a CIDR serialization of the subnet, or an empty
|
|
// string if the subnet is nil.
|
|
func (ipnet *IPNet) String() string {
|
|
if ipnet == nil {
|
|
return ""
|
|
}
|
|
return ipnet.IPNet.String()
|
|
}
|
|
|
|
// MarshalJSON interface for an IPNet
|
|
func (ipnet IPNet) MarshalJSON() (data []byte, err error) {
|
|
if reflect.DeepEqual(ipnet.IPNet, emptyIPNet) {
|
|
return nullBytes, nil
|
|
}
|
|
|
|
return json.Marshal(ipnet.String())
|
|
}
|
|
|
|
// UnmarshalJSON interface for an IPNet
|
|
func (ipnet *IPNet) UnmarshalJSON(b []byte) (err error) {
|
|
if string(b) == nullString {
|
|
ipnet.IP = net.IP{}
|
|
ipnet.Mask = net.IPMask{}
|
|
return nil
|
|
}
|
|
|
|
var cidr string
|
|
err = json.Unmarshal(b, &cidr)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to Unmarshal string")
|
|
}
|
|
|
|
parsedIPNet, err := ParseCIDR(cidr)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to Parse cidr string to net.IPNet")
|
|
}
|
|
|
|
*ipnet = *parsedIPNet
|
|
|
|
return nil
|
|
}
|
|
|
|
// ParseCIDR parses a CIDR from its string representation.
|
|
func ParseCIDR(s string) (*IPNet, error) {
|
|
ip, cidr, err := net.ParseCIDR(s)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// This check is needed in order to work around a strange quirk in the Go
|
|
// standard library. All of the addresses returned by net.ParseCIDR() are
|
|
// 16-byte addresses. This does _not_ imply that they are IPv6 addresses,
|
|
// which is what some libraries (e.g. github.com/apparentlymart/go-cidr)
|
|
// assume. By forcing the address to be the expected length, we can work
|
|
// around these bugs.
|
|
if ip.To4() != nil {
|
|
ip = ip.To4()
|
|
}
|
|
|
|
return &IPNet{
|
|
IPNet: net.IPNet{
|
|
IP: ip,
|
|
Mask: cidr.Mask,
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// MustParseCIDR parses a CIDR from its string representation. If the parse fails,
|
|
// the function will panic.
|
|
func MustParseCIDR(s string) *IPNet {
|
|
cidr, err := ParseCIDR(s)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return cidr
|
|
}
|
|
|
|
// DeepCopyInto copies the receiver into out. out must be non-nil.
|
|
func (ipnet *IPNet) DeepCopyInto(out *IPNet) {
|
|
if ipnet == nil {
|
|
*out = IPNet{}
|
|
} else {
|
|
*out = *ipnet
|
|
}
|
|
}
|
|
|
|
// DeepCopy copies the receiver, creating a new IPNet.
|
|
func (ipnet *IPNet) DeepCopy() *IPNet {
|
|
if ipnet == nil {
|
|
return nil
|
|
}
|
|
out := new(IPNet)
|
|
ipnet.DeepCopyInto(out)
|
|
return out
|
|
}
|