66 lines
1.5 KiB
Go
Raw Normal View History

2023-05-19 16:23:55 -04:00
package multicast
2019-06-09 09:24:20 +02:00
import (
"net"
)
2019-06-09 09:24:20 +02:00
2023-05-19 16:23:55 -04:00
type InterfacesProviderFunc func() []net.Interface
2019-06-09 09:24:20 +02:00
2023-05-19 16:23:55 -04:00
// InterfacesProvider specify a function to list all interfaces to multicast.
// If no provider are given, all possible interfaces will be used.
var InterfacesProvider InterfacesProviderFunc
// interfaces gets list of net.Interface to multicast UDP packet.
func interfaces() ([]net.Interface, error) {
2023-05-19 16:23:55 -04:00
if p := InterfacesProvider; p != nil {
if list := p(); len(list) > 0 {
return list, nil
}
2019-06-09 09:24:20 +02:00
}
2023-05-19 16:23:55 -04:00
return interfacesIPv4()
2019-06-09 09:24:20 +02:00
}
// interfacesIPv4 lists net.Interface on IPv4.
func interfacesIPv4() ([]net.Interface, error) {
2019-06-09 09:24:20 +02:00
iflist, err := net.Interfaces()
if err != nil {
return nil, err
2019-06-09 09:24:20 +02:00
}
list := make([]net.Interface, 0, len(iflist))
for _, ifi := range iflist {
2023-05-19 16:23:55 -04:00
if !hasLinkUp(&ifi) || !hasMulticast(&ifi) || !hasIPv4Address(&ifi) {
2019-06-09 09:24:20 +02:00
continue
}
list = append(list, ifi)
}
return list, nil
2019-06-09 09:24:20 +02:00
}
// hasLinkUp checks an I/F have link-up or not.
func hasLinkUp(ifi *net.Interface) bool {
return ifi.Flags&net.FlagUp != 0
}
2023-05-19 16:23:55 -04:00
// hasMulticast checks an I/F supports multicast or not.
func hasMulticast(ifi *net.Interface) bool {
return ifi.Flags&net.FlagMulticast != 0
}
2019-06-09 09:24:20 +02:00
// hasIPv4Address checks an I/F have IPv4 address.
func hasIPv4Address(ifi *net.Interface) bool {
addrs, err := ifi.Addrs()
if err != nil {
return false
}
for _, a := range addrs {
ip, _, err := net.ParseCIDR(a.String())
if err != nil {
continue
}
if len(ip.To4()) == net.IPv4len && !ip.IsUnspecified() {
return true
}
}
return false
}