| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | package nebula | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"net" | 
					
						
							|  |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2021-03-05 21:18:33 -05:00
										 |  |  | 	"sync/atomic" | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/rcrowley/go-metrics" | 
					
						
							|  |  |  | 	"github.com/sirupsen/logrus" | 
					
						
							|  |  |  | 	"github.com/slackhq/nebula/cert" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //const ProbeLen = 100 | 
					
						
							|  |  |  | const PromoteEvery = 1000 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | const ReQueryEvery = 5000 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | const MaxRemotes = 10 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // How long we should prevent roaming back to the previous IP. | 
					
						
							|  |  |  | // This helps prevent flapping due to packets already in flight | 
					
						
							| 
									
										
										
										
											2021-03-01 17:14:34 +01:00
										 |  |  | const RoamingSuppressSeconds = 2 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | type HostMap struct { | 
					
						
							|  |  |  | 	sync.RWMutex    //Because we concurrently read and write to our maps | 
					
						
							|  |  |  | 	name            string | 
					
						
							|  |  |  | 	Indexes         map[uint32]*HostInfo | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 	RemoteIndexes   map[uint32]*HostInfo | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	Hosts           map[uint32]*HostInfo | 
					
						
							|  |  |  | 	preferredRanges []*net.IPNet | 
					
						
							|  |  |  | 	vpnCIDR         *net.IPNet | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | 	unsafeRoutes    *CIDRTree | 
					
						
							| 
									
										
										
										
											2020-06-26 13:45:48 -04:00
										 |  |  | 	metricsEnabled  bool | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	l               *logrus.Logger | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type HostInfo struct { | 
					
						
							| 
									
										
										
										
											2021-03-05 21:18:33 -05:00
										 |  |  | 	sync.RWMutex | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	remote            *udpAddr | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	remotes           *RemoteList | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	promoteCounter    uint32 | 
					
						
							|  |  |  | 	ConnectionState   *ConnectionState | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	handshakeStart    time.Time        //todo: this an entry in the handshake manager | 
					
						
							|  |  |  | 	HandshakeReady    bool             //todo: being in the manager means you are ready | 
					
						
							|  |  |  | 	HandshakeCounter  int              //todo: another handshake manager entry | 
					
						
							|  |  |  | 	HandshakeComplete bool             //todo: this should go away in favor of ConnectionState.ready | 
					
						
							|  |  |  | 	HandshakePacket   map[uint8][]byte //todo: this is other handshake manager entry | 
					
						
							|  |  |  | 	packetStore       []*cachedPacket  //todo: this is other handshake manager entry | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	remoteIndexId     uint32 | 
					
						
							|  |  |  | 	localIndexId      uint32 | 
					
						
							|  |  |  | 	hostId            uint32 | 
					
						
							|  |  |  | 	recvError         int | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | 	remoteCidr        *CIDRTree | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-01 19:06:01 -06:00
										 |  |  | 	// lastRebindCount is the other side of Interface.rebindCount, if these values don't match then we need to ask LH | 
					
						
							|  |  |  | 	// for a punch from the remote end of this tunnel. The goal being to prime their conntrack for our traffic just like | 
					
						
							|  |  |  | 	// with a handshake | 
					
						
							|  |  |  | 	lastRebindCount int8 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-27 21:15:34 -05:00
										 |  |  | 	// lastHandshakeTime records the time the remote side told us about at the stage when the handshake was completed locally | 
					
						
							|  |  |  | 	// Stage 1 packet will contain it if I am a responder, stage 2 packet if I am an initiator | 
					
						
							|  |  |  | 	// This is used to avoid an attack where a handshake packet is replayed after some time | 
					
						
							|  |  |  | 	lastHandshakeTime uint64 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	lastRoam       time.Time | 
					
						
							|  |  |  | 	lastRoamRemote *udpAddr | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type cachedPacket struct { | 
					
						
							|  |  |  | 	messageType    NebulaMessageType | 
					
						
							|  |  |  | 	messageSubType NebulaMessageSubType | 
					
						
							|  |  |  | 	callback       packetCallback | 
					
						
							|  |  |  | 	packet         []byte | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type packetCallback func(t NebulaMessageType, st NebulaMessageSubType, h *HostInfo, p, nb, out []byte) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-27 22:23:18 -04:00
										 |  |  | type cachedPacketMetrics struct { | 
					
						
							|  |  |  | 	sent    metrics.Counter | 
					
						
							|  |  |  | 	dropped metrics.Counter | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | func NewHostMap(l *logrus.Logger, name string, vpnCIDR *net.IPNet, preferredRanges []*net.IPNet) *HostMap { | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	h := map[uint32]*HostInfo{} | 
					
						
							|  |  |  | 	i := map[uint32]*HostInfo{} | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 	r := map[uint32]*HostInfo{} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	m := HostMap{ | 
					
						
							|  |  |  | 		name:            name, | 
					
						
							|  |  |  | 		Indexes:         i, | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 		RemoteIndexes:   r, | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 		Hosts:           h, | 
					
						
							|  |  |  | 		preferredRanges: preferredRanges, | 
					
						
							|  |  |  | 		vpnCIDR:         vpnCIDR, | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | 		unsafeRoutes:    NewCIDRTree(), | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 		l:               l, | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return &m | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // UpdateStats takes a name and reports host and index counts to the stats collection system | 
					
						
							|  |  |  | func (hm *HostMap) EmitStats(name string) { | 
					
						
							|  |  |  | 	hm.RLock() | 
					
						
							|  |  |  | 	hostLen := len(hm.Hosts) | 
					
						
							|  |  |  | 	indexLen := len(hm.Indexes) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 	remoteIndexLen := len(hm.RemoteIndexes) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	hm.RUnlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	metrics.GetOrRegisterGauge("hostmap."+name+".hosts", nil).Update(int64(hostLen)) | 
					
						
							|  |  |  | 	metrics.GetOrRegisterGauge("hostmap."+name+".indexes", nil).Update(int64(indexLen)) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 	metrics.GetOrRegisterGauge("hostmap."+name+".remoteIndexes", nil).Update(int64(remoteIndexLen)) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) GetIndexByVpnIP(vpnIP uint32) (uint32, error) { | 
					
						
							|  |  |  | 	hm.RLock() | 
					
						
							|  |  |  | 	if i, ok := hm.Hosts[vpnIP]; ok { | 
					
						
							|  |  |  | 		index := i.localIndexId | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		return index, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	hm.RUnlock() | 
					
						
							|  |  |  | 	return 0, errors.New("vpn IP not found") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) Add(ip uint32, hostinfo *HostInfo) { | 
					
						
							|  |  |  | 	hm.Lock() | 
					
						
							|  |  |  | 	hm.Hosts[ip] = hostinfo | 
					
						
							|  |  |  | 	hm.Unlock() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) AddVpnIP(vpnIP uint32) *HostInfo { | 
					
						
							|  |  |  | 	h := &HostInfo{} | 
					
						
							|  |  |  | 	hm.RLock() | 
					
						
							|  |  |  | 	if _, ok := hm.Hosts[vpnIP]; !ok { | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		h = &HostInfo{ | 
					
						
							|  |  |  | 			promoteCounter:  0, | 
					
						
							|  |  |  | 			hostId:          vpnIP, | 
					
						
							|  |  |  | 			HandshakePacket: make(map[uint8][]byte, 0), | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		hm.Lock() | 
					
						
							|  |  |  | 		hm.Hosts[vpnIP] = h | 
					
						
							|  |  |  | 		hm.Unlock() | 
					
						
							|  |  |  | 		return h | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		h = hm.Hosts[vpnIP] | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		return h | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) DeleteVpnIP(vpnIP uint32) { | 
					
						
							|  |  |  | 	hm.Lock() | 
					
						
							|  |  |  | 	delete(hm.Hosts, vpnIP) | 
					
						
							|  |  |  | 	if len(hm.Hosts) == 0 { | 
					
						
							|  |  |  | 		hm.Hosts = map[uint32]*HostInfo{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	hm.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	if hm.l.Level >= logrus.DebugLevel { | 
					
						
							|  |  |  | 		hm.l.WithField("hostMap", m{"mapName": hm.name, "vpnIp": IntIp(vpnIP), "mapTotalSize": len(hm.Hosts)}). | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 			Debug("Hostmap vpnIp deleted") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | // Only used by pendingHostMap when the remote index is not initially known | 
					
						
							|  |  |  | func (hm *HostMap) addRemoteIndexHostInfo(index uint32, h *HostInfo) { | 
					
						
							|  |  |  | 	hm.Lock() | 
					
						
							|  |  |  | 	h.remoteIndexId = index | 
					
						
							|  |  |  | 	hm.RemoteIndexes[index] = h | 
					
						
							|  |  |  | 	hm.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	if hm.l.Level > logrus.DebugLevel { | 
					
						
							|  |  |  | 		hm.l.WithField("hostMap", m{"mapName": hm.name, "indexNumber": index, "mapTotalSize": len(hm.Indexes), | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 			"hostinfo": m{"existing": true, "localIndexId": h.localIndexId, "hostId": IntIp(h.hostId)}}). | 
					
						
							|  |  |  | 			Debug("Hostmap remoteIndex added") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | func (hm *HostMap) AddVpnIPHostInfo(vpnIP uint32, h *HostInfo) { | 
					
						
							|  |  |  | 	hm.Lock() | 
					
						
							|  |  |  | 	h.hostId = vpnIP | 
					
						
							|  |  |  | 	hm.Hosts[vpnIP] = h | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 	hm.Indexes[h.localIndexId] = h | 
					
						
							|  |  |  | 	hm.RemoteIndexes[h.remoteIndexId] = h | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	hm.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	if hm.l.Level > logrus.DebugLevel { | 
					
						
							|  |  |  | 		hm.l.WithField("hostMap", m{"mapName": hm.name, "vpnIp": IntIp(vpnIP), "mapTotalSize": len(hm.Hosts), | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 			"hostinfo": m{"existing": true, "localIndexId": h.localIndexId, "hostId": IntIp(h.hostId)}}). | 
					
						
							|  |  |  | 			Debug("Hostmap vpnIp added") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-01 12:40:46 -05:00
										 |  |  | // This is only called in pendingHostmap, to cleanup an inbound handshake | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | func (hm *HostMap) DeleteIndex(index uint32) { | 
					
						
							|  |  |  | 	hm.Lock() | 
					
						
							| 
									
										
										
										
											2021-03-01 12:40:46 -05:00
										 |  |  | 	hostinfo, ok := hm.Indexes[index] | 
					
						
							|  |  |  | 	if ok { | 
					
						
							|  |  |  | 		delete(hm.Indexes, index) | 
					
						
							|  |  |  | 		delete(hm.RemoteIndexes, hostinfo.remoteIndexId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Check if we have an entry under hostId that matches the same hostinfo | 
					
						
							|  |  |  | 		// instance. Clean it up as well if we do. | 
					
						
							| 
									
										
										
										
											2021-03-12 14:16:25 -05:00
										 |  |  | 		hostinfo2, ok := hm.Hosts[hostinfo.hostId] | 
					
						
							| 
									
										
										
										
											2021-03-01 12:40:46 -05:00
										 |  |  | 		if ok && hostinfo2 == hostinfo { | 
					
						
							|  |  |  | 			delete(hm.Hosts, hostinfo.hostId) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	hm.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	if hm.l.Level >= logrus.DebugLevel { | 
					
						
							|  |  |  | 		hm.l.WithField("hostMap", m{"mapName": hm.name, "indexNumber": index, "mapTotalSize": len(hm.Indexes)}). | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 			Debug("Hostmap index deleted") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-01 12:40:46 -05:00
										 |  |  | // This is used to cleanup on recv_error | 
					
						
							|  |  |  | func (hm *HostMap) DeleteReverseIndex(index uint32) { | 
					
						
							|  |  |  | 	hm.Lock() | 
					
						
							|  |  |  | 	hostinfo, ok := hm.RemoteIndexes[index] | 
					
						
							|  |  |  | 	if ok { | 
					
						
							|  |  |  | 		delete(hm.Indexes, hostinfo.localIndexId) | 
					
						
							|  |  |  | 		delete(hm.RemoteIndexes, index) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Check if we have an entry under hostId that matches the same hostinfo | 
					
						
							|  |  |  | 		// instance. Clean it up as well if we do (they might not match in pendingHostmap) | 
					
						
							|  |  |  | 		var hostinfo2 *HostInfo | 
					
						
							|  |  |  | 		hostinfo2, ok = hm.Hosts[hostinfo.hostId] | 
					
						
							|  |  |  | 		if ok && hostinfo2 == hostinfo { | 
					
						
							|  |  |  | 			delete(hm.Hosts, hostinfo.hostId) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	hm.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	if hm.l.Level >= logrus.DebugLevel { | 
					
						
							|  |  |  | 		hm.l.WithField("hostMap", m{"mapName": hm.name, "indexNumber": index, "mapTotalSize": len(hm.Indexes)}). | 
					
						
							| 
									
										
										
										
											2021-03-01 12:40:46 -05:00
										 |  |  | 			Debug("Hostmap remote index deleted") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | func (hm *HostMap) DeleteHostInfo(hostinfo *HostInfo) { | 
					
						
							|  |  |  | 	hm.Lock() | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	defer hm.Unlock() | 
					
						
							|  |  |  | 	hm.unlockedDeleteHostInfo(hostinfo) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-01 12:40:46 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | func (hm *HostMap) unlockedDeleteHostInfo(hostinfo *HostInfo) { | 
					
						
							| 
									
										
										
										
											2021-03-01 12:40:46 -05:00
										 |  |  | 	// Check if this same hostId is in the hostmap with a different instance. | 
					
						
							|  |  |  | 	// This could happen if we have an entry in the pending hostmap with different | 
					
						
							|  |  |  | 	// index values than the one in the main hostmap. | 
					
						
							|  |  |  | 	hostinfo2, ok := hm.Hosts[hostinfo.hostId] | 
					
						
							|  |  |  | 	if ok && hostinfo2 != hostinfo { | 
					
						
							|  |  |  | 		delete(hm.Hosts, hostinfo2.hostId) | 
					
						
							|  |  |  | 		delete(hm.Indexes, hostinfo2.localIndexId) | 
					
						
							|  |  |  | 		delete(hm.RemoteIndexes, hostinfo2.remoteIndexId) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 	delete(hm.Hosts, hostinfo.hostId) | 
					
						
							|  |  |  | 	if len(hm.Hosts) == 0 { | 
					
						
							|  |  |  | 		hm.Hosts = map[uint32]*HostInfo{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	delete(hm.Indexes, hostinfo.localIndexId) | 
					
						
							|  |  |  | 	if len(hm.Indexes) == 0 { | 
					
						
							|  |  |  | 		hm.Indexes = map[uint32]*HostInfo{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	delete(hm.RemoteIndexes, hostinfo.remoteIndexId) | 
					
						
							|  |  |  | 	if len(hm.RemoteIndexes) == 0 { | 
					
						
							|  |  |  | 		hm.RemoteIndexes = map[uint32]*HostInfo{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	if hm.l.Level >= logrus.DebugLevel { | 
					
						
							|  |  |  | 		hm.l.WithField("hostMap", m{"mapName": hm.name, "mapTotalSize": len(hm.Hosts), | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 			"vpnIp": IntIp(hostinfo.hostId), "indexNumber": hostinfo.localIndexId, "remoteIndexNumber": hostinfo.remoteIndexId}). | 
					
						
							|  |  |  | 			Debug("Hostmap hostInfo deleted") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | func (hm *HostMap) QueryIndex(index uint32) (*HostInfo, error) { | 
					
						
							|  |  |  | 	//TODO: we probably just want ot return bool instead of error, or at least a static error | 
					
						
							|  |  |  | 	hm.RLock() | 
					
						
							|  |  |  | 	if h, ok := hm.Indexes[index]; ok { | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		return h, nil | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		return nil, errors.New("unable to find index") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) QueryReverseIndex(index uint32) (*HostInfo, error) { | 
					
						
							|  |  |  | 	hm.RLock() | 
					
						
							| 
									
										
										
										
											2020-11-23 14:51:16 -05:00
										 |  |  | 	if h, ok := hm.RemoteIndexes[index]; ok { | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		return h, nil | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		return nil, fmt.Errorf("unable to find reverse index or connectionstate nil in %s hostmap", hm.name) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) QueryVpnIP(vpnIp uint32) (*HostInfo, error) { | 
					
						
							|  |  |  | 	return hm.queryVpnIP(vpnIp, nil) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // PromoteBestQueryVpnIP will attempt to lazily switch to the best remote every | 
					
						
							|  |  |  | // `PromoteEvery` calls to this function for a given host. | 
					
						
							|  |  |  | func (hm *HostMap) PromoteBestQueryVpnIP(vpnIp uint32, ifce *Interface) (*HostInfo, error) { | 
					
						
							|  |  |  | 	return hm.queryVpnIP(vpnIp, ifce) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) queryVpnIP(vpnIp uint32, promoteIfce *Interface) (*HostInfo, error) { | 
					
						
							|  |  |  | 	hm.RLock() | 
					
						
							|  |  |  | 	if h, ok := hm.Hosts[vpnIp]; ok { | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 		// Do not attempt promotion if you are a lighthouse | 
					
						
							|  |  |  | 		if promoteIfce != nil && !promoteIfce.lightHouse.amLighthouse { | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 			h.TryPromoteBest(hm.preferredRanges, promoteIfce) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		return h, nil | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		//return &net.UDPAddr{}, nil, errors.New("Unable to find host") | 
					
						
							|  |  |  | 		hm.RUnlock() | 
					
						
							|  |  |  | 		/* | 
					
						
							|  |  |  | 			if lightHouse != nil { | 
					
						
							|  |  |  | 				lightHouse.Query(vpnIp) | 
					
						
							|  |  |  | 				return nil, errors.New("Unable to find host") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		*/ | 
					
						
							|  |  |  | 		return nil, errors.New("unable to find host") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | func (hm *HostMap) queryUnsafeRoute(ip uint32) uint32 { | 
					
						
							|  |  |  | 	r := hm.unsafeRoutes.MostSpecificContains(ip) | 
					
						
							|  |  |  | 	if r != nil { | 
					
						
							|  |  |  | 		return r.(uint32) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return 0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 14:16:25 -05:00
										 |  |  | // We already have the hm Lock when this is called, so make sure to not call | 
					
						
							|  |  |  | // any other methods that might try to grab it again | 
					
						
							|  |  |  | func (hm *HostMap) addHostInfo(hostinfo *HostInfo, f *Interface) { | 
					
						
							|  |  |  | 	if f.serveDns { | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 		remoteCert := hostinfo.ConnectionState.peerCert | 
					
						
							| 
									
										
										
										
											2021-03-12 14:16:25 -05:00
										 |  |  | 		dnsR.Add(remoteCert.Details.Name+".", remoteCert.Details.Ips[0].IP.String()) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-12 14:16:25 -05:00
										 |  |  | 	hm.Hosts[hostinfo.hostId] = hostinfo | 
					
						
							|  |  |  | 	hm.Indexes[hostinfo.localIndexId] = hostinfo | 
					
						
							|  |  |  | 	hm.RemoteIndexes[hostinfo.remoteIndexId] = hostinfo | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 	if hm.l.Level >= logrus.DebugLevel { | 
					
						
							|  |  |  | 		hm.l.WithField("hostMap", m{"mapName": hm.name, "vpnIp": IntIp(hostinfo.hostId), "mapTotalSize": len(hm.Hosts), | 
					
						
							| 
									
										
										
										
											2021-03-12 14:16:25 -05:00
										 |  |  | 			"hostinfo": m{"existing": true, "localIndexId": hostinfo.localIndexId, "hostId": IntIp(hostinfo.hostId)}}). | 
					
						
							|  |  |  | 			Debug("Hostmap vpnIp added") | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | // punchList assembles a list of all non nil RemoteList pointer entries in this hostmap | 
					
						
							|  |  |  | // The caller can then do the its work outside of the read lock | 
					
						
							|  |  |  | func (hm *HostMap) punchList(rl []*RemoteList) []*RemoteList { | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	hm.RLock() | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	defer hm.RUnlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	for _, v := range hm.Hosts { | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 		if v.remotes != nil { | 
					
						
							|  |  |  | 			rl = append(rl, v.remotes) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	return rl | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | // Punchy iterates through the result of punchList() to assemble all known addresses and sends a hole punch packet to them | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | func (hm *HostMap) Punchy(conn *udpConn) { | 
					
						
							| 
									
										
										
										
											2020-06-26 13:45:48 -04:00
										 |  |  | 	var metricsTxPunchy metrics.Counter | 
					
						
							|  |  |  | 	if hm.metricsEnabled { | 
					
						
							|  |  |  | 		metricsTxPunchy = metrics.GetOrRegisterCounter("messages.tx.punchy", nil) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		metricsTxPunchy = metrics.NilCounter{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	var remotes []*RemoteList | 
					
						
							| 
									
										
										
										
											2021-03-18 20:37:24 -05:00
										 |  |  | 	b := []byte{1} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	for { | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 		remotes = hm.punchList(remotes[:0]) | 
					
						
							|  |  |  | 		for _, rl := range remotes { | 
					
						
							|  |  |  | 			//TODO: CopyAddrs generates garbage but ForEach locks for the work here, figure out which way is better | 
					
						
							|  |  |  | 			for _, addr := range rl.CopyAddrs(hm.preferredRanges) { | 
					
						
							|  |  |  | 				metricsTxPunchy.Inc(1) | 
					
						
							|  |  |  | 				conn.WriteTo(b, addr) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 		time.Sleep(time.Second * 10) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | func (hm *HostMap) addUnsafeRoutes(routes *[]route) { | 
					
						
							|  |  |  | 	for _, r := range *routes { | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 		hm.l.WithField("route", r.route).WithField("via", r.via).Warn("Adding UNSAFE Route") | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | 		hm.unsafeRoutes.AddCIDR(r.route, ip2int(*r.via)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | func (i *HostInfo) BindConnectionState(cs *ConnectionState) { | 
					
						
							|  |  |  | 	i.ConnectionState = cs | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | // TryPromoteBest handles re-querying lighthouses and probing for better paths | 
					
						
							|  |  |  | // NOTE: It is an error to call this if you are a lighthouse since they should not roam clients! | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | func (i *HostInfo) TryPromoteBest(preferredRanges []*net.IPNet, ifce *Interface) { | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	c := atomic.AddUint32(&i.promoteCounter, 1) | 
					
						
							|  |  |  | 	if c%PromoteEvery == 0 { | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 		// return early if we are already on a preferred remote | 
					
						
							| 
									
										
										
										
											2021-03-18 20:37:24 -05:00
										 |  |  | 		rIP := i.remote.IP | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 		for _, l := range preferredRanges { | 
					
						
							|  |  |  | 			if l.Contains(rIP) { | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 		i.remotes.ForEach(preferredRanges, func(addr *udpAddr, preferred bool) { | 
					
						
							|  |  |  | 			if addr == nil || !preferred { | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// Try to send a test packet to that host, this should | 
					
						
							|  |  |  | 			// cause it to detect a roaming event and switch remotes | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 			ifce.send(test, testRequest, i.ConnectionState, i, addr, []byte(""), make([]byte, 12, 12), make([]byte, mtu)) | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	// Re query our lighthouses for new remotes occasionally | 
					
						
							|  |  |  | 	if c%ReQueryEvery == 0 && ifce.lightHouse != nil { | 
					
						
							|  |  |  | 		ifce.lightHouse.QueryServer(i.hostId, ifce) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-27 22:23:18 -04:00
										 |  |  | func (i *HostInfo) cachePacket(l *logrus.Logger, t NebulaMessageType, st NebulaMessageSubType, packet []byte, f packetCallback, m *cachedPacketMetrics) { | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	//TODO: return the error so we can log with more context | 
					
						
							|  |  |  | 	if len(i.packetStore) < 100 { | 
					
						
							|  |  |  | 		tempPacket := make([]byte, len(packet)) | 
					
						
							|  |  |  | 		copy(tempPacket, packet) | 
					
						
							|  |  |  | 		//l.WithField("trace", string(debug.Stack())).Error("Caching packet", tempPacket) | 
					
						
							|  |  |  | 		i.packetStore = append(i.packetStore, &cachedPacket{t, st, f, tempPacket}) | 
					
						
							| 
									
										
										
										
											2021-03-05 21:18:33 -05:00
										 |  |  | 		if l.Level >= logrus.DebugLevel { | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 			i.logger(l). | 
					
						
							| 
									
										
										
										
											2021-03-05 21:18:33 -05:00
										 |  |  | 				WithField("length", len(i.packetStore)). | 
					
						
							|  |  |  | 				WithField("stored", true). | 
					
						
							|  |  |  | 				Debugf("Packet store") | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	} else if l.Level >= logrus.DebugLevel { | 
					
						
							| 
									
										
										
										
											2021-04-27 22:23:18 -04:00
										 |  |  | 		m.dropped.Inc(1) | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 		i.logger(l). | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 			WithField("length", len(i.packetStore)). | 
					
						
							|  |  |  | 			WithField("stored", false). | 
					
						
							|  |  |  | 			Debugf("Packet store") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // handshakeComplete will set the connection as ready to communicate, as well as flush any stored packets | 
					
						
							| 
									
										
										
										
											2021-04-27 22:23:18 -04:00
										 |  |  | func (i *HostInfo) handshakeComplete(l *logrus.Logger, m *cachedPacketMetrics) { | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	//TODO: I'm not certain the distinction between handshake complete and ConnectionState being ready matters because: | 
					
						
							|  |  |  | 	//TODO: HandshakeComplete means send stored packets and ConnectionState.ready means we are ready to send | 
					
						
							|  |  |  | 	//TODO: if the transition from HandhsakeComplete to ConnectionState.ready happens all within this function they are identical | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i.ConnectionState.queueLock.Lock() | 
					
						
							|  |  |  | 	i.HandshakeComplete = true | 
					
						
							|  |  |  | 	//TODO: this should be managed by the handshake state machine to set it based on how many handshake were seen. | 
					
						
							|  |  |  | 	// Clamping it to 2 gets us out of the woods for now | 
					
						
							| 
									
										
										
										
											2021-03-05 21:18:33 -05:00
										 |  |  | 	atomic.StoreUint64(&i.ConnectionState.atomicMessageCounter, 2) | 
					
						
							| 
									
										
										
										
											2021-03-22 18:25:06 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if l.Level >= logrus.DebugLevel { | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | 		i.logger(l).Debugf("Sending %d stored packets", len(i.packetStore)) | 
					
						
							| 
									
										
										
										
											2021-03-22 18:25:06 -05:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(i.packetStore) > 0 { | 
					
						
							|  |  |  | 		nb := make([]byte, 12, 12) | 
					
						
							|  |  |  | 		out := make([]byte, mtu) | 
					
						
							|  |  |  | 		for _, cp := range i.packetStore { | 
					
						
							|  |  |  | 			cp.callback(cp.messageType, cp.messageSubType, i, cp.packet, nb, out) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-04-27 22:23:18 -04:00
										 |  |  | 		m.sent.Inc(int64(len(i.packetStore))) | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-03-22 18:25:06 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	i.remotes.ResetBlockedRemotes() | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	i.packetStore = make([]*cachedPacket, 0) | 
					
						
							|  |  |  | 	i.ConnectionState.ready = true | 
					
						
							|  |  |  | 	i.ConnectionState.queueLock.Unlock() | 
					
						
							|  |  |  | 	i.ConnectionState.certState = nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (i *HostInfo) GetCert() *cert.NebulaCertificate { | 
					
						
							|  |  |  | 	if i.ConnectionState != nil { | 
					
						
							|  |  |  | 		return i.ConnectionState.peerCert | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-18 20:37:24 -05:00
										 |  |  | func (i *HostInfo) SetRemote(remote *udpAddr) { | 
					
						
							| 
									
										
										
										
											2021-04-14 13:50:09 -05:00
										 |  |  | 	// We copy here because we likely got this remote from a source that reuses the object | 
					
						
							|  |  |  | 	if !i.remote.Equals(remote) { | 
					
						
							|  |  |  | 		i.remote = remote.Copy() | 
					
						
							|  |  |  | 		i.remotes.LearnRemote(i.hostId, remote.Copy()) | 
					
						
							| 
									
										
										
										
											2021-03-31 10:26:35 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (i *HostInfo) ClearConnectionState() { | 
					
						
							|  |  |  | 	i.ConnectionState = nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (i *HostInfo) RecvErrorExceeded() bool { | 
					
						
							|  |  |  | 	if i.recvError < 3 { | 
					
						
							|  |  |  | 		i.recvError += 1 | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | func (i *HostInfo) CreateRemoteCIDR(c *cert.NebulaCertificate) { | 
					
						
							| 
									
										
										
										
											2020-03-02 16:21:33 -05:00
										 |  |  | 	if len(c.Details.Ips) == 1 && len(c.Details.Subnets) == 0 { | 
					
						
							|  |  |  | 		// Simple case, no CIDRTree needed | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 16:34:17 +00:00
										 |  |  | 	remoteCidr := NewCIDRTree() | 
					
						
							|  |  |  | 	for _, ip := range c.Details.Ips { | 
					
						
							|  |  |  | 		remoteCidr.AddCIDR(&net.IPNet{IP: ip.IP, Mask: net.IPMask{255, 255, 255, 255}}, struct{}{}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, n := range c.Details.Subnets { | 
					
						
							|  |  |  | 		remoteCidr.AddCIDR(n, struct{}{}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	i.remoteCidr = remoteCidr | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | func (i *HostInfo) logger(l *logrus.Logger) *logrus.Entry { | 
					
						
							| 
									
										
										
										
											2020-04-06 14:34:00 -04:00
										 |  |  | 	if i == nil { | 
					
						
							|  |  |  | 		return logrus.NewEntry(l) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	li := l.WithField("vpnIp", IntIp(i.hostId)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if connState := i.ConnectionState; connState != nil { | 
					
						
							|  |  |  | 		if peerCert := connState.peerCert; peerCert != nil { | 
					
						
							|  |  |  | 			li = li.WithField("certName", peerCert.Details.Name) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return li | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | //######################## | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (hm *HostMap) DebugRemotes(vpnIp uint32) string { | 
					
						
							|  |  |  | 	s := "\n" | 
					
						
							|  |  |  | 	for _, h := range hm.Hosts { | 
					
						
							|  |  |  | 		for _, r := range h.Remotes { | 
					
						
							|  |  |  | 			s += fmt.Sprintf("%s : %d ## %v\n", r.addr.IP.String(), r.addr.Port, r.probes) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return s | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (i *HostInfo) HandleReply(addr *net.UDPAddr, counter int) { | 
					
						
							|  |  |  | 	for _, r := range i.Remotes { | 
					
						
							|  |  |  | 		if r.addr.IP.Equal(addr.IP) && r.addr.Port == addr.Port { | 
					
						
							|  |  |  | 			r.ProbeReceived(counter) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (i *HostInfo) Probes() []*Probe { | 
					
						
							|  |  |  | 	p := []*Probe{} | 
					
						
							|  |  |  | 	for _, d := range i.Remotes { | 
					
						
							|  |  |  | 		p = append(p, &Probe{Addr: d.addr, Counter: d.Probe()}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return p | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Utility functions | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 09:46:30 -05:00
										 |  |  | func localIps(l *logrus.Logger, allowList *AllowList) *[]net.IP { | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 	//FIXME: This function is pretty garbage | 
					
						
							|  |  |  | 	var ips []net.IP | 
					
						
							|  |  |  | 	ifaces, _ := net.Interfaces() | 
					
						
							|  |  |  | 	for _, i := range ifaces { | 
					
						
							| 
									
										
											  
											
												Add lighthouse.{remoteAllowList,localAllowList} (#217)
These settings make it possible to blacklist / whitelist IP addresses
that are used for remote connections.
`lighthouse.remoteAllowList` filters which remote IPs are allow when
fetching from the lighthouse (or, if you are the lighthouse, which IPs
you store and forward to querying hosts). By default, any remote IPs are
allowed. You can provide CIDRs here with `true` to allow and `false` to
deny. The most specific CIDR rule applies to each remote.  If all rules
are "allow", the default will be "deny", and vice-versa. If both "allow"
and "deny" rules are present, then you MUST set a rule for "0.0.0.0/0"
as the default.
    lighthouse:
      remoteAllowList:
        # Example to block IPs from this subnet from being used for remote IPs.
        "172.16.0.0/12": false
        # A more complicated example, allow public IPs but only private IPs from a specific subnet
        "0.0.0.0/0": true
        "10.0.0.0/8": false
        "10.42.42.0/24": true
`lighthouse.localAllowList` has the same logic as above, but it applies
to the local addresses we advertise to the lighthouse. Additionally, you
can specify an `interfaces` map of regular expressions to match against
interface names. The regexp must match the entire name. All interface
rules must be either true or false (and the default rule will be the
inverse). CIDR rules are matched after interface name rules.
Default is all local IP addresses.
    lighthouse:
      localAllowList:
        # Example to blacklist docker interfaces.
        interfaces:
          'docker.*': false
        # Example to only advertise IPs in this subnet to the lighthouse.
        "10.0.0.0/8": true
											
										 
											2020-04-08 15:36:43 -04:00
										 |  |  | 		allow := allowList.AllowName(i.Name) | 
					
						
							| 
									
										
										
										
											2021-03-31 10:26:35 -05:00
										 |  |  | 		if l.Level >= logrus.TraceLevel { | 
					
						
							|  |  |  | 			l.WithField("interfaceName", i.Name).WithField("allow", allow).Trace("localAllowList.AllowName") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Add lighthouse.{remoteAllowList,localAllowList} (#217)
These settings make it possible to blacklist / whitelist IP addresses
that are used for remote connections.
`lighthouse.remoteAllowList` filters which remote IPs are allow when
fetching from the lighthouse (or, if you are the lighthouse, which IPs
you store and forward to querying hosts). By default, any remote IPs are
allowed. You can provide CIDRs here with `true` to allow and `false` to
deny. The most specific CIDR rule applies to each remote.  If all rules
are "allow", the default will be "deny", and vice-versa. If both "allow"
and "deny" rules are present, then you MUST set a rule for "0.0.0.0/0"
as the default.
    lighthouse:
      remoteAllowList:
        # Example to block IPs from this subnet from being used for remote IPs.
        "172.16.0.0/12": false
        # A more complicated example, allow public IPs but only private IPs from a specific subnet
        "0.0.0.0/0": true
        "10.0.0.0/8": false
        "10.42.42.0/24": true
`lighthouse.localAllowList` has the same logic as above, but it applies
to the local addresses we advertise to the lighthouse. Additionally, you
can specify an `interfaces` map of regular expressions to match against
interface names. The regexp must match the entire name. All interface
rules must be either true or false (and the default rule will be the
inverse). CIDR rules are matched after interface name rules.
Default is all local IP addresses.
    lighthouse:
      localAllowList:
        # Example to blacklist docker interfaces.
        interfaces:
          'docker.*': false
        # Example to only advertise IPs in this subnet to the lighthouse.
        "10.0.0.0/8": true
											
										 
											2020-04-08 15:36:43 -04:00
										 |  |  | 		if !allow { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 		addrs, _ := i.Addrs() | 
					
						
							|  |  |  | 		for _, addr := range addrs { | 
					
						
							|  |  |  | 			var ip net.IP | 
					
						
							|  |  |  | 			switch v := addr.(type) { | 
					
						
							|  |  |  | 			case *net.IPNet: | 
					
						
							|  |  |  | 				//continue | 
					
						
							|  |  |  | 				ip = v.IP | 
					
						
							|  |  |  | 			case *net.IPAddr: | 
					
						
							|  |  |  | 				ip = v.IP | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-03-18 20:37:24 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			//TODO: Filtering out link local for now, this is probably the most correct thing | 
					
						
							|  |  |  | 			//TODO: Would be nice to filter out SLAAC MAC based ips as well | 
					
						
							|  |  |  | 			if ip.IsLoopback() == false && !ip.IsLinkLocalUnicast() { | 
					
						
							|  |  |  | 				allow := allowList.Allow(ip) | 
					
						
							| 
									
										
										
										
											2021-03-31 10:26:35 -05:00
										 |  |  | 				if l.Level >= logrus.TraceLevel { | 
					
						
							|  |  |  | 					l.WithField("localIp", ip).WithField("allow", allow).Trace("localAllowList.Allow") | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
											  
											
												Add lighthouse.{remoteAllowList,localAllowList} (#217)
These settings make it possible to blacklist / whitelist IP addresses
that are used for remote connections.
`lighthouse.remoteAllowList` filters which remote IPs are allow when
fetching from the lighthouse (or, if you are the lighthouse, which IPs
you store and forward to querying hosts). By default, any remote IPs are
allowed. You can provide CIDRs here with `true` to allow and `false` to
deny. The most specific CIDR rule applies to each remote.  If all rules
are "allow", the default will be "deny", and vice-versa. If both "allow"
and "deny" rules are present, then you MUST set a rule for "0.0.0.0/0"
as the default.
    lighthouse:
      remoteAllowList:
        # Example to block IPs from this subnet from being used for remote IPs.
        "172.16.0.0/12": false
        # A more complicated example, allow public IPs but only private IPs from a specific subnet
        "0.0.0.0/0": true
        "10.0.0.0/8": false
        "10.42.42.0/24": true
`lighthouse.localAllowList` has the same logic as above, but it applies
to the local addresses we advertise to the lighthouse. Additionally, you
can specify an `interfaces` map of regular expressions to match against
interface names. The regexp must match the entire name. All interface
rules must be either true or false (and the default rule will be the
inverse). CIDR rules are matched after interface name rules.
Default is all local IP addresses.
    lighthouse:
      localAllowList:
        # Example to blacklist docker interfaces.
        interfaces:
          'docker.*': false
        # Example to only advertise IPs in this subnet to the lighthouse.
        "10.0.0.0/8": true
											
										 
											2020-04-08 15:36:43 -04:00
										 |  |  | 				if !allow { | 
					
						
							|  |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-19 17:00:20 +00:00
										 |  |  | 				ips = append(ips, ip) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return &ips | 
					
						
							|  |  |  | } |