2022-10-20 13:00:50 +02:00
|
|
|
// addr provides functions to retrieve local IP addresses from device interfaces.
|
2019-05-31 00:52:10 +02:00
|
|
|
package addr
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net"
|
2022-10-20 13:00:50 +02:00
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2019-05-31 00:52:10 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2022-10-20 13:00:50 +02:00
|
|
|
// ErrIPNotFound no IP address found, and explicit IP not provided.
|
|
|
|
ErrIPNotFound = errors.New("no IP address found, and explicit IP not provided")
|
2019-05-31 00:52:10 +02:00
|
|
|
)
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
// IsLocal checks whether an IP belongs to one of the device's interfaces.
|
2020-01-14 15:23:16 +02:00
|
|
|
func IsLocal(addr string) bool {
|
2022-10-20 13:00:50 +02:00
|
|
|
// Extract the host
|
2020-01-14 15:23:16 +02:00
|
|
|
host, _, err := net.SplitHostPort(addr)
|
|
|
|
if err == nil {
|
|
|
|
addr = host
|
|
|
|
}
|
|
|
|
|
|
|
|
if addr == "localhost" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
// Check against all local ips
|
2020-01-14 15:23:16 +02:00
|
|
|
for _, ip := range IPs() {
|
|
|
|
if addr == ip {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
// Extract returns a valid IP address. If the address provided is a valid
|
2023-11-26 12:06:55 +02:00
|
|
|
// address, it will be returned directly. Otherwise, the available interfaces
|
|
|
|
// will be iterated over to find an IP address, preferably private.
|
2019-05-31 00:52:10 +02:00
|
|
|
func Extract(addr string) (string, error) {
|
2022-10-20 13:00:50 +02:00
|
|
|
// if addr is already specified then it's directly returned
|
2019-07-17 09:38:50 +02:00
|
|
|
if len(addr) > 0 && (addr != "0.0.0.0" && addr != "[::]" && addr != "::") {
|
2019-05-31 00:52:10 +02:00
|
|
|
return addr, nil
|
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
var (
|
|
|
|
addrs []net.Addr
|
|
|
|
loAddrs []net.Addr
|
|
|
|
)
|
|
|
|
|
2019-05-31 00:52:10 +02:00
|
|
|
ifaces, err := net.Interfaces()
|
|
|
|
if err != nil {
|
2022-10-20 13:00:50 +02:00
|
|
|
return "", errors.Wrap(err, "failed to get interfaces")
|
2019-05-31 00:52:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, iface := range ifaces {
|
|
|
|
ifaceAddrs, err := iface.Addrs()
|
|
|
|
if err != nil {
|
2020-05-29 18:49:22 +02:00
|
|
|
// ignore error, interface can disappear from system
|
2019-05-31 00:52:10 +02:00
|
|
|
continue
|
|
|
|
}
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2019-08-26 08:37:49 +02:00
|
|
|
if iface.Flags&net.FlagLoopback != 0 {
|
2019-08-27 17:33:30 +02:00
|
|
|
loAddrs = append(loAddrs, ifaceAddrs...)
|
2019-08-26 08:37:49 +02:00
|
|
|
continue
|
|
|
|
}
|
2019-05-31 00:52:10 +02:00
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
addrs = append(addrs, ifaceAddrs...)
|
2019-05-31 00:52:10 +02:00
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
// Add loopback addresses to the end of the list
|
|
|
|
addrs = append(addrs, loAddrs...)
|
2019-05-31 00:52:10 +02:00
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
// Try to find private IP in list, public IP otherwise
|
|
|
|
ip, err := findIP(addrs)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
2019-05-31 00:52:10 +02:00
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
return ip.String(), nil
|
2019-05-31 00:52:10 +02:00
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
// IPs returns all available interface IP addresses.
|
2019-05-31 00:52:10 +02:00
|
|
|
func IPs() []string {
|
|
|
|
ifaces, err := net.Interfaces()
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var ipAddrs []string
|
|
|
|
|
|
|
|
for _, i := range ifaces {
|
|
|
|
addrs, err := i.Addrs()
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, addr := range addrs {
|
|
|
|
var ip net.IP
|
|
|
|
switch v := addr.(type) {
|
|
|
|
case *net.IPNet:
|
|
|
|
ip = v.IP
|
|
|
|
case *net.IPAddr:
|
|
|
|
ip = v.IP
|
|
|
|
}
|
|
|
|
|
|
|
|
if ip == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
ipAddrs = append(ipAddrs, ip.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ipAddrs
|
|
|
|
}
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2023-11-26 12:06:55 +02:00
|
|
|
// findIP will return the first private IP available in the list.
|
|
|
|
// If no private IP is available it will return the first public IP, if present.
|
|
|
|
// If no public IP is available, it will return the first loopback IP, if present.
|
2022-10-20 13:00:50 +02:00
|
|
|
func findIP(addresses []net.Addr) (net.IP, error) {
|
|
|
|
var publicIP net.IP
|
2023-11-26 12:06:55 +02:00
|
|
|
var localIP net.IP
|
2022-10-20 13:00:50 +02:00
|
|
|
|
|
|
|
for _, rawAddr := range addresses {
|
|
|
|
var ip net.IP
|
|
|
|
switch addr := rawAddr.(type) {
|
|
|
|
case *net.IPAddr:
|
|
|
|
ip = addr.IP
|
|
|
|
case *net.IPNet:
|
|
|
|
ip = addr.IP
|
|
|
|
default:
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-11-26 12:06:55 +02:00
|
|
|
if ip.IsLoopback() {
|
|
|
|
if localIP == nil {
|
|
|
|
localIP = ip
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
if !ip.IsPrivate() {
|
2023-11-26 12:06:55 +02:00
|
|
|
if publicIP == nil {
|
|
|
|
publicIP = ip
|
|
|
|
}
|
2022-10-20 13:00:50 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return private IP if available
|
|
|
|
return ip, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return public or virtual IP
|
|
|
|
if len(publicIP) > 0 {
|
|
|
|
return publicIP, nil
|
|
|
|
}
|
|
|
|
|
2023-11-26 12:06:55 +02:00
|
|
|
// Return local IP
|
|
|
|
if len(localIP) > 0 {
|
|
|
|
return localIP, nil
|
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
return nil, ErrIPNotFound
|
|
|
|
}
|