nodes.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676
  1. package logic
  2. import (
  3. "context"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "net"
  8. "sort"
  9. "sync"
  10. "time"
  11. validator "github.com/go-playground/validator/v10"
  12. "github.com/google/uuid"
  13. "github.com/gravitl/netmaker/database"
  14. "github.com/gravitl/netmaker/logger"
  15. "github.com/gravitl/netmaker/logic/acls"
  16. "github.com/gravitl/netmaker/logic/acls/nodeacls"
  17. "github.com/gravitl/netmaker/models"
  18. "github.com/gravitl/netmaker/servercfg"
  19. "github.com/gravitl/netmaker/validation"
  20. "github.com/seancfoley/ipaddress-go/ipaddr"
  21. "golang.org/x/exp/slog"
  22. )
  23. var (
  24. nodeCacheMutex = &sync.RWMutex{}
  25. nodesCacheMap = make(map[string]models.Node)
  26. )
  27. func getNodeFromCache(nodeID string) (node models.Node, ok bool) {
  28. nodeCacheMutex.RLock()
  29. node, ok = nodesCacheMap[nodeID]
  30. nodeCacheMutex.RUnlock()
  31. return
  32. }
  33. func getNodesFromCache() (nodes []models.Node) {
  34. nodeCacheMutex.RLock()
  35. for _, node := range nodesCacheMap {
  36. nodes = append(nodes, node)
  37. }
  38. nodeCacheMutex.RUnlock()
  39. return
  40. }
  41. func deleteNodeFromCache(nodeID string) {
  42. nodeCacheMutex.Lock()
  43. delete(nodesCacheMap, nodeID)
  44. nodeCacheMutex.Unlock()
  45. }
  46. func storeNodeInCache(node models.Node) {
  47. nodeCacheMutex.Lock()
  48. nodesCacheMap[node.ID.String()] = node
  49. nodeCacheMutex.Unlock()
  50. }
  51. func loadNodesIntoCache(nMap map[string]models.Node) {
  52. nodeCacheMutex.Lock()
  53. nodesCacheMap = nMap
  54. nodeCacheMutex.Unlock()
  55. }
  56. func ClearNodeCache() {
  57. nodeCacheMutex.Lock()
  58. nodesCacheMap = make(map[string]models.Node)
  59. nodeCacheMutex.Unlock()
  60. }
  61. const (
  62. // RELAY_NODE_ERR - error to return if relay node is unfound
  63. RELAY_NODE_ERR = "could not find relay for node"
  64. // NodePurgeTime time to wait for node to response to a NODE_DELETE actions
  65. NodePurgeTime = time.Second * 10
  66. // NodePurgeCheckTime is how often to check nodes for Pending Delete
  67. NodePurgeCheckTime = time.Second * 30
  68. )
  69. // GetNetworkNodes - gets the nodes of a network
  70. func GetNetworkNodes(network string) ([]models.Node, error) {
  71. allnodes, err := GetAllNodes()
  72. if err != nil {
  73. return []models.Node{}, err
  74. }
  75. return GetNetworkNodesMemory(allnodes, network), nil
  76. }
  77. // GetHostNodes - fetches all nodes part of the host
  78. func GetHostNodes(host *models.Host) []models.Node {
  79. nodes := []models.Node{}
  80. for _, nodeID := range host.Nodes {
  81. node, err := GetNodeByID(nodeID)
  82. if err == nil {
  83. nodes = append(nodes, node)
  84. }
  85. }
  86. return nodes
  87. }
  88. // GetNetworkNodesMemory - gets all nodes belonging to a network from list in memory
  89. func GetNetworkNodesMemory(allNodes []models.Node, network string) []models.Node {
  90. var nodes = []models.Node{}
  91. for i := range allNodes {
  92. node := allNodes[i]
  93. if node.Network == network {
  94. nodes = append(nodes, node)
  95. }
  96. }
  97. return nodes
  98. }
  99. // UpdateNodeCheckin - updates the checkin time of a node
  100. func UpdateNodeCheckin(node *models.Node) error {
  101. node.SetLastCheckIn()
  102. data, err := json.Marshal(node)
  103. if err != nil {
  104. return err
  105. }
  106. err = database.Insert(node.ID.String(), string(data), database.NODES_TABLE_NAME)
  107. if err != nil {
  108. return err
  109. }
  110. if servercfg.CacheEnabled() {
  111. storeNodeInCache(*node)
  112. }
  113. return nil
  114. }
  115. // UpsertNode - updates node in the DB
  116. func UpsertNode(newNode *models.Node) error {
  117. newNode.SetLastModified()
  118. data, err := json.Marshal(newNode)
  119. if err != nil {
  120. return err
  121. }
  122. err = database.Insert(newNode.ID.String(), string(data), database.NODES_TABLE_NAME)
  123. if err != nil {
  124. return err
  125. }
  126. if servercfg.CacheEnabled() {
  127. storeNodeInCache(*newNode)
  128. }
  129. return nil
  130. }
  131. // UpdateNode - takes a node and updates another node with it's values
  132. func UpdateNode(currentNode *models.Node, newNode *models.Node) error {
  133. if newNode.Address.IP.String() != currentNode.Address.IP.String() {
  134. if network, err := GetParentNetwork(newNode.Network); err == nil {
  135. if !IsAddressInCIDR(newNode.Address.IP, network.AddressRange) {
  136. return fmt.Errorf("invalid address provided; out of network range for node %s", newNode.ID)
  137. }
  138. }
  139. }
  140. nodeACLDelta := currentNode.DefaultACL != newNode.DefaultACL
  141. newNode.Fill(currentNode, servercfg.IsPro)
  142. // check for un-settable server values
  143. if err := ValidateNode(newNode, true); err != nil {
  144. return err
  145. }
  146. if newNode.ID == currentNode.ID {
  147. if nodeACLDelta {
  148. if err := UpdateProNodeACLs(newNode); err != nil {
  149. logger.Log(1, "failed to apply node level ACLs during creation of node", newNode.ID.String(), "-", err.Error())
  150. return err
  151. }
  152. }
  153. newNode.SetLastModified()
  154. if data, err := json.Marshal(newNode); err != nil {
  155. return err
  156. } else {
  157. err = database.Insert(newNode.ID.String(), string(data), database.NODES_TABLE_NAME)
  158. if err != nil {
  159. return err
  160. }
  161. if servercfg.CacheEnabled() {
  162. storeNodeInCache(*newNode)
  163. }
  164. return nil
  165. }
  166. }
  167. return fmt.Errorf("failed to update node " + currentNode.ID.String() + ", cannot change ID.")
  168. }
  169. // DeleteNode - marks node for deletion (and adds to zombie list) if called by UI or deletes node if called by node
  170. func DeleteNode(node *models.Node, purge bool) error {
  171. alreadyDeleted := node.PendingDelete || node.Action == models.NODE_DELETE
  172. node.Action = models.NODE_DELETE
  173. //delete ext clients if node is ingress gw
  174. if node.IsIngressGateway {
  175. if err := DeleteGatewayExtClients(node.ID.String(), node.Network); err != nil {
  176. slog.Error("failed to delete ext clients", "nodeid", node.ID.String(), "error", err.Error())
  177. }
  178. }
  179. if node.IsRelayed {
  180. // cleanup node from relayednodes on relay node
  181. relayNode, err := GetNodeByID(node.RelayedBy)
  182. if err == nil {
  183. relayedNodes := []string{}
  184. for _, relayedNodeID := range relayNode.RelayedNodes {
  185. if relayedNodeID == node.ID.String() {
  186. continue
  187. }
  188. relayedNodes = append(relayedNodes, relayedNodeID)
  189. }
  190. relayNode.RelayedNodes = relayedNodes
  191. UpsertNode(&relayNode)
  192. }
  193. }
  194. if node.FailedOverBy != uuid.Nil {
  195. ResetFailedOverPeer(node)
  196. }
  197. if node.IsRelay {
  198. // unset all the relayed nodes
  199. SetRelayedNodes(false, node.ID.String(), node.RelayedNodes)
  200. }
  201. if node.InternetGwID != "" {
  202. inetNode, err := GetNodeByID(node.InternetGwID)
  203. if err == nil {
  204. clientNodeIDs := []string{}
  205. for _, inetNodeClientID := range inetNode.InetNodeReq.InetNodeClientIDs {
  206. if inetNodeClientID == node.ID.String() {
  207. continue
  208. }
  209. clientNodeIDs = append(clientNodeIDs, inetNodeClientID)
  210. }
  211. inetNode.InetNodeReq.InetNodeClientIDs = clientNodeIDs
  212. UpsertNode(&inetNode)
  213. }
  214. }
  215. if node.IsInternetGateway {
  216. UnsetInternetGw(node)
  217. }
  218. if !purge && !alreadyDeleted {
  219. newnode := *node
  220. newnode.PendingDelete = true
  221. if err := UpdateNode(node, &newnode); err != nil {
  222. return err
  223. }
  224. newZombie <- node.ID
  225. return nil
  226. }
  227. if alreadyDeleted {
  228. logger.Log(1, "forcibly deleting node", node.ID.String())
  229. }
  230. host, err := GetHost(node.HostID.String())
  231. if err != nil {
  232. logger.Log(1, "no host found for node", node.ID.String(), "deleting..")
  233. if delErr := DeleteNodeByID(node); delErr != nil {
  234. logger.Log(0, "failed to delete node", node.ID.String(), delErr.Error())
  235. }
  236. return err
  237. }
  238. if err := DissasociateNodeFromHost(node, host); err != nil {
  239. return err
  240. }
  241. return nil
  242. }
  243. // GetNodeByHostRef - gets the node by host id and network
  244. func GetNodeByHostRef(hostid, network string) (node models.Node, err error) {
  245. nodes, err := GetNetworkNodes(network)
  246. if err != nil {
  247. return models.Node{}, err
  248. }
  249. for _, node := range nodes {
  250. if node.HostID.String() == hostid && node.Network == network {
  251. return node, nil
  252. }
  253. }
  254. return models.Node{}, errors.New("node not found")
  255. }
  256. // DeleteNodeByID - deletes a node from database
  257. func DeleteNodeByID(node *models.Node) error {
  258. var err error
  259. var key = node.ID.String()
  260. if err = database.DeleteRecord(database.NODES_TABLE_NAME, key); err != nil {
  261. if !database.IsEmptyRecord(err) {
  262. return err
  263. }
  264. }
  265. if servercfg.CacheEnabled() {
  266. deleteNodeFromCache(node.ID.String())
  267. }
  268. if servercfg.IsDNSMode() {
  269. SetDNS()
  270. }
  271. _, err = nodeacls.RemoveNodeACL(nodeacls.NetworkID(node.Network), nodeacls.NodeID(node.ID.String()))
  272. if err != nil {
  273. // ignoring for now, could hit a nil pointer if delete called twice
  274. logger.Log(2, "attempted to remove node ACL for node", node.ID.String())
  275. }
  276. // removeZombie <- node.ID
  277. if err = DeleteMetrics(node.ID.String()); err != nil {
  278. logger.Log(1, "unable to remove metrics from DB for node", node.ID.String(), err.Error())
  279. }
  280. return nil
  281. }
  282. // IsNodeIDUnique - checks if node id is unique
  283. func IsNodeIDUnique(node *models.Node) (bool, error) {
  284. _, err := database.FetchRecord(database.NODES_TABLE_NAME, node.ID.String())
  285. return database.IsEmptyRecord(err), err
  286. }
  287. // ValidateNode - validates node values
  288. func ValidateNode(node *models.Node, isUpdate bool) error {
  289. v := validator.New()
  290. _ = v.RegisterValidation("id_unique", func(fl validator.FieldLevel) bool {
  291. if isUpdate {
  292. return true
  293. }
  294. isFieldUnique, _ := IsNodeIDUnique(node)
  295. return isFieldUnique
  296. })
  297. _ = v.RegisterValidation("network_exists", func(fl validator.FieldLevel) bool {
  298. _, err := GetNetworkByNode(node)
  299. return err == nil
  300. })
  301. _ = v.RegisterValidation("checkyesornoorunset", func(f1 validator.FieldLevel) bool {
  302. return validation.CheckYesOrNoOrUnset(f1)
  303. })
  304. err := v.Struct(node)
  305. return err
  306. }
  307. // GetAllNodes - returns all nodes in the DB
  308. func GetAllNodes() ([]models.Node, error) {
  309. var nodes []models.Node
  310. if servercfg.CacheEnabled() {
  311. nodes = getNodesFromCache()
  312. if len(nodes) != 0 {
  313. return nodes, nil
  314. }
  315. }
  316. nodesMap := make(map[string]models.Node)
  317. if servercfg.CacheEnabled() {
  318. defer loadNodesIntoCache(nodesMap)
  319. }
  320. collection, err := database.FetchRecords(database.NODES_TABLE_NAME)
  321. if err != nil {
  322. if database.IsEmptyRecord(err) {
  323. return []models.Node{}, nil
  324. }
  325. return []models.Node{}, err
  326. }
  327. for _, value := range collection {
  328. var node models.Node
  329. // ignore legacy nodes in database
  330. if err := json.Unmarshal([]byte(value), &node); err != nil {
  331. logger.Log(3, "legacy node detected: ", err.Error())
  332. continue
  333. }
  334. // add node to our array
  335. nodes = append(nodes, node)
  336. nodesMap[node.ID.String()] = node
  337. }
  338. return nodes, nil
  339. }
  340. // GetNetworkByNode - gets the network model from a node
  341. func GetNetworkByNode(node *models.Node) (models.Network, error) {
  342. var network = models.Network{}
  343. networkData, err := database.FetchRecord(database.NETWORKS_TABLE_NAME, node.Network)
  344. if err != nil {
  345. return network, err
  346. }
  347. if err = json.Unmarshal([]byte(networkData), &network); err != nil {
  348. return models.Network{}, err
  349. }
  350. return network, nil
  351. }
  352. // SetNodeDefaults - sets the defaults of a node to avoid empty fields
  353. func SetNodeDefaults(node *models.Node) {
  354. parentNetwork, _ := GetNetworkByNode(node)
  355. _, cidr, err := net.ParseCIDR(parentNetwork.AddressRange)
  356. if err == nil {
  357. node.NetworkRange = *cidr
  358. }
  359. _, cidr, err = net.ParseCIDR(parentNetwork.AddressRange6)
  360. if err == nil {
  361. node.NetworkRange6 = *cidr
  362. }
  363. if node.DefaultACL == "" {
  364. node.DefaultACL = parentNetwork.DefaultACL
  365. }
  366. if node.FailOverPeers == nil {
  367. node.FailOverPeers = make(map[string]struct{})
  368. }
  369. node.SetLastModified()
  370. node.SetLastCheckIn()
  371. node.SetDefaultConnected()
  372. node.SetExpirationDateTime()
  373. }
  374. // GetRecordKey - get record key
  375. // depricated
  376. func GetRecordKey(id string, network string) (string, error) {
  377. if id == "" || network == "" {
  378. return "", errors.New("unable to get record key")
  379. }
  380. return id + "###" + network, nil
  381. }
  382. func GetNodeByID(uuid string) (models.Node, error) {
  383. if servercfg.CacheEnabled() {
  384. if node, ok := getNodeFromCache(uuid); ok {
  385. return node, nil
  386. }
  387. }
  388. var record, err = database.FetchRecord(database.NODES_TABLE_NAME, uuid)
  389. if err != nil {
  390. return models.Node{}, err
  391. }
  392. var node models.Node
  393. if err = json.Unmarshal([]byte(record), &node); err != nil {
  394. return models.Node{}, err
  395. }
  396. if servercfg.CacheEnabled() {
  397. storeNodeInCache(node)
  398. }
  399. return node, nil
  400. }
  401. // GetDeletedNodeByID - get a deleted node
  402. func GetDeletedNodeByID(uuid string) (models.Node, error) {
  403. var node models.Node
  404. record, err := database.FetchRecord(database.DELETED_NODES_TABLE_NAME, uuid)
  405. if err != nil {
  406. return models.Node{}, err
  407. }
  408. if err = json.Unmarshal([]byte(record), &node); err != nil {
  409. return models.Node{}, err
  410. }
  411. SetNodeDefaults(&node)
  412. return node, nil
  413. }
  414. // FindRelay - returns the node that is the relay for a relayed node
  415. func FindRelay(node *models.Node) *models.Node {
  416. relay, err := GetNodeByID(node.RelayedBy)
  417. if err != nil {
  418. logger.Log(0, "FindRelay: "+err.Error())
  419. return nil
  420. }
  421. return &relay
  422. }
  423. // GetAllNodesAPI - get all nodes for api usage
  424. func GetAllNodesAPI(nodes []models.Node) []models.ApiNode {
  425. apiNodes := []models.ApiNode{}
  426. for i := range nodes {
  427. newApiNode := nodes[i].ConvertToAPINode()
  428. apiNodes = append(apiNodes, *newApiNode)
  429. }
  430. return apiNodes[:]
  431. }
  432. // DeleteExpiredNodes - goroutine which deletes nodes which are expired
  433. func DeleteExpiredNodes(ctx context.Context, peerUpdate chan *models.Node) {
  434. // Delete Expired Nodes Every Hour
  435. ticker := time.NewTicker(time.Hour)
  436. for {
  437. select {
  438. case <-ctx.Done():
  439. ticker.Stop()
  440. return
  441. case <-ticker.C:
  442. allnodes, err := GetAllNodes()
  443. if err != nil {
  444. slog.Error("failed to retrieve all nodes", "error", err.Error())
  445. return
  446. }
  447. for _, node := range allnodes {
  448. node := node
  449. if time.Now().After(node.ExpirationDateTime) {
  450. peerUpdate <- &node
  451. slog.Info("deleting expired node", "nodeid", node.ID.String())
  452. }
  453. }
  454. }
  455. }
  456. }
  457. // createNode - creates a node in database
  458. func createNode(node *models.Node) error {
  459. // lock because we need unique IPs and having it concurrent makes parallel calls result in same "unique" IPs
  460. addressLock.Lock()
  461. defer addressLock.Unlock()
  462. host, err := GetHost(node.HostID.String())
  463. if err != nil {
  464. return err
  465. }
  466. if !node.DNSOn {
  467. if servercfg.IsDNSMode() {
  468. node.DNSOn = true
  469. } else {
  470. node.DNSOn = false
  471. }
  472. }
  473. SetNodeDefaults(node)
  474. defaultACLVal := acls.Allowed
  475. parentNetwork, err := GetNetwork(node.Network)
  476. if err == nil {
  477. if parentNetwork.DefaultACL != "yes" {
  478. defaultACLVal = acls.NotAllowed
  479. }
  480. }
  481. if node.DefaultACL == "" {
  482. node.DefaultACL = "unset"
  483. }
  484. if node.Address.IP == nil {
  485. if parentNetwork.IsIPv4 == "yes" {
  486. if node.Address.IP, err = UniqueAddress(node.Network, false); err != nil {
  487. return err
  488. }
  489. _, cidr, err := net.ParseCIDR(parentNetwork.AddressRange)
  490. if err != nil {
  491. return err
  492. }
  493. node.Address.Mask = net.CIDRMask(cidr.Mask.Size())
  494. }
  495. } else if !IsIPUnique(node.Network, node.Address.String(), database.NODES_TABLE_NAME, false) {
  496. return fmt.Errorf("invalid address: ipv4 " + node.Address.String() + " is not unique")
  497. }
  498. if node.Address6.IP == nil {
  499. if parentNetwork.IsIPv6 == "yes" {
  500. if node.Address6.IP, err = UniqueAddress6(node.Network, false); err != nil {
  501. return err
  502. }
  503. _, cidr, err := net.ParseCIDR(parentNetwork.AddressRange6)
  504. if err != nil {
  505. return err
  506. }
  507. node.Address6.Mask = net.CIDRMask(cidr.Mask.Size())
  508. }
  509. } else if !IsIPUnique(node.Network, node.Address6.String(), database.NODES_TABLE_NAME, true) {
  510. return fmt.Errorf("invalid address: ipv6 " + node.Address6.String() + " is not unique")
  511. }
  512. node.ID = uuid.New()
  513. //Create a JWT for the node
  514. tokenString, _ := CreateJWT(node.ID.String(), host.MacAddress.String(), node.Network)
  515. if tokenString == "" {
  516. //logic.ReturnErrorResponse(w, r, errorResponse)
  517. return err
  518. }
  519. err = ValidateNode(node, false)
  520. if err != nil {
  521. return err
  522. }
  523. CheckZombies(node)
  524. nodebytes, err := json.Marshal(&node)
  525. if err != nil {
  526. return err
  527. }
  528. err = database.Insert(node.ID.String(), string(nodebytes), database.NODES_TABLE_NAME)
  529. if err != nil {
  530. return err
  531. }
  532. if servercfg.CacheEnabled() {
  533. storeNodeInCache(*node)
  534. }
  535. _, err = nodeacls.CreateNodeACL(nodeacls.NetworkID(node.Network), nodeacls.NodeID(node.ID.String()), defaultACLVal)
  536. if err != nil {
  537. logger.Log(1, "failed to create node ACL for node,", node.ID.String(), "err:", err.Error())
  538. return err
  539. }
  540. if err = UpdateProNodeACLs(node); err != nil {
  541. logger.Log(1, "failed to apply node level ACLs during creation of node", node.ID.String(), "-", err.Error())
  542. return err
  543. }
  544. if err = UpdateMetrics(node.ID.String(), &models.Metrics{Connectivity: make(map[string]models.Metric)}); err != nil {
  545. logger.Log(1, "failed to initialize metrics for node", node.ID.String(), err.Error())
  546. }
  547. SetNetworkNodesLastModified(node.Network)
  548. if servercfg.IsDNSMode() {
  549. err = SetDNS()
  550. }
  551. return err
  552. }
  553. // SortApiNodes - Sorts slice of ApiNodes by their ID alphabetically with numbers first
  554. func SortApiNodes(unsortedNodes []models.ApiNode) {
  555. sort.Slice(unsortedNodes, func(i, j int) bool {
  556. return unsortedNodes[i].ID < unsortedNodes[j].ID
  557. })
  558. }
  559. func ValidateParams(nodeid, netid string) (models.Node, error) {
  560. node, err := GetNodeByID(nodeid)
  561. if err != nil {
  562. slog.Error("error fetching node", "node", nodeid, "error", err.Error())
  563. return node, fmt.Errorf("error fetching node during parameter validation: %v", err)
  564. }
  565. if node.Network != netid {
  566. slog.Error("network url param does not match node id", "url nodeid", netid, "node", node.Network)
  567. return node, fmt.Errorf("network url param does not match node network")
  568. }
  569. return node, nil
  570. }
  571. func ValidateEgressRange(gateway models.EgressGatewayRequest) error {
  572. network, err := GetNetworkSettings(gateway.NetID)
  573. if err != nil {
  574. slog.Error("error getting network with netid", "error", gateway.NetID, err.Error)
  575. return errors.New("error getting network with netid: " + gateway.NetID + " " + err.Error())
  576. }
  577. ipv4Net := network.AddressRange
  578. ipv6Net := network.AddressRange6
  579. for _, v := range gateway.Ranges {
  580. if ipv4Net != "" {
  581. if ContainsCIDR(ipv4Net, v) {
  582. slog.Error("egress range should not be the same as or contained in the netmaker network address", "error", v, ipv4Net)
  583. return errors.New("egress range should not be the same as or contained in the netmaker network address" + v + " " + ipv4Net)
  584. }
  585. }
  586. if ipv6Net != "" {
  587. if ContainsCIDR(ipv6Net, v) {
  588. slog.Error("egress range should not be the same as or contained in the netmaker network address", "error", v, ipv6Net)
  589. return errors.New("egress range should not be the same as or contained in the netmaker network address" + v + " " + ipv6Net)
  590. }
  591. }
  592. }
  593. return nil
  594. }
  595. func ContainsCIDR(net1, net2 string) bool {
  596. one, two := ipaddr.NewIPAddressString(net1),
  597. ipaddr.NewIPAddressString(net2)
  598. return one.Contains(two) || two.Contains(one)
  599. }
  600. // GetAllFailOvers - gets all the nodes that are failovers
  601. func GetAllFailOvers() ([]models.Node, error) {
  602. nodes, err := GetAllNodes()
  603. if err != nil {
  604. return nil, err
  605. }
  606. igs := make([]models.Node, 0)
  607. for _, node := range nodes {
  608. if node.IsFailOver {
  609. igs = append(igs, node)
  610. }
  611. }
  612. return igs, nil
  613. }