micro/util/addr/addr.go

177 lines
3.2 KiB
Go
Raw Permalink Normal View History

package addr // import "go.unistack.org/micro/v4/util/addr"
2019-05-30 23:52:10 +01:00
import (
"fmt"
"net"
)
var privateBlocks []*net.IPNet
2019-05-30 23:52:10 +01:00
func init() {
for _, b := range []string{"10.0.0.0/8", "172.16.0.0/12", "192.168.0.0/16", "100.64.0.0/10", "fd00::/8"} {
if _, block, err := net.ParseCIDR(b); err == nil {
privateBlocks = append(privateBlocks, block)
}
}
}
// AppendPrivateBlocks append private network blocks
func AppendPrivateBlocks(bs ...string) {
for _, b := range bs {
if _, block, err := net.ParseCIDR(b); err == nil {
privateBlocks = append(privateBlocks, block)
}
}
}
2019-05-30 23:52:10 +01:00
func isPrivateIP(ipAddr string) bool {
ip := net.ParseIP(ipAddr)
for _, priv := range privateBlocks {
if priv.Contains(ip) {
return true
}
}
return false
}
// IsLocal tells us whether an ip is local
func IsLocal(addr string) bool {
// extract the host
host, _, err := net.SplitHostPort(addr)
if err == nil {
addr = host
}
// check if its localhost
if addr == "localhost" {
return true
}
// check against all local ips
for _, ip := range IPs() {
if addr == ip {
return true
}
}
return false
}
2019-05-30 23:52:10 +01:00
// Extract returns a real ip
//
//nolint:gocyclo
2019-05-30 23:52:10 +01:00
func Extract(addr string) (string, error) {
// if addr specified then its returned
if len(addr) > 0 && (addr != "0.0.0.0" && addr != "[::]" && addr != "::") {
2019-05-30 23:52:10 +01:00
return addr, nil
}
ifaces, err := net.Interfaces()
if err != nil {
return "", fmt.Errorf("Failed to get interfaces! Err: %v", err)
}
//nolint:prealloc
2019-05-30 23:52:10 +01:00
var addrs []net.Addr
var loAddrs []net.Addr
2019-05-30 23:52:10 +01:00
for _, iface := range ifaces {
ifaceAddrs, err := iface.Addrs()
if err != nil {
2020-05-30 00:49:22 +08:00
// ignore error, interface can disappear from system
2019-05-30 23:52:10 +01:00
continue
}
2019-08-26 14:37:49 +08:00
if iface.Flags&net.FlagLoopback != 0 {
loAddrs = append(loAddrs, ifaceAddrs...)
2019-08-26 14:37:49 +08:00
continue
}
2019-05-30 23:52:10 +01:00
addrs = append(addrs, ifaceAddrs...)
}
addrs = append(addrs, loAddrs...)
2019-05-30 23:52:10 +01:00
var ipAddr string
var publicIP string
2019-05-30 23:52:10 +01:00
for _, rawAddr := range addrs {
var ip net.IP
switch addr := rawAddr.(type) {
case *net.IPAddr:
ip = addr.IP
case *net.IPNet:
ip = addr.IP
default:
continue
}
if !isPrivateIP(ip.String()) {
publicIP = ip.String()
2019-05-30 23:52:10 +01:00
continue
}
ipAddr = ip.String()
2019-05-30 23:52:10 +01:00
break
}
// return private ip
if len(ipAddr) > 0 {
a := net.ParseIP(ipAddr)
if a == nil {
return "", fmt.Errorf("ip addr %s is invalid", ipAddr)
}
return a.String(), nil
2019-05-30 23:52:10 +01:00
}
// return public or virtual ip
if len(publicIP) > 0 {
a := net.ParseIP(publicIP)
if a == nil {
return "", fmt.Errorf("ip addr %s is invalid", publicIP)
}
return a.String(), nil
2019-05-30 23:52:10 +01:00
}
return "", fmt.Errorf("No IP address found, and explicit IP not provided")
}
// IPs returns all known ips
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
}
// dont skip ipv6 addrs
/*
ip = ip.To4()
if ip == nil {
continue
}
*/
2019-05-30 23:52:10 +01:00
ipAddrs = append(ipAddrs, ip.String())
}
}
return ipAddrs
}