networks.go 18 KB

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