2021-01-08 13:41:50 +01:00
|
|
|
From ac34a8d839f91285f4ced605250422a1ecf5cb55 Mon Sep 17 00:00:00 2001
|
2019-09-06 14:06:25 +02:00
|
|
|
From: EricS <54029547+ESiebigteroth@users.noreply.github.com>
|
|
|
|
Date: Tue, 3 Sep 2019 11:22:53 +0200
|
|
|
|
Subject: [PATCH] Implement network.fqdns module function (bsc#1134860)
|
|
|
|
(#172)
|
|
|
|
|
|
|
|
* Duplicate fqdns logic in module.network
|
|
|
|
* Move _get_interfaces to utils.network
|
|
|
|
* Reuse network.fqdns in grains.core.fqdns
|
|
|
|
* Return empty list when fqdns grains is disabled
|
|
|
|
|
|
|
|
Co-authored-by: Eric Siebigteroth <eric.siebigteroth@suse.de>
|
|
|
|
---
|
2021-01-08 13:41:50 +01:00
|
|
|
salt/grains/core.py | 58 +++-------------------------------
|
|
|
|
salt/modules/network.py | 12 +++----
|
|
|
|
salt/utils/network.py | 2 +-
|
|
|
|
tests/unit/grains/test_core.py | 55 ++++++++++++--------------------
|
|
|
|
4 files changed, 31 insertions(+), 96 deletions(-)
|
2019-09-06 14:06:25 +02:00
|
|
|
|
|
|
|
diff --git a/salt/grains/core.py b/salt/grains/core.py
|
2021-01-08 13:41:50 +01:00
|
|
|
index 5f18ba4a58..0dc1d97f97 100644
|
2019-09-06 14:06:25 +02:00
|
|
|
--- a/salt/grains/core.py
|
|
|
|
+++ b/salt/grains/core.py
|
2021-01-08 13:41:50 +01:00
|
|
|
@@ -23,7 +23,6 @@ import uuid
|
2019-09-06 14:06:25 +02:00
|
|
|
import warnings
|
2021-01-08 13:41:50 +01:00
|
|
|
import zlib
|
|
|
|
from errno import EACCES, EPERM
|
2019-09-06 14:06:25 +02:00
|
|
|
-from multiprocessing.pool import ThreadPool
|
|
|
|
|
2021-01-08 13:41:50 +01:00
|
|
|
import distro
|
|
|
|
import salt.exceptions
|
|
|
|
@@ -2406,59 +2405,10 @@ def fqdns():
|
2019-09-06 14:06:25 +02:00
|
|
|
then trying to reverse resolve them (excluding 'lo' interface).
|
2021-01-08 13:41:50 +01:00
|
|
|
To disable the fqdns grain, set enable_fqdns_grains: False in the minion configuration file.
|
|
|
|
"""
|
2019-09-06 14:06:25 +02:00
|
|
|
- # Provides:
|
|
|
|
- # fqdns
|
|
|
|
-
|
|
|
|
- grains = {}
|
|
|
|
- fqdns = set()
|
|
|
|
-
|
|
|
|
- def _lookup_fqdn(ip):
|
|
|
|
- try:
|
|
|
|
- name, aliaslist, addresslist = socket.gethostbyaddr(ip)
|
2021-01-08 13:41:50 +01:00
|
|
|
- return [socket.getfqdn(name)] + [
|
|
|
|
- als for als in aliaslist if salt.utils.network.is_fqdn(als)
|
|
|
|
- ]
|
2019-09-06 14:06:25 +02:00
|
|
|
- except socket.herror as err:
|
|
|
|
- if err.errno in (0, HOST_NOT_FOUND, NO_DATA):
|
|
|
|
- # No FQDN for this IP address, so we don't need to know this all the time.
|
|
|
|
- log.debug("Unable to resolve address %s: %s", ip, err)
|
|
|
|
- else:
|
2019-11-28 16:41:55 +01:00
|
|
|
- log.error(err_message, ip, err)
|
2021-01-08 13:41:50 +01:00
|
|
|
- except (OSError, socket.gaierror, socket.timeout) as err:
|
2019-11-28 16:41:55 +01:00
|
|
|
- log.error(err_message, ip, err)
|
2019-09-06 14:06:25 +02:00
|
|
|
-
|
|
|
|
- start = time.time()
|
|
|
|
-
|
2021-01-08 13:41:50 +01:00
|
|
|
- addresses = salt.utils.network.ip_addrs(
|
|
|
|
- include_loopback=False, interface_data=_get_interfaces()
|
|
|
|
- )
|
|
|
|
- addresses.extend(
|
|
|
|
- salt.utils.network.ip_addrs6(
|
|
|
|
- include_loopback=False, interface_data=_get_interfaces()
|
|
|
|
- )
|
|
|
|
- )
|
|
|
|
- err_message = "Exception during resolving address: %s"
|
2019-09-06 14:06:25 +02:00
|
|
|
-
|
|
|
|
- # Create a ThreadPool to process the underlying calls to 'socket.gethostbyaddr' in parallel.
|
|
|
|
- # This avoid blocking the execution when the "fqdn" is not defined for certains IP addresses, which was causing
|
|
|
|
- # that "socket.timeout" was reached multiple times secuencially, blocking execution for several seconds.
|
|
|
|
-
|
|
|
|
- try:
|
2021-01-08 13:41:50 +01:00
|
|
|
- pool = ThreadPool(8)
|
|
|
|
- results = pool.map(_lookup_fqdn, addresses)
|
|
|
|
- pool.close()
|
|
|
|
- pool.join()
|
2019-09-06 14:06:25 +02:00
|
|
|
- except Exception as exc:
|
2021-01-08 13:41:50 +01:00
|
|
|
- log.error("Exception while creating a ThreadPool for resolving FQDNs: %s", exc)
|
2019-09-06 14:06:25 +02:00
|
|
|
-
|
|
|
|
- for item in results:
|
|
|
|
- if item:
|
|
|
|
- fqdns.update(item)
|
|
|
|
-
|
|
|
|
- elapsed = time.time() - start
|
2021-01-08 13:41:50 +01:00
|
|
|
- log.debug("Elapsed time getting FQDNs: {} seconds".format(elapsed))
|
2019-09-06 14:06:25 +02:00
|
|
|
-
|
|
|
|
- return {"fqdns": sorted(list(fqdns))}
|
|
|
|
+ opt = {"fqdns": []}
|
2021-01-08 13:41:50 +01:00
|
|
|
+ if __opts__.get("enable_fqdns_grains", True) == True:
|
|
|
|
+ opt = __salt__["network.fqdns"]()
|
2019-09-06 14:06:25 +02:00
|
|
|
+ return opt
|
|
|
|
|
|
|
|
|
|
|
|
def ip_fqdn():
|
|
|
|
diff --git a/salt/modules/network.py b/salt/modules/network.py
|
2021-01-08 13:41:50 +01:00
|
|
|
index 2e1410c288..59ed43bba6 100644
|
2019-09-06 14:06:25 +02:00
|
|
|
--- a/salt/modules/network.py
|
|
|
|
+++ b/salt/modules/network.py
|
2021-01-08 13:41:50 +01:00
|
|
|
@@ -2,7 +2,6 @@
|
|
|
|
Module for gathering and managing network information
|
|
|
|
"""
|
|
|
|
|
|
|
|
-# Import python libs
|
|
|
|
import datetime
|
|
|
|
import hashlib
|
|
|
|
import logging
|
|
|
|
@@ -12,7 +11,6 @@ import socket
|
|
|
|
import time
|
|
|
|
from multiprocessing.pool import ThreadPool
|
2019-09-06 14:06:25 +02:00
|
|
|
|
2021-01-08 13:41:50 +01:00
|
|
|
-# Import salt libs
|
2019-09-06 14:06:25 +02:00
|
|
|
import salt.utils.decorators.path
|
2021-01-08 13:41:50 +01:00
|
|
|
import salt.utils.functools
|
|
|
|
import salt.utils.network
|
|
|
|
@@ -20,8 +18,6 @@ import salt.utils.platform
|
|
|
|
import salt.utils.validate.net
|
|
|
|
from salt._compat import ipaddress
|
|
|
|
from salt.exceptions import CommandExecutionError
|
|
|
|
-
|
|
|
|
-# Import 3rd-party libs
|
|
|
|
from salt.ext.six.moves import range
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
@@ -2076,7 +2072,10 @@ def fqdns():
|
|
|
|
|
|
|
|
def _lookup_fqdn(ip):
|
|
|
|
try:
|
|
|
|
- return [socket.getfqdn(socket.gethostbyaddr(ip)[0])]
|
2019-09-06 14:06:25 +02:00
|
|
|
+ name, aliaslist, addresslist = socket.gethostbyaddr(ip)
|
2021-01-08 13:41:50 +01:00
|
|
|
+ return [socket.getfqdn(name)] + [
|
|
|
|
+ als for als in aliaslist if salt.utils.network.is_fqdn(als)
|
|
|
|
+ ]
|
|
|
|
except socket.herror as err:
|
|
|
|
if err.errno in (0, HOST_NOT_FOUND, NO_DATA):
|
|
|
|
# No FQDN for this IP address, so we don't need to know this all the time.
|
|
|
|
@@ -2102,13 +2101,12 @@ def fqdns():
|
|
|
|
# This avoid blocking the execution when the "fqdn" is not defined for certains IP addresses, which was causing
|
|
|
|
# that "socket.timeout" was reached multiple times secuencially, blocking execution for several seconds.
|
|
|
|
|
|
|
|
- results = []
|
|
|
|
try:
|
|
|
|
pool = ThreadPool(8)
|
|
|
|
results = pool.map(_lookup_fqdn, addresses)
|
|
|
|
pool.close()
|
|
|
|
pool.join()
|
|
|
|
- except Exception as exc: # pylint: disable=broad-except
|
2019-09-06 14:06:25 +02:00
|
|
|
+ except Exception as exc:
|
2021-01-08 13:41:50 +01:00
|
|
|
log.error("Exception while creating a ThreadPool for resolving FQDNs: %s", exc)
|
|
|
|
|
|
|
|
for item in results:
|
2019-09-06 14:06:25 +02:00
|
|
|
diff --git a/salt/utils/network.py b/salt/utils/network.py
|
2021-01-08 13:41:50 +01:00
|
|
|
index d253ded3ab..25b2d06758 100644
|
2019-09-06 14:06:25 +02:00
|
|
|
--- a/salt/utils/network.py
|
|
|
|
+++ b/salt/utils/network.py
|
2021-01-08 13:41:50 +01:00
|
|
|
@@ -49,7 +49,7 @@ except (ImportError, OSError, AttributeError, TypeError):
|
|
|
|
_INTERFACES = {}
|
2019-09-06 14:06:25 +02:00
|
|
|
|
|
|
|
|
2021-01-08 13:41:50 +01:00
|
|
|
-def _get_interfaces():
|
|
|
|
+def _get_interfaces(): #! function
|
|
|
|
"""
|
|
|
|
Provide a dict of the connected interfaces and their ip addresses
|
|
|
|
"""
|
2019-09-06 14:06:25 +02:00
|
|
|
diff --git a/tests/unit/grains/test_core.py b/tests/unit/grains/test_core.py
|
2021-01-08 13:41:50 +01:00
|
|
|
index d760e57a54..a5ceeb8317 100644
|
2019-09-06 14:06:25 +02:00
|
|
|
--- a/tests/unit/grains/test_core.py
|
|
|
|
+++ b/tests/unit/grains/test_core.py
|
2021-01-08 13:41:50 +01:00
|
|
|
@@ -18,6 +18,7 @@ import salt.utils.network
|
|
|
|
import salt.utils.path
|
|
|
|
import salt.utils.platform
|
|
|
|
from salt._compat import ipaddress
|
|
|
|
+from salt.ext import six
|
|
|
|
from tests.support.mixins import LoaderModuleMockMixin
|
|
|
|
from tests.support.mock import MagicMock, Mock, mock_open, patch
|
|
|
|
from tests.support.unit import TestCase, skipIf
|
|
|
|
@@ -1293,14 +1294,14 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
|
|
|
):
|
2019-09-06 14:06:25 +02:00
|
|
|
assert core.dns() == ret
|
|
|
|
|
2021-01-08 13:41:50 +01:00
|
|
|
- def test_enable_fqdns_false(self):
|
2019-09-06 14:06:25 +02:00
|
|
|
+ def test_enablefqdnsFalse(self):
|
2021-01-08 13:41:50 +01:00
|
|
|
"""
|
|
|
|
tests enable_fqdns_grains is set to False
|
|
|
|
"""
|
|
|
|
with patch.dict("salt.grains.core.__opts__", {"enable_fqdns_grains": False}):
|
|
|
|
assert core.fqdns() == {"fqdns": []}
|
|
|
|
|
|
|
|
- def test_enable_fqdns_true(self):
|
2019-09-06 14:06:25 +02:00
|
|
|
+ def test_enablefqdnsTrue(self):
|
2021-01-08 13:41:50 +01:00
|
|
|
"""
|
|
|
|
testing that grains uses network.fqdns module
|
|
|
|
"""
|
|
|
|
@@ -1311,14 +1312,14 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
|
|
|
with patch.dict("salt.grains.core.__opts__", {"enable_fqdns_grains": True}):
|
|
|
|
assert core.fqdns() == "my.fake.domain"
|
|
|
|
|
|
|
|
- def test_enable_fqdns_none(self):
|
2019-09-06 14:06:25 +02:00
|
|
|
+ def test_enablefqdnsNone(self):
|
2021-01-08 13:41:50 +01:00
|
|
|
"""
|
|
|
|
testing default fqdns grains is returned when enable_fqdns_grains is None
|
|
|
|
"""
|
|
|
|
with patch.dict("salt.grains.core.__opts__", {"enable_fqdns_grains": None}):
|
|
|
|
assert core.fqdns() == {"fqdns": []}
|
|
|
|
|
|
|
|
- def test_enable_fqdns_without_patching(self):
|
2019-09-06 14:06:25 +02:00
|
|
|
+ def test_enablefqdnswithoutpaching(self):
|
2021-01-08 13:41:50 +01:00
|
|
|
"""
|
|
|
|
testing fqdns grains is enabled by default
|
|
|
|
"""
|
|
|
|
@@ -1326,23 +1327,7 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
|
|
|
"salt.grains.core.__salt__",
|
|
|
|
{"network.fqdns": MagicMock(return_value="my.fake.domain")},
|
|
|
|
):
|
|
|
|
- # fqdns is disabled by default on Windows
|
|
|
|
- if salt.utils.platform.is_windows():
|
|
|
|
- assert core.fqdns() == {"fqdns": []}
|
|
|
|
- else:
|
|
|
|
- assert core.fqdns() == "my.fake.domain"
|
|
|
|
-
|
|
|
|
- def test_enable_fqdns_false_is_proxy(self):
|
|
|
|
- """
|
|
|
|
- testing fqdns grains is disabled by default for proxy minions
|
|
|
|
- """
|
|
|
|
- with patch("salt.utils.platform.is_proxy", return_value=True, autospec=True):
|
|
|
|
- with patch.dict(
|
|
|
|
- "salt.grains.core.__salt__",
|
|
|
|
- {"network.fqdns": MagicMock(return_value="my.fake.domain")},
|
|
|
|
- ):
|
|
|
|
- # fqdns is disabled by default on proxy minions
|
|
|
|
- assert core.fqdns() == {"fqdns": []}
|
|
|
|
+ assert core.fqdns() == "my.fake.domain"
|
|
|
|
|
|
|
|
@skipIf(not salt.utils.platform.is_linux(), "System is not Linux")
|
|
|
|
@patch(
|
|
|
|
@@ -1367,11 +1352,12 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
|
|
|
("bluesniff.foo.bar", [], ["fe80::a8b2:93ff:dead:beef"]),
|
|
|
|
]
|
|
|
|
ret = {"fqdns": ["bluesniff.foo.bar", "foo.bar.baz", "rinzler.evil-corp.com"]}
|
|
|
|
- with patch.object(socket, "gethostbyaddr", side_effect=reverse_resolv_mock):
|
2019-09-06 14:06:25 +02:00
|
|
|
- fqdns = core.fqdns()
|
|
|
|
- assert "fqdns" in fqdns
|
2021-01-08 13:41:50 +01:00
|
|
|
- assert len(fqdns["fqdns"]) == len(ret["fqdns"])
|
|
|
|
- assert set(fqdns["fqdns"]) == set(ret["fqdns"])
|
|
|
|
+ with patch.dict(core.__salt__, {"network.fqdns": salt.modules.network.fqdns}):
|
|
|
|
+ with patch.object(socket, "gethostbyaddr", side_effect=reverse_resolv_mock):
|
2019-09-06 14:06:25 +02:00
|
|
|
+ fqdns = core.fqdns()
|
|
|
|
+ assert "fqdns" in fqdns
|
2021-01-08 13:41:50 +01:00
|
|
|
+ assert len(fqdns["fqdns"]) == len(ret["fqdns"])
|
|
|
|
+ assert set(fqdns["fqdns"]) == set(ret["fqdns"])
|
2019-09-06 14:06:25 +02:00
|
|
|
|
2021-01-08 13:41:50 +01:00
|
|
|
@skipIf(not salt.utils.platform.is_linux(), "System is not Linux")
|
|
|
|
@patch("salt.utils.network.ip_addrs", MagicMock(return_value=["1.2.3.4"]))
|
|
|
|
@@ -1437,14 +1423,15 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
|
|
|
["fe80::a8b2:93ff:dead:beef"],
|
|
|
|
),
|
|
|
|
]
|
|
|
|
- with patch.object(socket, "gethostbyaddr", side_effect=reverse_resolv_mock):
|
2019-09-06 14:06:25 +02:00
|
|
|
- fqdns = core.fqdns()
|
|
|
|
- assert "fqdns" in fqdns
|
|
|
|
- for alias in ["this.is.valid.alias", "alias.bluesniff.foo.bar"]:
|
|
|
|
- assert alias in fqdns["fqdns"]
|
|
|
|
-
|
|
|
|
- for alias in ["throwmeaway", "false-hostname", "badaliass"]:
|
|
|
|
- assert alias not in fqdns["fqdns"]
|
2021-01-08 13:41:50 +01:00
|
|
|
+ with patch.dict(core.__salt__, {"network.fqdns": salt.modules.network.fqdns}):
|
|
|
|
+ with patch.object(socket, "gethostbyaddr", side_effect=reverse_resolv_mock):
|
2019-09-06 14:06:25 +02:00
|
|
|
+ fqdns = core.fqdns()
|
|
|
|
+ assert "fqdns" in fqdns
|
|
|
|
+ for alias in ["this.is.valid.alias", "alias.bluesniff.foo.bar"]:
|
|
|
|
+ assert alias in fqdns["fqdns"]
|
|
|
|
+
|
|
|
|
+ for alias in ["throwmeaway", "false-hostname", "badaliass"]:
|
|
|
|
+ assert alias not in fqdns["fqdns"]
|
2020-04-07 14:14:01 +02:00
|
|
|
|
2019-09-06 14:06:25 +02:00
|
|
|
def test_core_virtual(self):
|
2021-01-08 13:41:50 +01:00
|
|
|
"""
|
2019-09-06 14:06:25 +02:00
|
|
|
--
|
2021-01-08 13:41:50 +01:00
|
|
|
2.29.2
|
2019-09-06 14:06:25 +02:00
|
|
|
|
|
|
|
|