networks.go 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767
  1. package logic
  2. import (
  3. "encoding/binary"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "net"
  8. "os/exec"
  9. "strings"
  10. "time"
  11. "github.com/go-playground/validator/v10"
  12. "github.com/gravitl/netmaker/database"
  13. "github.com/gravitl/netmaker/logger"
  14. "github.com/gravitl/netmaker/models"
  15. "github.com/gravitl/netmaker/netclient/ncutils"
  16. "github.com/gravitl/netmaker/validation"
  17. )
  18. // GetNetworks - returns all networks from database
  19. func GetNetworks() ([]models.Network, error) {
  20. var networks []models.Network
  21. collection, err := database.FetchRecords(database.NETWORKS_TABLE_NAME)
  22. if err != nil {
  23. networks = make([]models.Network, 0)
  24. return networks, err
  25. }
  26. for _, value := range collection {
  27. var network models.Network
  28. if err := json.Unmarshal([]byte(value), &network); err != nil {
  29. return networks, err
  30. }
  31. // add network our array
  32. networks = append(networks, network)
  33. }
  34. return networks, err
  35. }
  36. // DeleteNetwork - deletes a network
  37. func DeleteNetwork(network string) error {
  38. nodeCount, err := GetNetworkNonServerNodeCount(network)
  39. if nodeCount == 0 || database.IsEmptyRecord(err) {
  40. // delete server nodes first then db records
  41. servers, err := GetSortedNetworkServerNodes(network)
  42. if err == nil {
  43. for _, s := range servers {
  44. if err = DeleteNodeByID(&s, true); err != nil {
  45. logger.Log(2, "could not removed server", s.Name, "before deleting network", network)
  46. } else {
  47. logger.Log(2, "removed server", s.Name, "before deleting network", network)
  48. }
  49. }
  50. } else {
  51. logger.Log(1, "could not remove servers before deleting network", network)
  52. }
  53. return database.DeleteRecord(database.NETWORKS_TABLE_NAME, network)
  54. }
  55. return errors.New("node check failed. All nodes must be deleted before deleting network")
  56. }
  57. // CreateNetwork - creates a network in database
  58. func CreateNetwork(network models.Network) error {
  59. network.SetDefaults()
  60. network.SetNodesLastModified()
  61. network.SetNetworkLastModified()
  62. network.KeyUpdateTimeStamp = time.Now().Unix()
  63. err := ValidateNetwork(&network, false)
  64. if err != nil {
  65. //returnErrorResponse(w, r, formatError(err, "badrequest"))
  66. return err
  67. }
  68. data, err := json.Marshal(&network)
  69. if err != nil {
  70. return err
  71. }
  72. if err = database.Insert(network.NetID, string(data), database.NETWORKS_TABLE_NAME); err != nil {
  73. return err
  74. }
  75. return err
  76. }
  77. // NetworkNodesUpdatePullChanges - tells nodes on network to pull
  78. func NetworkNodesUpdatePullChanges(networkName string) error {
  79. collections, err := database.FetchRecords(database.NODES_TABLE_NAME)
  80. if err != nil {
  81. if database.IsEmptyRecord(err) {
  82. return nil
  83. }
  84. return err
  85. }
  86. for _, value := range collections {
  87. var node models.Node
  88. err := json.Unmarshal([]byte(value), &node)
  89. if err != nil {
  90. fmt.Println("error in node address assignment!")
  91. return err
  92. }
  93. if node.Network == networkName {
  94. node.PullChanges = "yes"
  95. data, err := json.Marshal(&node)
  96. if err != nil {
  97. return err
  98. }
  99. database.Insert(node.ID, string(data), database.NODES_TABLE_NAME)
  100. }
  101. }
  102. return nil
  103. }
  104. // GetNetworkNonServerNodeCount - get number of network non server nodes
  105. func GetNetworkNonServerNodeCount(networkName string) (int, error) {
  106. collection, err := database.FetchRecords(database.NODES_TABLE_NAME)
  107. count := 0
  108. if err != nil && !database.IsEmptyRecord(err) {
  109. return count, err
  110. }
  111. for _, value := range collection {
  112. var node models.Node
  113. if err = json.Unmarshal([]byte(value), &node); err != nil {
  114. return count, err
  115. } else {
  116. if node.Network == networkName && node.IsServer != "yes" {
  117. count++
  118. }
  119. }
  120. }
  121. return count, nil
  122. }
  123. // GetParentNetwork - get parent network
  124. func GetParentNetwork(networkname string) (models.Network, error) {
  125. var network models.Network
  126. networkData, err := database.FetchRecord(database.NETWORKS_TABLE_NAME, networkname)
  127. if err != nil {
  128. return network, err
  129. }
  130. if err = json.Unmarshal([]byte(networkData), &network); err != nil {
  131. return models.Network{}, err
  132. }
  133. return network, nil
  134. }
  135. // GetParentNetwork - get parent network
  136. func GetNetworkSettings(networkname string) (models.Network, error) {
  137. var network models.Network
  138. networkData, err := database.FetchRecord(database.NETWORKS_TABLE_NAME, networkname)
  139. if err != nil {
  140. return network, err
  141. }
  142. if err = json.Unmarshal([]byte(networkData), &network); err != nil {
  143. return models.Network{}, err
  144. }
  145. network.AccessKeys = []models.AccessKey{}
  146. return network, nil
  147. }
  148. // UniqueAddress - see if address is unique
  149. func UniqueAddress(networkName string) (string, error) {
  150. var network models.Network
  151. network, err := GetParentNetwork(networkName)
  152. if err != nil {
  153. fmt.Println("UniqueAddress encountered an error")
  154. return "666", err
  155. }
  156. offset := true
  157. ip, ipnet, err := net.ParseCIDR(network.AddressRange)
  158. if err != nil {
  159. fmt.Println("UniqueAddress encountered an error")
  160. return "666", err
  161. }
  162. for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); Inc(ip) {
  163. if offset {
  164. offset = false
  165. continue
  166. }
  167. if networkName == "comms" {
  168. if IsIPUnique(networkName, ip.String(), database.INT_CLIENTS_TABLE_NAME, false) {
  169. return ip.String(), err
  170. }
  171. } else {
  172. if IsIPUnique(networkName, ip.String(), database.NODES_TABLE_NAME, false) && IsIPUnique(networkName, ip.String(), database.EXT_CLIENT_TABLE_NAME, false) {
  173. return ip.String(), err
  174. }
  175. }
  176. }
  177. //TODO
  178. err1 := errors.New("ERROR: No unique addresses available. Check network subnet.")
  179. return "W1R3: NO UNIQUE ADDRESSES AVAILABLE", err1
  180. }
  181. // UniqueAddressServer - get unique address starting from last available
  182. func UniqueAddressServer(networkName string) (string, error) {
  183. var network models.Network
  184. network, err := GetParentNetwork(networkName)
  185. if err != nil {
  186. logger.Log(0, "UniqueAddressServer encountered an error")
  187. return "666", err
  188. }
  189. _, ipv4Net, err := net.ParseCIDR(network.AddressRange)
  190. if err != nil {
  191. logger.Log(0, "UniqueAddressServer encountered an error")
  192. return "666", err
  193. }
  194. // convert IPNet struct mask and address to uint32
  195. // network is BigEndian
  196. mask := binary.BigEndian.Uint32(ipv4Net.Mask)
  197. start := binary.BigEndian.Uint32(ipv4Net.IP)
  198. // find the final address
  199. finish := (start & mask) | (mask ^ 0xffffffff)
  200. // loop through addresses as uint32
  201. for i := finish - 1; i > start; i-- {
  202. // convert back to net.IP
  203. ip := make(net.IP, 4)
  204. binary.BigEndian.PutUint32(ip, i)
  205. if IsIPUnique(networkName, ip.String(), database.NODES_TABLE_NAME, false) && IsIPUnique(networkName, ip.String(), database.EXT_CLIENT_TABLE_NAME, false) {
  206. return ip.String(), err
  207. }
  208. }
  209. return "W1R3: NO UNIQUE ADDRESSES AVAILABLE", fmt.Errorf("no unique server addresses found")
  210. }
  211. // IsIPUnique - checks if an IP is unique
  212. func IsIPUnique(network string, ip string, tableName string, isIpv6 bool) bool {
  213. isunique := true
  214. collection, err := database.FetchRecords(tableName)
  215. if err != nil {
  216. return isunique
  217. }
  218. for _, value := range collection { // filter
  219. var node models.Node
  220. if err = json.Unmarshal([]byte(value), &node); err != nil {
  221. continue
  222. }
  223. if isIpv6 {
  224. if node.Address6 == ip && node.Network == network {
  225. return false
  226. }
  227. } else {
  228. if node.Address == ip && node.Network == network {
  229. return false
  230. }
  231. }
  232. }
  233. return isunique
  234. }
  235. // UniqueAddress6 - see if ipv6 address is unique
  236. func UniqueAddress6(networkName string) (string, error) {
  237. var network models.Network
  238. network, err := GetParentNetwork(networkName)
  239. if err != nil {
  240. fmt.Println("Network Not Found")
  241. return "", err
  242. }
  243. if network.IsDualStack == "no" {
  244. return "", nil
  245. }
  246. offset := true
  247. ip, ipnet, err := net.ParseCIDR(network.AddressRange6)
  248. if err != nil {
  249. fmt.Println("UniqueAddress6 encountered an error")
  250. return "666", err
  251. }
  252. for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip); Inc(ip) {
  253. if offset {
  254. offset = false
  255. continue
  256. }
  257. if IsIPUnique(networkName, ip.String(), database.NODES_TABLE_NAME, true) {
  258. return ip.String(), err
  259. }
  260. }
  261. //TODO
  262. err1 := errors.New("ERROR: No unique addresses available. Check network subnet.")
  263. return "W1R3: NO UNIQUE ADDRESSES AVAILABLE", err1
  264. }
  265. // GetLocalIP - gets the local ip
  266. func GetLocalIP(node models.Node) string {
  267. var local string
  268. ifaces, err := net.Interfaces()
  269. if err != nil {
  270. return local
  271. }
  272. _, localrange, err := net.ParseCIDR(node.LocalRange)
  273. if err != nil {
  274. return local
  275. }
  276. found := false
  277. for _, i := range ifaces {
  278. if i.Flags&net.FlagUp == 0 {
  279. continue // interface down
  280. }
  281. if i.Flags&net.FlagLoopback != 0 {
  282. continue // loopback interface
  283. }
  284. addrs, err := i.Addrs()
  285. if err != nil {
  286. return local
  287. }
  288. for _, addr := range addrs {
  289. var ip net.IP
  290. switch v := addr.(type) {
  291. case *net.IPNet:
  292. if !found {
  293. ip = v.IP
  294. local = ip.String()
  295. if node.IsLocal == "yes" {
  296. found = localrange.Contains(ip)
  297. } else {
  298. found = true
  299. }
  300. }
  301. case *net.IPAddr:
  302. if !found {
  303. ip = v.IP
  304. local = ip.String()
  305. if node.IsLocal == "yes" {
  306. found = localrange.Contains(ip)
  307. } else {
  308. found = true
  309. }
  310. }
  311. }
  312. }
  313. }
  314. return local
  315. }
  316. // UpdateNetworkLocalAddresses - updates network localaddresses
  317. func UpdateNetworkLocalAddresses(networkName string) error {
  318. collection, err := database.FetchRecords(database.NODES_TABLE_NAME)
  319. if err != nil {
  320. return err
  321. }
  322. for _, value := range collection {
  323. var node models.Node
  324. err := json.Unmarshal([]byte(value), &node)
  325. if err != nil {
  326. fmt.Println("error in node address assignment!")
  327. return err
  328. }
  329. if node.Network == networkName {
  330. ipaddr, iperr := UniqueAddress(networkName)
  331. if iperr != nil {
  332. fmt.Println("error in node address assignment!")
  333. return iperr
  334. }
  335. node.Address = ipaddr
  336. newNodeData, err := json.Marshal(&node)
  337. if err != nil {
  338. logger.Log(1, "error in node address assignment!")
  339. return err
  340. }
  341. database.Insert(node.ID, string(newNodeData), database.NODES_TABLE_NAME)
  342. }
  343. }
  344. return nil
  345. }
  346. // UpdateNetworkLocalAddresses - updates network localaddresses
  347. func UpdateNetworkHolePunching(networkName string, holepunch string) error {
  348. nodes, err := GetNetworkNodes(networkName)
  349. if err != nil {
  350. return err
  351. }
  352. for _, node := range nodes {
  353. if node.IsServer != "yes" {
  354. node.UDPHolePunch = holepunch
  355. newNodeData, err := json.Marshal(&node)
  356. if err != nil {
  357. logger.Log(1, "error in node hole punch assignment")
  358. return err
  359. }
  360. database.Insert(node.ID, string(newNodeData), database.NODES_TABLE_NAME)
  361. }
  362. }
  363. return nil
  364. }
  365. // RemoveNetworkNodeIPv6Addresses - removes network node IPv6 addresses
  366. func RemoveNetworkNodeIPv6Addresses(networkName string) error {
  367. collections, err := database.FetchRecords(database.NODES_TABLE_NAME)
  368. if err != nil {
  369. return err
  370. }
  371. for _, value := range collections {
  372. var node models.Node
  373. err := json.Unmarshal([]byte(value), &node)
  374. if err != nil {
  375. fmt.Println("error in node address assignment!")
  376. return err
  377. }
  378. if node.Network == networkName {
  379. node.IsDualStack = "no"
  380. node.Address6 = ""
  381. node.PullChanges = "yes"
  382. data, err := json.Marshal(&node)
  383. if err != nil {
  384. return err
  385. }
  386. database.Insert(node.ID, string(data), database.NODES_TABLE_NAME)
  387. }
  388. }
  389. return nil
  390. }
  391. // UpdateNetworkNodeAddresses - updates network node addresses
  392. func UpdateNetworkNodeAddresses(networkName string) error {
  393. collections, err := database.FetchRecords(database.NODES_TABLE_NAME)
  394. if err != nil {
  395. return err
  396. }
  397. for _, value := range collections {
  398. var node models.Node
  399. err := json.Unmarshal([]byte(value), &node)
  400. if err != nil {
  401. fmt.Println("error in node address assignment!")
  402. return err
  403. }
  404. if node.Network == networkName {
  405. ipaddr, iperr := UniqueAddress(networkName)
  406. if iperr != nil {
  407. fmt.Println("error in node address assignment!")
  408. return iperr
  409. }
  410. node.Address = ipaddr
  411. node.PullChanges = "yes"
  412. data, err := json.Marshal(&node)
  413. if err != nil {
  414. return err
  415. }
  416. database.Insert(node.ID, string(data), database.NODES_TABLE_NAME)
  417. }
  418. }
  419. return nil
  420. }
  421. // IsNetworkDisplayNameUnique - checks if displayname is unique from other networks
  422. func IsNetworkDisplayNameUnique(network *models.Network) (bool, error) {
  423. isunique := true
  424. records, err := GetNetworks()
  425. if err != nil && !database.IsEmptyRecord(err) {
  426. return false, err
  427. }
  428. for i := 0; i < len(records); i++ {
  429. if network.NetID == records[i].DisplayName {
  430. isunique = false
  431. }
  432. }
  433. return isunique, nil
  434. }
  435. // IsNetworkNameUnique - checks to see if any other networks have the same name (id)
  436. func IsNetworkNameUnique(network *models.Network) (bool, error) {
  437. isunique := true
  438. dbs, err := GetNetworks()
  439. if err != nil && !database.IsEmptyRecord(err) {
  440. return false, err
  441. }
  442. for i := 0; i < len(dbs); i++ {
  443. if network.NetID == dbs[i].NetID {
  444. isunique = false
  445. }
  446. }
  447. return isunique, nil
  448. }
  449. // UpdateNetwork - updates a network with another network's fields
  450. func UpdateNetwork(currentNetwork *models.Network, newNetwork *models.Network) (bool, bool, bool, error) {
  451. if err := ValidateNetwork(newNetwork, true); err != nil {
  452. return false, false, false, err
  453. }
  454. if newNetwork.NetID == currentNetwork.NetID {
  455. hasrangeupdate := newNetwork.AddressRange != currentNetwork.AddressRange
  456. localrangeupdate := newNetwork.LocalRange != currentNetwork.LocalRange
  457. hasholepunchupdate := newNetwork.DefaultUDPHolePunch != currentNetwork.DefaultUDPHolePunch
  458. data, err := json.Marshal(newNetwork)
  459. if err != nil {
  460. return false, false, false, err
  461. }
  462. newNetwork.SetNetworkLastModified()
  463. err = database.Insert(newNetwork.NetID, string(data), database.NETWORKS_TABLE_NAME)
  464. return hasrangeupdate, localrangeupdate, hasholepunchupdate, err
  465. }
  466. // copy values
  467. return false, false, false, errors.New("failed to update network " + newNetwork.NetID + ", cannot change netid.")
  468. }
  469. // Inc - increments an IP
  470. func Inc(ip net.IP) {
  471. for j := len(ip) - 1; j >= 0; j-- {
  472. ip[j]++
  473. if ip[j] > 0 {
  474. break
  475. }
  476. }
  477. }
  478. // GetNetwork - gets a network from database
  479. func GetNetwork(networkname string) (models.Network, error) {
  480. var network models.Network
  481. networkData, err := database.FetchRecord(database.NETWORKS_TABLE_NAME, networkname)
  482. if err != nil {
  483. return network, err
  484. }
  485. if err = json.Unmarshal([]byte(networkData), &network); err != nil {
  486. return models.Network{}, err
  487. }
  488. return network, nil
  489. }
  490. // Network.NetIDInNetworkCharSet - checks if a netid of a network uses valid characters
  491. func NetIDInNetworkCharSet(network *models.Network) bool {
  492. charset := "abcdefghijklmnopqrstuvwxyz1234567890-_."
  493. for _, char := range network.NetID {
  494. if !strings.Contains(charset, strings.ToLower(string(char))) {
  495. return false
  496. }
  497. }
  498. return true
  499. }
  500. // Network.Validate - validates fields of an network struct
  501. func ValidateNetwork(network *models.Network, isUpdate bool) error {
  502. v := validator.New()
  503. _ = v.RegisterValidation("netid_valid", func(fl validator.FieldLevel) bool {
  504. inCharSet := NetIDInNetworkCharSet(network)
  505. if isUpdate {
  506. return inCharSet
  507. }
  508. isFieldUnique, _ := IsNetworkNameUnique(network)
  509. return isFieldUnique && inCharSet
  510. })
  511. //
  512. _ = v.RegisterValidation("displayname_valid", func(fl validator.FieldLevel) bool {
  513. isFieldUnique, _ := IsNetworkDisplayNameUnique(network)
  514. inCharSet := network.DisplayNameInNetworkCharSet()
  515. if isUpdate {
  516. return inCharSet
  517. }
  518. return isFieldUnique && inCharSet
  519. })
  520. _ = v.RegisterValidation("checkyesorno", func(fl validator.FieldLevel) bool {
  521. return validation.CheckYesOrNo(fl)
  522. })
  523. err := v.Struct(network)
  524. if err != nil {
  525. for _, e := range err.(validator.ValidationErrors) {
  526. fmt.Println(e)
  527. }
  528. }
  529. return err
  530. }
  531. // ParseNetwork - parses a network into a model
  532. func ParseNetwork(value string) (models.Network, error) {
  533. var network models.Network
  534. err := json.Unmarshal([]byte(value), &network)
  535. return network, err
  536. }
  537. // ValidateNetworkUpdate - checks if network is valid to update
  538. func ValidateNetworkUpdate(network models.Network) error {
  539. v := validator.New()
  540. _ = v.RegisterValidation("netid_valid", func(fl validator.FieldLevel) bool {
  541. if fl.Field().String() == "" {
  542. return true
  543. }
  544. inCharSet := nameInNetworkCharSet(fl.Field().String())
  545. return inCharSet
  546. })
  547. err := v.Struct(network)
  548. if err != nil {
  549. for _, e := range err.(validator.ValidationErrors) {
  550. logger.Log(1, "validator", e.Error())
  551. }
  552. }
  553. return err
  554. }
  555. // KeyUpdate - updates keys on network
  556. func KeyUpdate(netname string) (models.Network, error) {
  557. err := networkNodesUpdateAction(netname, models.NODE_UPDATE_KEY)
  558. if err != nil {
  559. return models.Network{}, err
  560. }
  561. return models.Network{}, nil
  562. }
  563. //SaveNetwork - save network struct to database
  564. func SaveNetwork(network *models.Network) error {
  565. data, err := json.Marshal(network)
  566. if err != nil {
  567. return err
  568. }
  569. if err := database.Insert(network.NetID, string(data), database.NETWORKS_TABLE_NAME); err != nil {
  570. return err
  571. }
  572. return nil
  573. }
  574. // == Private ==
  575. func networkNodesUpdateAction(networkName string, action string) error {
  576. collections, err := database.FetchRecords(database.NODES_TABLE_NAME)
  577. if err != nil {
  578. if database.IsEmptyRecord(err) {
  579. return nil
  580. }
  581. return err
  582. }
  583. for _, value := range collections {
  584. var node models.Node
  585. err := json.Unmarshal([]byte(value), &node)
  586. if err != nil {
  587. fmt.Println("error in node address assignment!")
  588. return err
  589. }
  590. if action == models.NODE_UPDATE_KEY && node.IsStatic == "yes" {
  591. continue
  592. }
  593. if node.Network == networkName {
  594. node.Action = action
  595. data, err := json.Marshal(&node)
  596. if err != nil {
  597. return err
  598. }
  599. database.Insert(node.ID, string(data), database.NODES_TABLE_NAME)
  600. }
  601. }
  602. return nil
  603. }
  604. func nameInNetworkCharSet(name string) bool {
  605. charset := "abcdefghijklmnopqrstuvwxyz1234567890-_."
  606. for _, char := range name {
  607. if !strings.Contains(charset, strings.ToLower(string(char))) {
  608. return false
  609. }
  610. }
  611. return true
  612. }
  613. func deleteInterface(ifacename string, postdown string) error {
  614. var err error
  615. if !ncutils.IsKernel() {
  616. err = RemoveConf(ifacename, true)
  617. } else {
  618. ipExec, errN := exec.LookPath("ip")
  619. err = errN
  620. if err != nil {
  621. logger.Log(1, err.Error())
  622. }
  623. _, err = ncutils.RunCmd(ipExec+" link del "+ifacename, false)
  624. if postdown != "" {
  625. runcmds := strings.Split(postdown, "; ")
  626. err = ncutils.RunCmds(runcmds, false)
  627. }
  628. }
  629. return err
  630. }
  631. func isInterfacePresent(iface string, address string) (string, bool) {
  632. var interfaces []net.Interface
  633. var err error
  634. interfaces, err = net.Interfaces()
  635. if err != nil {
  636. logger.Log(0, "ERROR: could not read interfaces")
  637. return "", true
  638. }
  639. for _, currIface := range interfaces {
  640. var currAddrs []net.Addr
  641. currAddrs, err = currIface.Addrs()
  642. if err != nil || len(currAddrs) == 0 {
  643. continue
  644. }
  645. for _, addr := range currAddrs {
  646. if strings.Contains(addr.String(), address) && currIface.Name != iface {
  647. logger.Log(2, "found iface", addr.String(), currIface.Name)
  648. interfaces = nil
  649. currAddrs = nil
  650. return currIface.Name, false
  651. }
  652. }
  653. currAddrs = nil
  654. }
  655. interfaces = nil
  656. logger.Log(2, "failed to find iface", iface)
  657. return "", true
  658. }