-
Notifications
You must be signed in to change notification settings - Fork 35
/
Copy pathnodeid.go
111 lines (97 loc) · 2.57 KB
/
nodeid.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
// SPDX-License-Identifier: Apache-2.0
// Copyright 2024 Canonical Ltd.
package context
import (
"fmt"
"net"
"time"
"github.com/omec-project/smf/logger"
)
const (
NodeIdTypeIpv4Address uint8 = iota
NodeIdTypeIpv6Address
NodeIdTypeFqdn
)
type NodeID struct {
NodeIdValue []byte
NodeIdType uint8 // 0x00001111
}
var dnsHostIpCache map[string]net.IP
func NewNodeID(nodeID string) *NodeID {
ip := net.ParseIP(nodeID)
if ip == nil {
return &NodeID{
NodeIdType: NodeIdTypeFqdn,
NodeIdValue: []byte(nodeID),
}
} else if ip.To4() != nil {
return &NodeID{
NodeIdType: NodeIdTypeIpv4Address,
NodeIdValue: ip.To4(),
}
} else {
return &NodeID{
NodeIdType: NodeIdTypeIpv6Address,
NodeIdValue: ip.To16(),
}
}
}
func (n *NodeID) ResolveNodeIdToIp() net.IP {
switch n.NodeIdType {
case NodeIdTypeIpv4Address, NodeIdTypeIpv6Address:
return n.NodeIdValue
case NodeIdTypeFqdn:
if ip, err := getDnsHostIp(string(n.NodeIdValue)); err != nil {
logger.CtxLog.Warnf("host [%v] not found in smf dns cache ", string(n.NodeIdValue))
if ns, err := net.LookupHost(string(n.NodeIdValue)); err != nil {
logger.CtxLog.Warnf("host lookup failed: %+v", err)
return net.IPv4zero
} else {
logger.CtxLog.Infof("host [%v] dns resolved, updating smf dns cache ", string(n.NodeIdValue))
InsertDnsHostIp(string(n.NodeIdValue), net.ParseIP(ns[0]))
return net.ParseIP(ns[0])
}
} else {
logger.CtxLog.Debugf("host [%v] found in smf dns cache ", string(n.NodeIdValue))
return ip
}
default:
return net.IPv4zero
}
}
func init() {
dnsHostIpCache = make(map[string]net.IP)
ticker := time.NewTicker(time.Minute)
go func() {
for {
<-ticker.C
RefreshDnsHostIpCache()
}
}()
}
func RefreshDnsHostIpCache() {
for hostName := range dnsHostIpCache {
logger.CtxLog.Debugf("refreshing DNS for host [%v] ", hostName)
if ns, err := net.LookupHost(hostName); err != nil {
logger.CtxLog.Warnf("host lookup failed: %+v", err)
deleteDnsHost(hostName)
continue
} else if !dnsHostIpCache[hostName].Equal(net.ParseIP(ns[0])) {
logger.CtxLog.Infof("smf dns cache updated for host [%v]: [%v] ", hostName, net.ParseIP(ns[0]).String())
dnsHostIpCache[hostName] = net.ParseIP(ns[0])
}
}
}
func getDnsHostIp(hostName string) (net.IP, error) {
if ip, ok := dnsHostIpCache[hostName]; !ok {
return nil, fmt.Errorf("host [%v] not found in smf dns cache", hostName)
} else {
return ip, nil
}
}
func InsertDnsHostIp(hostName string, ip net.IP) {
dnsHostIpCache[hostName] = ip
}
func deleteDnsHost(hostName string) {
delete(dnsHostIpCache, hostName)
}