Accepting request 728977 from systemsmanagement:saltstack
- Require shadow instead of old pwdutils (bsc#1130588) - Conflict with tornado >= 5; for now we can only cope with Tornado 4.x (boo#1101780). - Fix virt.full_info (bsc#1146382) - virt.volume_infos: silence libvirt error message - virt.volume_infos needs to ignore inactive pools - Fix for various bugs in virt network and pool states - Implement network.fqdns module function (bsc#1134860) - Added: * 2019.2.0-pr-54196-backport-173.patch * virt.volume_infos-silence-libvirt-error-message-175.patch * fix-virt.full_info-176.patch * implement-network.fqdns-module-function-bsc-1134860-.patch * virt.volume_infos-needs-to-ignore-inactive-pools-174.patch - Restore default behaviour of pkg list return (bsc#1148714) - Strip trailing "/" from repo.uri when comparing repos in apktpkg.mod_repo (bsc#1146192) - Added: * strip-trailing-from-repo.uri-when-comparing-repos-in.patch * restore-default-behaviour-of-pkg-list-return.patch - Use python3 to build package Salt for RHEL8 - Make python3 default for RHEL8 - Fix aptpkg systemd call (bsc#1143301) - Added: * fix-aptpkg-systemd-call-bsc-1143301.patch - Move server_id deprecation warning to reduce log spamming (bsc#1135567) (bsc#1135732) OBS-URL: https://build.opensuse.org/request/show/728977 OBS-URL: https://build.opensuse.org/package/show/openSUSE:Factory/salt?expand=0&rev=92
This commit is contained in:
commit
a5ad683fc1
224
2019.2.0-pr-54196-backport-173.patch
Normal file
224
2019.2.0-pr-54196-backport-173.patch
Normal file
@ -0,0 +1,224 @@
|
|||||||
|
From 3119bc27584472b0f0d440a37ec4cff2504165f2 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Cedric Bosdonnat <cbosdonnat@suse.com>
|
||||||
|
Date: Tue, 3 Sep 2019 15:16:30 +0200
|
||||||
|
Subject: [PATCH] 2019.2.0 PR 54196 backport (#173)
|
||||||
|
|
||||||
|
* virt.network_define doesn't have vport as positional argument
|
||||||
|
|
||||||
|
virt.network_running state calls virt.network_define with vport as a
|
||||||
|
positional argument resulting in an error at runtime. Fix the state to
|
||||||
|
use the vport named argument instead.
|
||||||
|
|
||||||
|
* Fix virt.pool_running state documentation
|
||||||
|
|
||||||
|
virt.pool_running needs the source to be a dictionary, which the
|
||||||
|
documentation was not reflecting. Along the same lines the source hosts
|
||||||
|
need to be a list, adjust the example to show it.
|
||||||
|
|
||||||
|
* Get virt.pool_running to start the pool after creating it
|
||||||
|
|
||||||
|
Commit 25b96815 is wrong in assuming the pool build also starts it. The
|
||||||
|
pool needs to be stopped before building it, but we still need to start
|
||||||
|
it after the build: libvirt won't do it automagically for us.
|
||||||
|
|
||||||
|
* Fix states to match virt.{network,pool}_infos return
|
||||||
|
|
||||||
|
virt.network_infos and virt.pool_infos return the infos as a dictionary
|
||||||
|
with the network or pool name as a key even when there is only one
|
||||||
|
value. Adapt the network_running and pool_running states to this.
|
||||||
|
|
||||||
|
* Fix virt.running use of virt.vm_state
|
||||||
|
|
||||||
|
vm_state return a dictionary with the VM name as a key. Fix virt.running
|
||||||
|
state and its tests to match this. See issue #53107.
|
||||||
|
---
|
||||||
|
salt/states/virt.py | 26 ++++++++++++++++----------
|
||||||
|
tests/unit/states/test_virt.py | 27 +++++++++++++++------------
|
||||||
|
2 files changed, 31 insertions(+), 22 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/salt/states/virt.py b/salt/states/virt.py
|
||||||
|
index d411f864cd..32a9e31ae5 100644
|
||||||
|
--- a/salt/states/virt.py
|
||||||
|
+++ b/salt/states/virt.py
|
||||||
|
@@ -389,8 +389,8 @@ def running(name,
|
||||||
|
|
||||||
|
try:
|
||||||
|
try:
|
||||||
|
- __salt__['virt.vm_state'](name)
|
||||||
|
- if __salt__['virt.vm_state'](name) != 'running':
|
||||||
|
+ domain_state = __salt__['virt.vm_state'](name)
|
||||||
|
+ if domain_state.get(name, None) != 'running':
|
||||||
|
action_msg = 'started'
|
||||||
|
if update:
|
||||||
|
status = __salt__['virt.update'](name,
|
||||||
|
@@ -670,7 +670,7 @@ def network_running(name,
|
||||||
|
try:
|
||||||
|
info = __salt__['virt.network_info'](name, connection=connection, username=username, password=password)
|
||||||
|
if info:
|
||||||
|
- if info['active']:
|
||||||
|
+ if info[name]['active']:
|
||||||
|
ret['comment'] = 'Network {0} exists and is running'.format(name)
|
||||||
|
else:
|
||||||
|
__salt__['virt.network_start'](name, connection=connection, username=username, password=password)
|
||||||
|
@@ -680,7 +680,7 @@ def network_running(name,
|
||||||
|
__salt__['virt.network_define'](name,
|
||||||
|
bridge,
|
||||||
|
forward,
|
||||||
|
- vport,
|
||||||
|
+ vport=vport,
|
||||||
|
tag=tag,
|
||||||
|
autostart=autostart,
|
||||||
|
start=True,
|
||||||
|
@@ -744,11 +744,11 @@ def pool_running(name,
|
||||||
|
- owner: 1000
|
||||||
|
- group: 100
|
||||||
|
- source:
|
||||||
|
- - dir: samba_share
|
||||||
|
- - hosts:
|
||||||
|
- one.example.com
|
||||||
|
- two.example.com
|
||||||
|
- - format: cifs
|
||||||
|
+ dir: samba_share
|
||||||
|
+ hosts:
|
||||||
|
+ - one.example.com
|
||||||
|
+ - two.example.com
|
||||||
|
+ format: cifs
|
||||||
|
- autostart: True
|
||||||
|
|
||||||
|
'''
|
||||||
|
@@ -761,7 +761,7 @@ def pool_running(name,
|
||||||
|
try:
|
||||||
|
info = __salt__['virt.pool_info'](name, connection=connection, username=username, password=password)
|
||||||
|
if info:
|
||||||
|
- if info['state'] == 'running':
|
||||||
|
+ if info[name]['state'] == 'running':
|
||||||
|
ret['comment'] = 'Pool {0} exists and is running'.format(name)
|
||||||
|
else:
|
||||||
|
__salt__['virt.pool_start'](name, connection=connection, username=username, password=password)
|
||||||
|
@@ -795,6 +795,12 @@ def pool_running(name,
|
||||||
|
connection=connection,
|
||||||
|
username=username,
|
||||||
|
password=password)
|
||||||
|
+
|
||||||
|
+ __salt__['virt.pool_start'](name,
|
||||||
|
+ connection=connection,
|
||||||
|
+ username=username,
|
||||||
|
+ password=password)
|
||||||
|
+
|
||||||
|
ret['changes'][name] = 'Pool defined and started'
|
||||||
|
ret['comment'] = 'Pool {0} defined and started'.format(name)
|
||||||
|
except libvirt.libvirtError as err:
|
||||||
|
diff --git a/tests/unit/states/test_virt.py b/tests/unit/states/test_virt.py
|
||||||
|
index 8022989937..2904fa224d 100644
|
||||||
|
--- a/tests/unit/states/test_virt.py
|
||||||
|
+++ b/tests/unit/states/test_virt.py
|
||||||
|
@@ -229,7 +229,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
'result': True,
|
||||||
|
'comment': 'myvm is running'}
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.vm_state': MagicMock(return_value='stopped'),
|
||||||
|
+ 'virt.vm_state': MagicMock(return_value={'myvm': 'stopped'}),
|
||||||
|
'virt.start': MagicMock(return_value=0),
|
||||||
|
}):
|
||||||
|
ret.update({'changes': {'myvm': 'Domain started'},
|
||||||
|
@@ -322,7 +322,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
password='supersecret')
|
||||||
|
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.vm_state': MagicMock(return_value='stopped'),
|
||||||
|
+ 'virt.vm_state': MagicMock(return_value={'myvm': 'stopped'}),
|
||||||
|
'virt.start': MagicMock(side_effect=[self.mock_libvirt.libvirtError('libvirt error msg')])
|
||||||
|
}):
|
||||||
|
ret.update({'changes': {}, 'result': False, 'comment': 'libvirt error msg'})
|
||||||
|
@@ -330,7 +330,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
|
||||||
|
# Working update case when running
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.vm_state': MagicMock(return_value='running'),
|
||||||
|
+ 'virt.vm_state': MagicMock(return_value={'myvm': 'running'}),
|
||||||
|
'virt.update': MagicMock(return_value={'definition': True, 'cpu': True})
|
||||||
|
}):
|
||||||
|
ret.update({'changes': {'myvm': {'definition': True, 'cpu': True}},
|
||||||
|
@@ -340,7 +340,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
|
||||||
|
# Working update case when stopped
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.vm_state': MagicMock(return_value='stopped'),
|
||||||
|
+ 'virt.vm_state': MagicMock(return_value={'myvm': 'stopped'}),
|
||||||
|
'virt.start': MagicMock(return_value=0),
|
||||||
|
'virt.update': MagicMock(return_value={'definition': True})
|
||||||
|
}):
|
||||||
|
@@ -351,7 +351,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
|
||||||
|
# Failed live update case
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.vm_state': MagicMock(return_value='running'),
|
||||||
|
+ 'virt.vm_state': MagicMock(return_value={'myvm': 'running'}),
|
||||||
|
'virt.update': MagicMock(return_value={'definition': True, 'cpu': False, 'errors': ['some error']})
|
||||||
|
}):
|
||||||
|
ret.update({'changes': {'myvm': {'definition': True, 'cpu': False, 'errors': ['some error']}},
|
||||||
|
@@ -361,7 +361,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
|
||||||
|
# Failed definition update case
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.vm_state': MagicMock(return_value='running'),
|
||||||
|
+ 'virt.vm_state': MagicMock(return_value={'myvm': 'running'}),
|
||||||
|
'virt.update': MagicMock(side_effect=[self.mock_libvirt.libvirtError('error message')])
|
||||||
|
}):
|
||||||
|
ret.update({'changes': {},
|
||||||
|
@@ -573,7 +573,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
define_mock.assert_called_with('mynet',
|
||||||
|
'br2',
|
||||||
|
'bridge',
|
||||||
|
- 'openvswitch',
|
||||||
|
+ vport='openvswitch',
|
||||||
|
tag=180,
|
||||||
|
autostart=False,
|
||||||
|
start=True,
|
||||||
|
@@ -582,7 +582,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
password='secret')
|
||||||
|
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.network_info': MagicMock(return_value={'active': True}),
|
||||||
|
+ 'virt.network_info': MagicMock(return_value={'mynet': {'active': True}}),
|
||||||
|
'virt.network_define': define_mock,
|
||||||
|
}):
|
||||||
|
ret.update({'changes': {}, 'comment': 'Network mynet exists and is running'})
|
||||||
|
@@ -590,7 +590,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
|
||||||
|
start_mock = MagicMock(return_value=True)
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.network_info': MagicMock(return_value={'active': False}),
|
||||||
|
+ 'virt.network_info': MagicMock(return_value={'mynet': {'active': False}}),
|
||||||
|
'virt.network_start': start_mock,
|
||||||
|
'virt.network_define': define_mock,
|
||||||
|
}):
|
||||||
|
@@ -666,10 +666,13 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
connection='myconnection',
|
||||||
|
username='user',
|
||||||
|
password='secret')
|
||||||
|
- mocks['start'].assert_not_called()
|
||||||
|
+ mocks['start'].assert_called_with('mypool',
|
||||||
|
+ connection='myconnection',
|
||||||
|
+ username='user',
|
||||||
|
+ password='secret')
|
||||||
|
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.pool_info': MagicMock(return_value={'state': 'running'}),
|
||||||
|
+ 'virt.pool_info': MagicMock(return_value={'mypool': {'state': 'running'}}),
|
||||||
|
}):
|
||||||
|
ret.update({'changes': {}, 'comment': 'Pool mypool exists and is running'})
|
||||||
|
self.assertDictEqual(virt.pool_running('mypool',
|
||||||
|
@@ -680,7 +683,7 @@ class LibvirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
for mock in mocks:
|
||||||
|
mocks[mock].reset_mock()
|
||||||
|
with patch.dict(virt.__salt__, { # pylint: disable=no-member
|
||||||
|
- 'virt.pool_info': MagicMock(return_value={'state': 'stopped'}),
|
||||||
|
+ 'virt.pool_info': MagicMock(return_value={'mypool': {'state': 'stopped'}}),
|
||||||
|
'virt.pool_build': mocks['build'],
|
||||||
|
'virt.pool_start': mocks['start']
|
||||||
|
}):
|
||||||
|
--
|
||||||
|
2.20.1
|
||||||
|
|
||||||
|
|
@ -1 +1 @@
|
|||||||
3968202dca49a70046366b3807d1e6a2f762ffdf
|
24e59963d380d183a48f6ddd4d66dbf6a8fa4210
|
40
fix-aptpkg-systemd-call-bsc-1143301.patch
Normal file
40
fix-aptpkg-systemd-call-bsc-1143301.patch
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
From f667d6f0534498e2aaa6e46242727bafc13241fd Mon Sep 17 00:00:00 2001
|
||||||
|
From: Mihai Dinca <mdinca@suse.de>
|
||||||
|
Date: Wed, 31 Jul 2019 15:29:03 +0200
|
||||||
|
Subject: [PATCH] Fix aptpkg systemd call (bsc#1143301)
|
||||||
|
|
||||||
|
---
|
||||||
|
salt/modules/aptpkg.py | 2 +-
|
||||||
|
tests/unit/modules/test_aptpkg.py | 2 +-
|
||||||
|
2 files changed, 2 insertions(+), 2 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/salt/modules/aptpkg.py b/salt/modules/aptpkg.py
|
||||||
|
index e537f5b007..b7c1a342ef 100644
|
||||||
|
--- a/salt/modules/aptpkg.py
|
||||||
|
+++ b/salt/modules/aptpkg.py
|
||||||
|
@@ -165,7 +165,7 @@ def _call_apt(args, scope=True, **kwargs):
|
||||||
|
'''
|
||||||
|
cmd = []
|
||||||
|
if scope and salt.utils.systemd.has_scope(__context__) and __salt__['config.get']('systemd.scope', True):
|
||||||
|
- cmd.extend(['systemd-run', '--scope'])
|
||||||
|
+ cmd.extend(['systemd-run', '--scope', '--description "{0}"'.format(__name__)])
|
||||||
|
cmd.extend(args)
|
||||||
|
|
||||||
|
params = {'output_loglevel': 'trace',
|
||||||
|
diff --git a/tests/unit/modules/test_aptpkg.py b/tests/unit/modules/test_aptpkg.py
|
||||||
|
index 580b840197..06f3a9f6aa 100644
|
||||||
|
--- a/tests/unit/modules/test_aptpkg.py
|
||||||
|
+++ b/tests/unit/modules/test_aptpkg.py
|
||||||
|
@@ -544,7 +544,7 @@ class AptUtilsTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
with patch.dict(aptpkg.__salt__, {'cmd.run_all': MagicMock(), 'config.get': MagicMock(return_value=True)}):
|
||||||
|
aptpkg._call_apt(['apt-get', 'purge', 'vim']) # pylint: disable=W0106
|
||||||
|
aptpkg.__salt__['cmd.run_all'].assert_called_once_with(
|
||||||
|
- ['systemd-run', '--scope', 'apt-get', 'purge', 'vim'], env={},
|
||||||
|
+ ['systemd-run', '--scope', '--description "salt.modules.aptpkg"', 'apt-get', 'purge', 'vim'], env={},
|
||||||
|
output_loglevel='trace', python_shell=False)
|
||||||
|
|
||||||
|
def test_call_apt_with_kwargs(self):
|
||||||
|
--
|
||||||
|
2.22.0
|
||||||
|
|
||||||
|
|
374
fix-virt.full_info-176.patch
Normal file
374
fix-virt.full_info-176.patch
Normal file
@ -0,0 +1,374 @@
|
|||||||
|
From 4ce0bc544174fdb00482db4653fb4b0ef411e78b Mon Sep 17 00:00:00 2001
|
||||||
|
From: Cedric Bosdonnat <cbosdonnat@suse.com>
|
||||||
|
Date: Tue, 3 Sep 2019 15:18:04 +0200
|
||||||
|
Subject: [PATCH] Fix virt.full_info (#176)
|
||||||
|
|
||||||
|
* virt.get_xml doesn't take a domain object
|
||||||
|
|
||||||
|
In some places in the virt module, the get_xml function was called with
|
||||||
|
a domain object, leading to runtime errors like the following one:
|
||||||
|
|
||||||
|
'ERROR: The VM "<libvirt.virDomain object at 0x7fad04208650>" is not present'
|
||||||
|
|
||||||
|
* qemu-img info needs -U flag on running VMs
|
||||||
|
|
||||||
|
When getting VM disks informations on a running VM, the following error
|
||||||
|
occured:
|
||||||
|
|
||||||
|
The minion function caused an exception: Traceback (most recent call last):
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/minion.py", line 1673, in _thread_return
|
||||||
|
return_data = minion_instance.executors[fname](opts, data, func, args, kwargs)
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/executors/direct_call.py", line 12, in execute
|
||||||
|
return func(*args, **kwargs)
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/modules/virt.py", line 2411, in full_info
|
||||||
|
'vm_info': vm_info()}
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/modules/virt.py", line 2020, in vm_info
|
||||||
|
info[domain.name()] = _info(domain)
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/modules/virt.py", line 2004, in _info
|
||||||
|
'disks': _get_disks(dom),
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/modules/virt.py", line 465, in _get_disks
|
||||||
|
output = _parse_qemu_img_info(qemu_output)
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/modules/virt.py", line 287, in _parse_qemu_img_info
|
||||||
|
raw_infos = salt.utils.json.loads(info)
|
||||||
|
File "/usr/lib/python3.6/site-packages/salt/utils/json.py", line 92, in loads
|
||||||
|
return json_module.loads(s, **kwargs)
|
||||||
|
File "/usr/lib64/python3.6/json/__init__.py", line 354, in loads
|
||||||
|
return _default_decoder.decode(s)
|
||||||
|
File "/usr/lib64/python3.6/json/decoder.py", line 339, in decode
|
||||||
|
obj, end = self.raw_decode(s, idx=_w(s, 0).end())
|
||||||
|
File "/usr/lib64/python3.6/json/decoder.py", line 357, in raw_decode
|
||||||
|
raise JSONDecodeError("Expecting value", s, err.value) from None
|
||||||
|
json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)
|
||||||
|
|
||||||
|
This is due to the fact that qemu-img can't get infos on a disk that is
|
||||||
|
already used like by a running VM. Using the qemu-img -U flag gets it
|
||||||
|
running in all cases.
|
||||||
|
---
|
||||||
|
salt/modules/virt.py | 10 +-
|
||||||
|
tests/unit/modules/test_virt.py | 242 +++++++++++++++++---------------
|
||||||
|
2 files changed, 132 insertions(+), 120 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/salt/modules/virt.py b/salt/modules/virt.py
|
||||||
|
index 96c17bd60b..d01b6c3f1e 100644
|
||||||
|
--- a/salt/modules/virt.py
|
||||||
|
+++ b/salt/modules/virt.py
|
||||||
|
@@ -331,7 +331,7 @@ def _get_uuid(dom):
|
||||||
|
|
||||||
|
salt '*' virt.get_uuid <domain>
|
||||||
|
'''
|
||||||
|
- return ElementTree.fromstring(get_xml(dom)).find('uuid').text
|
||||||
|
+ return ElementTree.fromstring(dom.XMLDesc(0)).find('uuid').text
|
||||||
|
|
||||||
|
|
||||||
|
def _get_on_poweroff(dom):
|
||||||
|
@@ -344,7 +344,7 @@ def _get_on_poweroff(dom):
|
||||||
|
|
||||||
|
salt '*' virt.get_on_restart <domain>
|
||||||
|
'''
|
||||||
|
- node = ElementTree.fromstring(get_xml(dom)).find('on_poweroff')
|
||||||
|
+ node = ElementTree.fromstring(dom.XMLDesc(0)).find('on_poweroff')
|
||||||
|
return node.text if node is not None else ''
|
||||||
|
|
||||||
|
|
||||||
|
@@ -358,7 +358,7 @@ def _get_on_reboot(dom):
|
||||||
|
|
||||||
|
salt '*' virt.get_on_reboot <domain>
|
||||||
|
'''
|
||||||
|
- node = ElementTree.fromstring(get_xml(dom)).find('on_reboot')
|
||||||
|
+ node = ElementTree.fromstring(dom.XMLDesc(0)).find('on_reboot')
|
||||||
|
return node.text if node is not None else ''
|
||||||
|
|
||||||
|
|
||||||
|
@@ -372,7 +372,7 @@ def _get_on_crash(dom):
|
||||||
|
|
||||||
|
salt '*' virt.get_on_crash <domain>
|
||||||
|
'''
|
||||||
|
- node = ElementTree.fromstring(get_xml(dom)).find('on_crash')
|
||||||
|
+ node = ElementTree.fromstring(dom.XMLDesc(0)).find('on_crash')
|
||||||
|
return node.text if node is not None else ''
|
||||||
|
|
||||||
|
|
||||||
|
@@ -458,7 +458,7 @@ def _get_disks(dom):
|
||||||
|
if driver is not None and driver.get('type') == 'qcow2':
|
||||||
|
try:
|
||||||
|
stdout = subprocess.Popen(
|
||||||
|
- ['qemu-img', 'info', '--output', 'json', '--backing-chain', disk['file']],
|
||||||
|
+ ['qemu-img', 'info', '-U', '--output', 'json', '--backing-chain', disk['file']],
|
||||||
|
shell=False,
|
||||||
|
stdout=subprocess.PIPE).communicate()[0]
|
||||||
|
qemu_output = salt.utils.stringutils.to_str(stdout)
|
||||||
|
diff --git a/tests/unit/modules/test_virt.py b/tests/unit/modules/test_virt.py
|
||||||
|
index e644e62452..4d20e998d8 100644
|
||||||
|
--- a/tests/unit/modules/test_virt.py
|
||||||
|
+++ b/tests/unit/modules/test_virt.py
|
||||||
|
@@ -81,7 +81,9 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
mock_domain.XMLDesc.return_value = xml # pylint: disable=no-member
|
||||||
|
|
||||||
|
# Return state as shutdown
|
||||||
|
- mock_domain.info.return_value = [4, 0, 0, 0] # pylint: disable=no-member
|
||||||
|
+ mock_domain.info.return_value = [4, 2048 * 1024, 1024 * 1024, 2, 1234] # pylint: disable=no-member
|
||||||
|
+ mock_domain.ID.return_value = 1
|
||||||
|
+ mock_domain.name.return_value = name
|
||||||
|
return mock_domain
|
||||||
|
|
||||||
|
def test_disk_profile_merge(self):
|
||||||
|
@@ -1394,49 +1396,6 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
re.match('^([0-9A-F]{2}[:-]){5}([0-9A-F]{2})$',
|
||||||
|
interface_attrs['mac'], re.I))
|
||||||
|
|
||||||
|
- def test_get_graphics(self):
|
||||||
|
- '''
|
||||||
|
- Test virt.get_graphics()
|
||||||
|
- '''
|
||||||
|
- xml = '''<domain type='kvm' id='7'>
|
||||||
|
- <name>test-vm</name>
|
||||||
|
- <devices>
|
||||||
|
- <graphics type='vnc' port='5900' autoport='yes' listen='0.0.0.0'>
|
||||||
|
- <listen type='address' address='0.0.0.0'/>
|
||||||
|
- </graphics>
|
||||||
|
- </devices>
|
||||||
|
- </domain>
|
||||||
|
- '''
|
||||||
|
- self.set_mock_vm("test-vm", xml)
|
||||||
|
-
|
||||||
|
- graphics = virt.get_graphics('test-vm')
|
||||||
|
- self.assertEqual('vnc', graphics['type'])
|
||||||
|
- self.assertEqual('5900', graphics['port'])
|
||||||
|
- self.assertEqual('0.0.0.0', graphics['listen'])
|
||||||
|
-
|
||||||
|
- def test_get_nics(self):
|
||||||
|
- '''
|
||||||
|
- Test virt.get_nics()
|
||||||
|
- '''
|
||||||
|
- xml = '''<domain type='kvm' id='7'>
|
||||||
|
- <name>test-vm</name>
|
||||||
|
- <devices>
|
||||||
|
- <interface type='bridge'>
|
||||||
|
- <mac address='ac:de:48:b6:8b:59'/>
|
||||||
|
- <source bridge='br0'/>
|
||||||
|
- <model type='virtio'/>
|
||||||
|
- <address type='pci' domain='0x0000' bus='0x00' slot='0x03' function='0x0'/>
|
||||||
|
- </interface>
|
||||||
|
- </devices>
|
||||||
|
- </domain>
|
||||||
|
- '''
|
||||||
|
- self.set_mock_vm("test-vm", xml)
|
||||||
|
-
|
||||||
|
- nics = virt.get_nics('test-vm')
|
||||||
|
- nic = nics[list(nics)[0]]
|
||||||
|
- self.assertEqual('bridge', nic['type'])
|
||||||
|
- self.assertEqual('ac:de:48:b6:8b:59', nic['mac'])
|
||||||
|
-
|
||||||
|
def test_parse_qemu_img_info(self):
|
||||||
|
'''
|
||||||
|
Make sure that qemu-img info output is properly parsed
|
||||||
|
@@ -1558,77 +1517,6 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
],
|
||||||
|
}, virt._parse_qemu_img_info(qemu_infos))
|
||||||
|
|
||||||
|
- def test_get_disks(self):
|
||||||
|
- '''
|
||||||
|
- Test virt.get_disks()
|
||||||
|
- '''
|
||||||
|
- xml = '''<domain type='kvm' id='7'>
|
||||||
|
- <name>test-vm</name>
|
||||||
|
- <devices>
|
||||||
|
- <disk type='file' device='disk'>
|
||||||
|
- <driver name='qemu' type='qcow2'/>
|
||||||
|
- <source file='/disks/test.qcow2'/>
|
||||||
|
- <target dev='vda' bus='virtio'/>
|
||||||
|
- </disk>
|
||||||
|
- <disk type='file' device='cdrom'>
|
||||||
|
- <driver name='qemu' type='raw'/>
|
||||||
|
- <source file='/disks/test-cdrom.iso'/>
|
||||||
|
- <target dev='hda' bus='ide'/>
|
||||||
|
- <readonly/>
|
||||||
|
- </disk>
|
||||||
|
- </devices>
|
||||||
|
- </domain>
|
||||||
|
- '''
|
||||||
|
- self.set_mock_vm("test-vm", xml)
|
||||||
|
-
|
||||||
|
- qemu_infos = '''[{
|
||||||
|
- "virtual-size": 25769803776,
|
||||||
|
- "filename": "/disks/test.qcow2",
|
||||||
|
- "cluster-size": 65536,
|
||||||
|
- "format": "qcow2",
|
||||||
|
- "actual-size": 217088,
|
||||||
|
- "format-specific": {
|
||||||
|
- "type": "qcow2",
|
||||||
|
- "data": {
|
||||||
|
- "compat": "1.1",
|
||||||
|
- "lazy-refcounts": false,
|
||||||
|
- "refcount-bits": 16,
|
||||||
|
- "corrupt": false
|
||||||
|
- }
|
||||||
|
- },
|
||||||
|
- "full-backing-filename": "/disks/mybacking.qcow2",
|
||||||
|
- "backing-filename": "mybacking.qcow2",
|
||||||
|
- "dirty-flag": false
|
||||||
|
- },
|
||||||
|
- {
|
||||||
|
- "virtual-size": 25769803776,
|
||||||
|
- "filename": "/disks/mybacking.qcow2",
|
||||||
|
- "cluster-size": 65536,
|
||||||
|
- "format": "qcow2",
|
||||||
|
- "actual-size": 393744384,
|
||||||
|
- "format-specific": {
|
||||||
|
- "type": "qcow2",
|
||||||
|
- "data": {
|
||||||
|
- "compat": "1.1",
|
||||||
|
- "lazy-refcounts": false,
|
||||||
|
- "refcount-bits": 16,
|
||||||
|
- "corrupt": false
|
||||||
|
- }
|
||||||
|
- },
|
||||||
|
- "dirty-flag": false
|
||||||
|
- }]'''
|
||||||
|
-
|
||||||
|
- self.mock_popen.communicate.return_value = [qemu_infos] # pylint: disable=no-member
|
||||||
|
- disks = virt.get_disks('test-vm')
|
||||||
|
- disk = disks.get('vda')
|
||||||
|
- self.assertEqual('/disks/test.qcow2', disk['file'])
|
||||||
|
- self.assertEqual('disk', disk['type'])
|
||||||
|
- self.assertEqual('/disks/mybacking.qcow2', disk['backing file']['file'])
|
||||||
|
- cdrom = disks.get('hda')
|
||||||
|
- self.assertEqual('/disks/test-cdrom.iso', cdrom['file'])
|
||||||
|
- self.assertEqual('cdrom', cdrom['type'])
|
||||||
|
- self.assertFalse('backing file' in cdrom.keys())
|
||||||
|
-
|
||||||
|
@patch('salt.modules.virt.stop', return_value=True)
|
||||||
|
@patch('salt.modules.virt.undefine')
|
||||||
|
@patch('os.remove')
|
||||||
|
@@ -2994,3 +2882,127 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
virt.volume_delete('default', 'missing')
|
||||||
|
virt.volume_delete('missing', 'test_volume')
|
||||||
|
self.assertEqual(mock_delete.call_count, 2)
|
||||||
|
+
|
||||||
|
+ def test_full_info(self):
|
||||||
|
+ '''
|
||||||
|
+ Test virt.full_info
|
||||||
|
+ '''
|
||||||
|
+ xml = '''<domain type='kvm' id='7'>
|
||||||
|
+ <uuid>28deee33-4859-4f23-891c-ee239cffec94</uuid>
|
||||||
|
+ <name>test-vm</name>
|
||||||
|
+ <on_poweroff>destroy</on_poweroff>
|
||||||
|
+ <on_reboot>restart</on_reboot>
|
||||||
|
+ <on_crash>destroy</on_crash>
|
||||||
|
+ <devices>
|
||||||
|
+ <disk type='file' device='disk'>
|
||||||
|
+ <driver name='qemu' type='qcow2'/>
|
||||||
|
+ <source file='/disks/test.qcow2'/>
|
||||||
|
+ <target dev='vda' bus='virtio'/>
|
||||||
|
+ </disk>
|
||||||
|
+ <disk type='file' device='cdrom'>
|
||||||
|
+ <driver name='qemu' type='raw'/>
|
||||||
|
+ <source file='/disks/test-cdrom.iso'/>
|
||||||
|
+ <target dev='hda' bus='ide'/>
|
||||||
|
+ <readonly/>
|
||||||
|
+ </disk>
|
||||||
|
+ <interface type='bridge'>
|
||||||
|
+ <mac address='ac:de:48:b6:8b:59'/>
|
||||||
|
+ <source bridge='br0'/>
|
||||||
|
+ <model type='virtio'/>
|
||||||
|
+ <address type='pci' domain='0x0000' bus='0x00' slot='0x03' function='0x0'/>
|
||||||
|
+ </interface>
|
||||||
|
+ <graphics type='vnc' port='5900' autoport='yes' listen='0.0.0.0'>
|
||||||
|
+ <listen type='address' address='0.0.0.0'/>
|
||||||
|
+ </graphics>
|
||||||
|
+ </devices>
|
||||||
|
+ </domain>
|
||||||
|
+ '''
|
||||||
|
+ self.set_mock_vm("test-vm", xml)
|
||||||
|
+
|
||||||
|
+ qemu_infos = '''[{
|
||||||
|
+ "virtual-size": 25769803776,
|
||||||
|
+ "filename": "/disks/test.qcow2",
|
||||||
|
+ "cluster-size": 65536,
|
||||||
|
+ "format": "qcow2",
|
||||||
|
+ "actual-size": 217088,
|
||||||
|
+ "format-specific": {
|
||||||
|
+ "type": "qcow2",
|
||||||
|
+ "data": {
|
||||||
|
+ "compat": "1.1",
|
||||||
|
+ "lazy-refcounts": false,
|
||||||
|
+ "refcount-bits": 16,
|
||||||
|
+ "corrupt": false
|
||||||
|
+ }
|
||||||
|
+ },
|
||||||
|
+ "full-backing-filename": "/disks/mybacking.qcow2",
|
||||||
|
+ "backing-filename": "mybacking.qcow2",
|
||||||
|
+ "dirty-flag": false
|
||||||
|
+ },
|
||||||
|
+ {
|
||||||
|
+ "virtual-size": 25769803776,
|
||||||
|
+ "filename": "/disks/mybacking.qcow2",
|
||||||
|
+ "cluster-size": 65536,
|
||||||
|
+ "format": "qcow2",
|
||||||
|
+ "actual-size": 393744384,
|
||||||
|
+ "format-specific": {
|
||||||
|
+ "type": "qcow2",
|
||||||
|
+ "data": {
|
||||||
|
+ "compat": "1.1",
|
||||||
|
+ "lazy-refcounts": false,
|
||||||
|
+ "refcount-bits": 16,
|
||||||
|
+ "corrupt": false
|
||||||
|
+ }
|
||||||
|
+ },
|
||||||
|
+ "dirty-flag": false
|
||||||
|
+ }]'''
|
||||||
|
+
|
||||||
|
+ self.mock_popen.communicate.return_value = [qemu_infos] # pylint: disable=no-member
|
||||||
|
+
|
||||||
|
+ self.mock_conn.getInfo = MagicMock(return_value=['x86_64', 4096, 8, 2712, 1, 2, 4, 2])
|
||||||
|
+
|
||||||
|
+ actual = virt.full_info()
|
||||||
|
+
|
||||||
|
+ # Test the hypervisor infos
|
||||||
|
+ self.assertEqual(2816, actual['freemem'])
|
||||||
|
+ self.assertEqual(6, actual['freecpu'])
|
||||||
|
+ self.assertEqual(4, actual['node_info']['cpucores'])
|
||||||
|
+ self.assertEqual(2712, actual['node_info']['cpumhz'])
|
||||||
|
+ self.assertEqual('x86_64', actual['node_info']['cpumodel'])
|
||||||
|
+ self.assertEqual(8, actual['node_info']['cpus'])
|
||||||
|
+ self.assertEqual(2, actual['node_info']['cputhreads'])
|
||||||
|
+ self.assertEqual(1, actual['node_info']['numanodes'])
|
||||||
|
+ self.assertEqual(4096, actual['node_info']['phymemory'])
|
||||||
|
+ self.assertEqual(2, actual['node_info']['sockets'])
|
||||||
|
+
|
||||||
|
+ # Test the vm_info output:
|
||||||
|
+ self.assertEqual(2, actual['vm_info']['test-vm']['cpu'])
|
||||||
|
+ self.assertEqual(1234, actual['vm_info']['test-vm']['cputime'])
|
||||||
|
+ self.assertEqual(1024 * 1024, actual['vm_info']['test-vm']['mem'])
|
||||||
|
+ self.assertEqual(2048 * 1024, actual['vm_info']['test-vm']['maxMem'])
|
||||||
|
+ self.assertEqual('shutdown', actual['vm_info']['test-vm']['state'])
|
||||||
|
+ self.assertEqual('28deee33-4859-4f23-891c-ee239cffec94', actual['vm_info']['test-vm']['uuid'])
|
||||||
|
+ self.assertEqual('destroy', actual['vm_info']['test-vm']['on_crash'])
|
||||||
|
+ self.assertEqual('restart', actual['vm_info']['test-vm']['on_reboot'])
|
||||||
|
+ self.assertEqual('destroy', actual['vm_info']['test-vm']['on_poweroff'])
|
||||||
|
+
|
||||||
|
+ # Test the nics
|
||||||
|
+ nic = actual['vm_info']['test-vm']['nics']['ac:de:48:b6:8b:59']
|
||||||
|
+ self.assertEqual('bridge', nic['type'])
|
||||||
|
+ self.assertEqual('ac:de:48:b6:8b:59', nic['mac'])
|
||||||
|
+
|
||||||
|
+ # Test the disks
|
||||||
|
+ disks = actual['vm_info']['test-vm']['disks']
|
||||||
|
+ disk = disks.get('vda')
|
||||||
|
+ self.assertEqual('/disks/test.qcow2', disk['file'])
|
||||||
|
+ self.assertEqual('disk', disk['type'])
|
||||||
|
+ self.assertEqual('/disks/mybacking.qcow2', disk['backing file']['file'])
|
||||||
|
+ cdrom = disks.get('hda')
|
||||||
|
+ self.assertEqual('/disks/test-cdrom.iso', cdrom['file'])
|
||||||
|
+ self.assertEqual('cdrom', cdrom['type'])
|
||||||
|
+ self.assertFalse('backing file' in cdrom.keys())
|
||||||
|
+
|
||||||
|
+ # Test the graphics
|
||||||
|
+ graphics = actual['vm_info']['test-vm']['graphics']
|
||||||
|
+ self.assertEqual('vnc', graphics['type'])
|
||||||
|
+ self.assertEqual('5900', graphics['port'])
|
||||||
|
+ self.assertEqual('0.0.0.0', graphics['listen'])
|
||||||
|
--
|
||||||
|
2.20.1
|
||||||
|
|
||||||
|
|
326
implement-network.fqdns-module-function-bsc-1134860-.patch
Normal file
326
implement-network.fqdns-module-function-bsc-1134860-.patch
Normal file
@ -0,0 +1,326 @@
|
|||||||
|
From 76d0ec5ec0764f6c5e71ddc2dc03bd12c25045a0 Mon Sep 17 00:00:00 2001
|
||||||
|
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>
|
||||||
|
---
|
||||||
|
salt/grains/core.py | 66 +++++-----------------------------
|
||||||
|
salt/modules/network.py | 60 +++++++++++++++++++++++++++++++
|
||||||
|
salt/utils/network.py | 12 +++++++
|
||||||
|
tests/unit/grains/test_core.py | 64 ++++++++++++++++++++++++++-------
|
||||||
|
4 files changed, 131 insertions(+), 71 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/salt/grains/core.py b/salt/grains/core.py
|
||||||
|
index e54212edfb..fa188a6ff7 100644
|
||||||
|
--- a/salt/grains/core.py
|
||||||
|
+++ b/salt/grains/core.py
|
||||||
|
@@ -25,8 +25,9 @@ import zlib
|
||||||
|
from errno import EACCES, EPERM
|
||||||
|
import datetime
|
||||||
|
import warnings
|
||||||
|
+import salt.modules.network
|
||||||
|
|
||||||
|
-from multiprocessing.pool import ThreadPool
|
||||||
|
+from salt.utils.network import _get_interfaces
|
||||||
|
|
||||||
|
# pylint: disable=import-error
|
||||||
|
try:
|
||||||
|
@@ -83,6 +84,7 @@ __salt__ = {
|
||||||
|
'cmd.run_all': salt.modules.cmdmod._run_all_quiet,
|
||||||
|
'smbios.records': salt.modules.smbios.records,
|
||||||
|
'smbios.get': salt.modules.smbios.get,
|
||||||
|
+ 'network.fqdns': salt.modules.network.fqdns,
|
||||||
|
}
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@@ -106,7 +108,6 @@ HAS_UNAME = True
|
||||||
|
if not hasattr(os, 'uname'):
|
||||||
|
HAS_UNAME = False
|
||||||
|
|
||||||
|
-_INTERFACES = {}
|
||||||
|
|
||||||
|
# Possible value for h_errno defined in netdb.h
|
||||||
|
HOST_NOT_FOUND = 1
|
||||||
|
@@ -1506,17 +1507,6 @@ def _linux_bin_exists(binary):
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
-def _get_interfaces():
|
||||||
|
- '''
|
||||||
|
- Provide a dict of the connected interfaces and their ip addresses
|
||||||
|
- '''
|
||||||
|
-
|
||||||
|
- global _INTERFACES
|
||||||
|
- if not _INTERFACES:
|
||||||
|
- _INTERFACES = salt.utils.network.interfaces()
|
||||||
|
- return _INTERFACES
|
||||||
|
-
|
||||||
|
-
|
||||||
|
def _parse_lsb_release():
|
||||||
|
ret = {}
|
||||||
|
try:
|
||||||
|
@@ -2200,52 +2190,12 @@ def fqdns():
|
||||||
|
'''
|
||||||
|
Return all known FQDNs for the system by enumerating all interfaces and
|
||||||
|
then trying to reverse resolve them (excluding 'lo' interface).
|
||||||
|
+ To disable the fqdns grain, set enable_fqdns_grains: False in the minion configuration file.
|
||||||
|
'''
|
||||||
|
- # Provides:
|
||||||
|
- # fqdns
|
||||||
|
-
|
||||||
|
- grains = {}
|
||||||
|
- fqdns = set()
|
||||||
|
-
|
||||||
|
- def _lookup_fqdn(ip):
|
||||||
|
- try:
|
||||||
|
- name, aliaslist, addresslist = socket.gethostbyaddr(ip)
|
||||||
|
- 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.
|
||||||
|
- log.debug("Unable to resolve address %s: %s", ip, err)
|
||||||
|
- else:
|
||||||
|
- log.error(err_message, err)
|
||||||
|
- except (socket.error, socket.gaierror, socket.timeout) as err:
|
||||||
|
- log.error(err_message, err)
|
||||||
|
-
|
||||||
|
- start = time.time()
|
||||||
|
-
|
||||||
|
- 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'
|
||||||
|
-
|
||||||
|
- # 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:
|
||||||
|
- pool = ThreadPool(8)
|
||||||
|
- results = pool.map(_lookup_fqdn, addresses)
|
||||||
|
- pool.close()
|
||||||
|
- pool.join()
|
||||||
|
- except Exception as exc:
|
||||||
|
- log.error("Exception while creating a ThreadPool for resolving FQDNs: %s", exc)
|
||||||
|
-
|
||||||
|
- for item in results:
|
||||||
|
- if item:
|
||||||
|
- fqdns.update(item)
|
||||||
|
-
|
||||||
|
- elapsed = time.time() - start
|
||||||
|
- log.debug('Elapsed time getting FQDNs: {} seconds'.format(elapsed))
|
||||||
|
-
|
||||||
|
- return {"fqdns": sorted(list(fqdns))}
|
||||||
|
+ opt = {"fqdns": []}
|
||||||
|
+ if __opts__.get('enable_fqdns_grains', True) == True:
|
||||||
|
+ opt = __salt__['network.fqdns']()
|
||||||
|
+ return opt
|
||||||
|
|
||||||
|
|
||||||
|
def ip_fqdn():
|
||||||
|
diff --git a/salt/modules/network.py b/salt/modules/network.py
|
||||||
|
index 28bcff1622..5b6ac930ea 100644
|
||||||
|
--- a/salt/modules/network.py
|
||||||
|
+++ b/salt/modules/network.py
|
||||||
|
@@ -11,6 +11,10 @@ import logging
|
||||||
|
import re
|
||||||
|
import os
|
||||||
|
import socket
|
||||||
|
+import time
|
||||||
|
+
|
||||||
|
+from multiprocessing.pool import ThreadPool
|
||||||
|
+
|
||||||
|
|
||||||
|
# Import salt libs
|
||||||
|
import salt.utils.decorators.path
|
||||||
|
@@ -1881,3 +1885,59 @@ def iphexval(ip):
|
||||||
|
a = ip.split('.')
|
||||||
|
hexval = ['%02X' % int(x) for x in a] # pylint: disable=E1321
|
||||||
|
return ''.join(hexval)
|
||||||
|
+
|
||||||
|
+
|
||||||
|
+def fqdns():
|
||||||
|
+ '''
|
||||||
|
+ Return all known FQDNs for the system by enumerating all interfaces and
|
||||||
|
+ then trying to reverse resolve them (excluding 'lo' interface).
|
||||||
|
+ '''
|
||||||
|
+ # Provides:
|
||||||
|
+ # fqdns
|
||||||
|
+
|
||||||
|
+ # Possible value for h_errno defined in netdb.h
|
||||||
|
+ HOST_NOT_FOUND = 1
|
||||||
|
+ NO_DATA = 4
|
||||||
|
+
|
||||||
|
+ grains = {}
|
||||||
|
+ fqdns = set()
|
||||||
|
+
|
||||||
|
+ def _lookup_fqdn(ip):
|
||||||
|
+ try:
|
||||||
|
+ name, aliaslist, addresslist = socket.gethostbyaddr(ip)
|
||||||
|
+ 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.
|
||||||
|
+ log.debug("Unable to resolve address %s: %s", ip, err)
|
||||||
|
+ else:
|
||||||
|
+ log.error(err_message, err)
|
||||||
|
+ except (socket.error, socket.gaierror, socket.timeout) as err:
|
||||||
|
+ log.error(err_message, err)
|
||||||
|
+
|
||||||
|
+ start = time.time()
|
||||||
|
+
|
||||||
|
+ addresses = salt.utils.network.ip_addrs(include_loopback=False, interface_data=salt.utils.network._get_interfaces())
|
||||||
|
+ addresses.extend(salt.utils.network.ip_addrs6(include_loopback=False, interface_data=salt.utils.network._get_interfaces()))
|
||||||
|
+ err_message = 'Exception during resolving address: %s'
|
||||||
|
+
|
||||||
|
+ # 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:
|
||||||
|
+ pool = ThreadPool(8)
|
||||||
|
+ results = pool.map(_lookup_fqdn, addresses)
|
||||||
|
+ pool.close()
|
||||||
|
+ pool.join()
|
||||||
|
+ except Exception as exc:
|
||||||
|
+ log.error("Exception while creating a ThreadPool for resolving FQDNs: %s", exc)
|
||||||
|
+
|
||||||
|
+ for item in results:
|
||||||
|
+ if item:
|
||||||
|
+ fqdns.update(item)
|
||||||
|
+
|
||||||
|
+ elapsed = time.time() - start
|
||||||
|
+ log.debug('Elapsed time getting FQDNs: {} seconds'.format(elapsed))
|
||||||
|
+
|
||||||
|
+ return {"fqdns": sorted(list(fqdns))}
|
||||||
|
\ No newline at end of file
|
||||||
|
diff --git a/salt/utils/network.py b/salt/utils/network.py
|
||||||
|
index 3f0522b9a5..942adf1ca4 100644
|
||||||
|
--- a/salt/utils/network.py
|
||||||
|
+++ b/salt/utils/network.py
|
||||||
|
@@ -55,6 +55,18 @@ except (ImportError, OSError, AttributeError, TypeError):
|
||||||
|
# pylint: disable=C0103
|
||||||
|
|
||||||
|
|
||||||
|
+_INTERFACES = {}
|
||||||
|
+def _get_interfaces(): #! function
|
||||||
|
+ '''
|
||||||
|
+ Provide a dict of the connected interfaces and their ip addresses
|
||||||
|
+ '''
|
||||||
|
+
|
||||||
|
+ global _INTERFACES
|
||||||
|
+ if not _INTERFACES:
|
||||||
|
+ _INTERFACES = interfaces()
|
||||||
|
+ return _INTERFACES
|
||||||
|
+
|
||||||
|
+
|
||||||
|
def sanitize_host(host):
|
||||||
|
'''
|
||||||
|
Sanitize host string.
|
||||||
|
diff --git a/tests/unit/grains/test_core.py b/tests/unit/grains/test_core.py
|
||||||
|
index 5fa0ea06f1..889fb90074 100644
|
||||||
|
--- a/tests/unit/grains/test_core.py
|
||||||
|
+++ b/tests/unit/grains/test_core.py
|
||||||
|
@@ -33,6 +33,7 @@ import salt.utils.network
|
||||||
|
import salt.utils.platform
|
||||||
|
import salt.utils.path
|
||||||
|
import salt.grains.core as core
|
||||||
|
+import salt.modules.network
|
||||||
|
|
||||||
|
# Import 3rd-party libs
|
||||||
|
from salt.ext import six
|
||||||
|
@@ -845,6 +846,40 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
with patch.object(salt.utils.dns, 'parse_resolv', MagicMock(return_value=resolv_mock)):
|
||||||
|
assert core.dns() == ret
|
||||||
|
|
||||||
|
+
|
||||||
|
+ def test_enablefqdnsFalse(self):
|
||||||
|
+ '''
|
||||||
|
+ 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_enablefqdnsTrue(self):
|
||||||
|
+ '''
|
||||||
|
+ testing that grains uses network.fqdns module
|
||||||
|
+ '''
|
||||||
|
+ with patch.dict('salt.grains.core.__salt__', {'network.fqdns': MagicMock(return_value="my.fake.domain")}):
|
||||||
|
+ with patch.dict('salt.grains.core.__opts__', {'enable_fqdns_grains':True}):
|
||||||
|
+ assert core.fqdns() == 'my.fake.domain'
|
||||||
|
+
|
||||||
|
+
|
||||||
|
+ def test_enablefqdnsNone(self):
|
||||||
|
+ '''
|
||||||
|
+ 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_enablefqdnswithoutpaching(self):
|
||||||
|
+ '''
|
||||||
|
+ testing fqdns grains is enabled by default
|
||||||
|
+ '''
|
||||||
|
+ with patch.dict('salt.grains.core.__salt__', {'network.fqdns': MagicMock(return_value="my.fake.domain")}):
|
||||||
|
+ assert core.fqdns() == 'my.fake.domain'
|
||||||
|
+
|
||||||
|
+
|
||||||
|
@skipIf(not salt.utils.platform.is_linux(), 'System is not Linux')
|
||||||
|
@patch.object(salt.utils, 'is_windows', MagicMock(return_value=False))
|
||||||
|
@patch('salt.utils.network.ip_addrs', MagicMock(return_value=['1.2.3.4', '5.6.7.8']))
|
||||||
|
@@ -861,11 +896,12 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
('foo.bar.baz', [], ['fe80::a8b2:93ff:fe00:0']),
|
||||||
|
('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):
|
||||||
|
- fqdns = core.fqdns()
|
||||||
|
- assert "fqdns" in fqdns
|
||||||
|
- 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):
|
||||||
|
+ fqdns = core.fqdns()
|
||||||
|
+ assert "fqdns" in fqdns
|
||||||
|
+ assert len(fqdns['fqdns']) == len(ret['fqdns'])
|
||||||
|
+ assert set(fqdns['fqdns']) == set(ret['fqdns'])
|
||||||
|
|
||||||
|
@skipIf(not salt.utils.platform.is_linux(), 'System is not Linux')
|
||||||
|
@patch.object(salt.utils.platform, 'is_windows', MagicMock(return_value=False))
|
||||||
|
@@ -881,14 +917,16 @@ class CoreGrainsTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
('rinzler.evil-corp.com', ["false-hostname", "badaliass"], ['5.6.7.8']),
|
||||||
|
('foo.bar.baz', [], ['fe80::a8b2:93ff:fe00:0']),
|
||||||
|
('bluesniff.foo.bar', ["alias.bluesniff.foo.bar"], ['fe80::a8b2:93ff:dead:beef'])]
|
||||||
|
- with patch.object(socket, 'gethostbyaddr', side_effect=reverse_resolv_mock):
|
||||||
|
- 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"]
|
||||||
|
+ with patch.dict(core.__salt__, {'network.fqdns': salt.modules.network.fqdns}):
|
||||||
|
+ with patch.object(socket, 'gethostbyaddr', side_effect=reverse_resolv_mock):
|
||||||
|
+ 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"]
|
||||||
|
+
|
||||||
|
def test_core_virtual(self):
|
||||||
|
'''
|
||||||
|
test virtual grain with cmd virt-what
|
||||||
|
--
|
||||||
|
2.22.0
|
||||||
|
|
||||||
|
|
57
move-server_id-deprecation-warning-to-reduce-log-spa.patch
Normal file
57
move-server_id-deprecation-warning-to-reduce-log-spa.patch
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
From dab9967f8e4a67e5b7ddd4e6718414d2e9b25e42 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Mihai Dinca <mdinca@suse.de>
|
||||||
|
Date: Fri, 14 Jun 2019 15:13:12 +0200
|
||||||
|
Subject: [PATCH] Move server_id deprecation warning to reduce log
|
||||||
|
spamming (bsc#1135567) (bsc#1135732)
|
||||||
|
|
||||||
|
---
|
||||||
|
salt/grains/core.py | 4 ----
|
||||||
|
salt/minion.py | 9 +++++++++
|
||||||
|
2 files changed, 9 insertions(+), 4 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/salt/grains/core.py b/salt/grains/core.py
|
||||||
|
index ce64620a24..e54212edfb 100644
|
||||||
|
--- a/salt/grains/core.py
|
||||||
|
+++ b/salt/grains/core.py
|
||||||
|
@@ -2812,10 +2812,6 @@ def get_server_id():
|
||||||
|
if bool(use_crc):
|
||||||
|
id_hash = getattr(zlib, use_crc, zlib.adler32)(__opts__.get('id', '').encode()) & 0xffffffff
|
||||||
|
else:
|
||||||
|
- salt.utils.versions.warn_until('Sodium', 'This server_id is computed nor by Adler32 neither by CRC32. '
|
||||||
|
- 'Please use "server_id_use_crc" option and define algorithm you'
|
||||||
|
- 'prefer (default "Adler32"). The server_id will be computed with'
|
||||||
|
- 'Adler32 by default.')
|
||||||
|
id_hash = _get_hash_by_shell()
|
||||||
|
server_id = {'server_id': id_hash}
|
||||||
|
|
||||||
|
diff --git a/salt/minion.py b/salt/minion.py
|
||||||
|
index 058b7ef6b8..97f74bf47e 100644
|
||||||
|
--- a/salt/minion.py
|
||||||
|
+++ b/salt/minion.py
|
||||||
|
@@ -103,6 +103,7 @@ from salt.utils.odict import OrderedDict
|
||||||
|
from salt.utils.process import (default_signals,
|
||||||
|
SignalHandlingMultiprocessingProcess,
|
||||||
|
ProcessManager)
|
||||||
|
+from salt.utils.versions import warn_until
|
||||||
|
from salt.exceptions import (
|
||||||
|
CommandExecutionError,
|
||||||
|
CommandNotFoundError,
|
||||||
|
@@ -992,6 +993,14 @@ class MinionManager(MinionBase):
|
||||||
|
if (self.opts['master_type'] in ('failover', 'distributed')) or not isinstance(self.opts['master'], list):
|
||||||
|
masters = [masters]
|
||||||
|
|
||||||
|
+ if not self.opts.get('server_id_use_crc'):
|
||||||
|
+ warn_until(
|
||||||
|
+ 'Sodium',
|
||||||
|
+ 'This server_id is computed nor by Adler32 neither by CRC32. '
|
||||||
|
+ 'Please use "server_id_use_crc" option and define algorithm you'
|
||||||
|
+ 'prefer (default "Adler32"). The server_id will be computed with'
|
||||||
|
+ 'Adler32 by default.')
|
||||||
|
+
|
||||||
|
for master in masters:
|
||||||
|
s_opts = copy.deepcopy(self.opts)
|
||||||
|
s_opts['master'] = master
|
||||||
|
--
|
||||||
|
2.22.0
|
||||||
|
|
||||||
|
|
135
restore-default-behaviour-of-pkg-list-return.patch
Normal file
135
restore-default-behaviour-of-pkg-list-return.patch
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
From 56fd68474f399a36b0a74ca9a01890649d997792 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Jochen Breuer <jbreuer@suse.de>
|
||||||
|
Date: Fri, 30 Aug 2019 14:20:06 +0200
|
||||||
|
Subject: [PATCH] Restore default behaviour of pkg list return
|
||||||
|
|
||||||
|
The default behaviour for pkg list return was to not include patches,
|
||||||
|
even when installing patches. Only the packages where returned. There
|
||||||
|
is now parameter to also return patches if that is needed.
|
||||||
|
|
||||||
|
Co-authored-by: Mihai Dinca <mdinca@suse.de>
|
||||||
|
---
|
||||||
|
salt/modules/zypperpkg.py | 32 +++++++++++++++++++++++---------
|
||||||
|
1 file changed, 23 insertions(+), 9 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/salt/modules/zypperpkg.py b/salt/modules/zypperpkg.py
|
||||||
|
index f71d6aac9e..da1953b2a5 100644
|
||||||
|
--- a/salt/modules/zypperpkg.py
|
||||||
|
+++ b/salt/modules/zypperpkg.py
|
||||||
|
@@ -1302,8 +1302,10 @@ def refresh_db(root=None):
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
|
-def _find_types(pkgs):
|
||||||
|
+def _detect_includes(pkgs, inclusion_detection):
|
||||||
|
'''Form a package names list, find prefixes of packages types.'''
|
||||||
|
+ if not inclusion_detection:
|
||||||
|
+ return None
|
||||||
|
return sorted({pkg.split(':', 1)[0] for pkg in pkgs
|
||||||
|
if len(pkg.split(':', 1)) == 2})
|
||||||
|
|
||||||
|
@@ -1319,6 +1321,7 @@ def install(name=None,
|
||||||
|
ignore_repo_failure=False,
|
||||||
|
no_recommends=False,
|
||||||
|
root=None,
|
||||||
|
+ inclusion_detection=False,
|
||||||
|
**kwargs):
|
||||||
|
'''
|
||||||
|
.. versionchanged:: 2015.8.12,2016.3.3,2016.11.0
|
||||||
|
@@ -1433,6 +1436,9 @@ def install(name=None,
|
||||||
|
|
||||||
|
.. versionadded:: 2018.3.0
|
||||||
|
|
||||||
|
+ inclusion_detection:
|
||||||
|
+ Detect ``includes`` based on ``sources``
|
||||||
|
+ By default packages are always included
|
||||||
|
|
||||||
|
Returns a dict containing the new package names and versions::
|
||||||
|
|
||||||
|
@@ -1498,7 +1504,8 @@ def install(name=None,
|
||||||
|
|
||||||
|
diff_attr = kwargs.get("diff_attr")
|
||||||
|
|
||||||
|
- includes = _find_types(targets)
|
||||||
|
+ includes = _detect_includes(targets, inclusion_detection)
|
||||||
|
+
|
||||||
|
old = list_pkgs(attr=diff_attr, root=root, includes=includes) if not downloadonly else list_downloaded(root)
|
||||||
|
|
||||||
|
downgrades = []
|
||||||
|
@@ -1688,7 +1695,7 @@ def upgrade(refresh=True,
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
|
-def _uninstall(name=None, pkgs=None, root=None):
|
||||||
|
+def _uninstall(inclusion_detection, name=None, pkgs=None, root=None):
|
||||||
|
'''
|
||||||
|
Remove and purge do identical things but with different Zypper commands,
|
||||||
|
this function performs the common logic.
|
||||||
|
@@ -1698,7 +1705,7 @@ def _uninstall(name=None, pkgs=None, root=None):
|
||||||
|
except MinionError as exc:
|
||||||
|
raise CommandExecutionError(exc)
|
||||||
|
|
||||||
|
- includes = _find_types(pkg_params.keys())
|
||||||
|
+ includes = _detect_includes(pkg_params.keys(), inclusion_detection)
|
||||||
|
old = list_pkgs(root=root, includes=includes)
|
||||||
|
targets = []
|
||||||
|
for target in pkg_params:
|
||||||
|
@@ -1757,7 +1764,7 @@ def normalize_name(name):
|
||||||
|
return name
|
||||||
|
|
||||||
|
|
||||||
|
-def remove(name=None, pkgs=None, root=None, **kwargs): # pylint: disable=unused-argument
|
||||||
|
+def remove(name=None, pkgs=None, root=None, inclusion_detection=False, **kwargs): # pylint: disable=unused-argument
|
||||||
|
'''
|
||||||
|
.. versionchanged:: 2015.8.12,2016.3.3,2016.11.0
|
||||||
|
On minions running systemd>=205, `systemd-run(1)`_ is now used to
|
||||||
|
@@ -1788,8 +1795,11 @@ def remove(name=None, pkgs=None, root=None, **kwargs): # pylint: disable=unused
|
||||||
|
root
|
||||||
|
Operate on a different root directory.
|
||||||
|
|
||||||
|
- .. versionadded:: 0.16.0
|
||||||
|
+ inclusion_detection:
|
||||||
|
+ Detect ``includes`` based on ``pkgs``
|
||||||
|
+ By default packages are always included
|
||||||
|
|
||||||
|
+ .. versionadded:: 0.16.0
|
||||||
|
|
||||||
|
Returns a dict containing the changes.
|
||||||
|
|
||||||
|
@@ -1801,10 +1811,10 @@ def remove(name=None, pkgs=None, root=None, **kwargs): # pylint: disable=unused
|
||||||
|
salt '*' pkg.remove <package1>,<package2>,<package3>
|
||||||
|
salt '*' pkg.remove pkgs='["foo", "bar"]'
|
||||||
|
'''
|
||||||
|
- return _uninstall(name=name, pkgs=pkgs, root=root)
|
||||||
|
+ return _uninstall(inclusion_detection, name=name, pkgs=pkgs, root=root)
|
||||||
|
|
||||||
|
|
||||||
|
-def purge(name=None, pkgs=None, root=None, **kwargs): # pylint: disable=unused-argument
|
||||||
|
+def purge(name=None, pkgs=None, root=None, inclusion_detection=False, **kwargs): # pylint: disable=unused-argument
|
||||||
|
'''
|
||||||
|
.. versionchanged:: 2015.8.12,2016.3.3,2016.11.0
|
||||||
|
On minions running systemd>=205, `systemd-run(1)`_ is now used to
|
||||||
|
@@ -1836,6 +1846,10 @@ def purge(name=None, pkgs=None, root=None, **kwargs): # pylint: disable=unused-
|
||||||
|
root
|
||||||
|
Operate on a different root directory.
|
||||||
|
|
||||||
|
+ inclusion_detection:
|
||||||
|
+ Detect ``includes`` based on ``pkgs``
|
||||||
|
+ By default packages are always included
|
||||||
|
+
|
||||||
|
.. versionadded:: 0.16.0
|
||||||
|
|
||||||
|
|
||||||
|
@@ -1849,7 +1863,7 @@ def purge(name=None, pkgs=None, root=None, **kwargs): # pylint: disable=unused-
|
||||||
|
salt '*' pkg.purge <package1>,<package2>,<package3>
|
||||||
|
salt '*' pkg.purge pkgs='["foo", "bar"]'
|
||||||
|
'''
|
||||||
|
- return _uninstall(name=name, pkgs=pkgs, root=root)
|
||||||
|
+ return _uninstall(inclusion_detection, name=name, pkgs=pkgs, root=root)
|
||||||
|
|
||||||
|
|
||||||
|
def list_locks(root=None):
|
||||||
|
--
|
||||||
|
2.20.1
|
||||||
|
|
||||||
|
|
58
salt.changes
58
salt.changes
@ -1,3 +1,61 @@
|
|||||||
|
-------------------------------------------------------------------
|
||||||
|
Thu Sep 5 17:45:50 UTC 2019 - Jochen Breuer <jbreuer@suse.de>
|
||||||
|
|
||||||
|
- Require shadow instead of old pwdutils (bsc#1130588)
|
||||||
|
|
||||||
|
-------------------------------------------------------------------
|
||||||
|
Wed Sep 4 18:45:56 UTC 2019 - Jochen Breuer <jbreuer@suse.de>
|
||||||
|
|
||||||
|
- Conflict with tornado >= 5; for now we can only cope with Tornado 4.x (boo#1101780).
|
||||||
|
|
||||||
|
-------------------------------------------------------------------
|
||||||
|
Tue Sep 3 15:16:15 UTC 2019 - Mihai Dincă <mihai.dinca@suse.com>
|
||||||
|
|
||||||
|
- Fix virt.full_info (bsc#1146382)
|
||||||
|
- virt.volume_infos: silence libvirt error message
|
||||||
|
- virt.volume_infos needs to ignore inactive pools
|
||||||
|
- Fix for various bugs in virt network and pool states
|
||||||
|
- Implement network.fqdns module function (bsc#1134860)
|
||||||
|
|
||||||
|
- Added:
|
||||||
|
* 2019.2.0-pr-54196-backport-173.patch
|
||||||
|
* virt.volume_infos-silence-libvirt-error-message-175.patch
|
||||||
|
* fix-virt.full_info-176.patch
|
||||||
|
* implement-network.fqdns-module-function-bsc-1134860-.patch
|
||||||
|
* virt.volume_infos-needs-to-ignore-inactive-pools-174.patch
|
||||||
|
|
||||||
|
-------------------------------------------------------------------
|
||||||
|
Fri Aug 30 13:36:05 UTC 2019 - Jochen Breuer <jbreuer@suse.de>
|
||||||
|
|
||||||
|
- Restore default behaviour of pkg list return (bsc#1148714)
|
||||||
|
- Strip trailing "/" from repo.uri when comparing repos in apktpkg.mod_repo (bsc#1146192)
|
||||||
|
|
||||||
|
- Added:
|
||||||
|
* strip-trailing-from-repo.uri-when-comparing-repos-in.patch
|
||||||
|
* restore-default-behaviour-of-pkg-list-return.patch
|
||||||
|
|
||||||
|
-------------------------------------------------------------------
|
||||||
|
Tue Aug 13 10:43:21 UTC 2019 - Pablo Suárez Hernández <pablo.suarezhernandez@suse.com>
|
||||||
|
|
||||||
|
- Use python3 to build package Salt for RHEL8
|
||||||
|
- Make python3 default for RHEL8
|
||||||
|
|
||||||
|
-------------------------------------------------------------------
|
||||||
|
Fri Aug 9 09:45:31 UTC 2019 - Mihai Dincă <mihai.dinca@suse.com>
|
||||||
|
|
||||||
|
- Fix aptpkg systemd call (bsc#1143301)
|
||||||
|
|
||||||
|
- Added:
|
||||||
|
* fix-aptpkg-systemd-call-bsc-1143301.patch
|
||||||
|
|
||||||
|
-------------------------------------------------------------------
|
||||||
|
Tue Jul 30 14:56:02 UTC 2019 - Mihai Dincă <mihai.dinca@suse.com>
|
||||||
|
|
||||||
|
- Move server_id deprecation warning to reduce log spamming (bsc#1135567) (bsc#1135732)
|
||||||
|
|
||||||
|
- Added:
|
||||||
|
* move-server_id-deprecation-warning-to-reduce-log-spa.patch
|
||||||
|
|
||||||
-------------------------------------------------------------------
|
-------------------------------------------------------------------
|
||||||
Tue Jul 30 09:34:27 UTC 2019 - Pablo Suárez Hernández <pablo.suarezhernandez@suse.com>
|
Tue Jul 30 09:34:27 UTC 2019 - Pablo Suárez Hernández <pablo.suarezhernandez@suse.com>
|
||||||
|
|
||||||
|
43
salt.spec
43
salt.spec
@ -26,8 +26,13 @@
|
|||||||
%global build_py3 1
|
%global build_py3 1
|
||||||
%global build_py2 1
|
%global build_py2 1
|
||||||
%else
|
%else
|
||||||
|
%if 0%{?rhel} == 7
|
||||||
# RES7
|
# RES7
|
||||||
%global build_py2 1
|
%global build_py2 1
|
||||||
|
%else
|
||||||
|
%global build_py3 1
|
||||||
|
%global default_py3 1
|
||||||
|
%endif
|
||||||
%endif
|
%endif
|
||||||
%endif
|
%endif
|
||||||
%define pythonX %{?default_py3: python3}%{!?default_py3: python2}
|
%define pythonX %{?default_py3: python3}%{!?default_py3: python2}
|
||||||
@ -215,6 +220,25 @@ Patch72: avoid-traceback-when-http.query-request-cannot-be-pe.patch
|
|||||||
# https://github.com/saltstack/salt/pull/54022
|
# https://github.com/saltstack/salt/pull/54022
|
||||||
# https://github.com/saltstack/salt/pull/54024
|
# https://github.com/saltstack/salt/pull/54024
|
||||||
Patch73: accumulated-changes-required-for-yomi-165.patch
|
Patch73: accumulated-changes-required-for-yomi-165.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/159
|
||||||
|
Patch74: move-server_id-deprecation-warning-to-reduce-log-spa.patch
|
||||||
|
# PATCH_FIX_UPSTREAM: https://github.com/saltstack/salt/pull/54077
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/166
|
||||||
|
Patch75: fix-aptpkg-systemd-call-bsc-1143301.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/170
|
||||||
|
Patch76: strip-trailing-from-repo.uri-when-comparing-repos-in.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/177
|
||||||
|
Patch77: restore-default-behaviour-of-pkg-list-return.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/172
|
||||||
|
Patch78: implement-network.fqdns-module-function-bsc-1134860-.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/173
|
||||||
|
Patch79: 2019.2.0-pr-54196-backport-173.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/174
|
||||||
|
Patch80: virt.volume_infos-needs-to-ignore-inactive-pools-174.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/175
|
||||||
|
Patch81: virt.volume_infos-silence-libvirt-error-message-175.patch
|
||||||
|
# PATCH-FIX_OPENSUSE: https://github.com/openSUSE/salt/pull/176
|
||||||
|
Patch82: fix-virt.full_info-176.patch
|
||||||
|
|
||||||
# BuildRoot: %{_tmppath}/%{name}-%{version}-build
|
# BuildRoot: %{_tmppath}/%{name}-%{version}-build
|
||||||
BuildRoot: %{_tmppath}/%{name}-%{version}-build
|
BuildRoot: %{_tmppath}/%{name}-%{version}-build
|
||||||
@ -230,7 +254,7 @@ Requires(pre): %{_sbindir}/useradd
|
|||||||
|
|
||||||
%if 0%{?suse_version}
|
%if 0%{?suse_version}
|
||||||
Requires(pre): %fillup_prereq
|
Requires(pre): %fillup_prereq
|
||||||
Requires(pre): pwdutils
|
Requires(pre): shadow
|
||||||
%endif
|
%endif
|
||||||
|
|
||||||
%if 0%{?suse_version}
|
%if 0%{?suse_version}
|
||||||
@ -313,6 +337,8 @@ BuildRequires: python-msgpack-python > 0.3
|
|||||||
BuildRequires: python-psutil
|
BuildRequires: python-psutil
|
||||||
BuildRequires: python-requests >= 1.0.0
|
BuildRequires: python-requests >= 1.0.0
|
||||||
BuildRequires: python-tornado >= 4.2.1
|
BuildRequires: python-tornado >= 4.2.1
|
||||||
|
# We can't cope with tornado 5.x and newer (boo#1101780)
|
||||||
|
BuildConflicts: python3-tornado >= 5
|
||||||
|
|
||||||
# requirements/zeromq.txt
|
# requirements/zeromq.txt
|
||||||
BuildRequires: python-pycrypto >= 2.6.1
|
BuildRequires: python-pycrypto >= 2.6.1
|
||||||
@ -355,6 +381,8 @@ Requires: python-msgpack-python > 0.3
|
|||||||
Requires: python-psutil
|
Requires: python-psutil
|
||||||
Requires: python-requests >= 1.0.0
|
Requires: python-requests >= 1.0.0
|
||||||
Requires: python-tornado >= 4.2.1
|
Requires: python-tornado >= 4.2.1
|
||||||
|
# We can't cope with tornado 5.x and newer (boo#1101780)
|
||||||
|
Conflicts: python3-tornado >= 5
|
||||||
%if 0%{?suse_version}
|
%if 0%{?suse_version}
|
||||||
# required for zypper.py
|
# required for zypper.py
|
||||||
Requires: rpm-python
|
Requires: rpm-python
|
||||||
@ -401,6 +429,8 @@ BuildRequires: python3-msgpack-python > 0.3
|
|||||||
BuildRequires: python3-psutil
|
BuildRequires: python3-psutil
|
||||||
BuildRequires: python3-requests >= 1.0.0
|
BuildRequires: python3-requests >= 1.0.0
|
||||||
BuildRequires: python3-tornado >= 4.2.1
|
BuildRequires: python3-tornado >= 4.2.1
|
||||||
|
# We can't cope with tornado 5.x and newer (boo#1101780)
|
||||||
|
BuildConflicts: python3-tornado >= 5
|
||||||
|
|
||||||
# requirements/zeromq.txt
|
# requirements/zeromq.txt
|
||||||
BuildRequires: python3-pycrypto >= 2.6.1
|
BuildRequires: python3-pycrypto >= 2.6.1
|
||||||
@ -439,6 +469,8 @@ Requires: python3-msgpack-python > 0.3
|
|||||||
Requires: python3-psutil
|
Requires: python3-psutil
|
||||||
Requires: python3-requests >= 1.0.0
|
Requires: python3-requests >= 1.0.0
|
||||||
Requires: python3-tornado >= 4.2.1
|
Requires: python3-tornado >= 4.2.1
|
||||||
|
# We can't cope with tornado 5.x and newer (boo#1101780)
|
||||||
|
Conflicts: python3-tornado >= 5
|
||||||
%if 0%{?suse_version}
|
%if 0%{?suse_version}
|
||||||
# required for zypper.py
|
# required for zypper.py
|
||||||
Requires: python3-rpm
|
Requires: python3-rpm
|
||||||
@ -758,6 +790,15 @@ cp %{S:5} ./.travis.yml
|
|||||||
%patch71 -p1
|
%patch71 -p1
|
||||||
%patch72 -p1
|
%patch72 -p1
|
||||||
%patch73 -p1
|
%patch73 -p1
|
||||||
|
%patch74 -p1
|
||||||
|
%patch75 -p1
|
||||||
|
%patch76 -p1
|
||||||
|
%patch77 -p1
|
||||||
|
%patch78 -p1
|
||||||
|
%patch79 -p1
|
||||||
|
%patch80 -p1
|
||||||
|
%patch81 -p1
|
||||||
|
%patch82 -p1
|
||||||
|
|
||||||
%build
|
%build
|
||||||
%if 0%{?build_py2}
|
%if 0%{?build_py2}
|
||||||
|
28
strip-trailing-from-repo.uri-when-comparing-repos-in.patch
Normal file
28
strip-trailing-from-repo.uri-when-comparing-repos-in.patch
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
From 1b6f3f2e8b88ddfaebd5bfd1ae8258d417a9f098 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Matei Albu <malbu@suse.de>
|
||||||
|
Date: Fri, 15 Feb 2019 14:34:13 +0100
|
||||||
|
Subject: [PATCH] Strip trailing "/" from repo.uri when comparing repos
|
||||||
|
in apktpkg.mod_repo (bsc#1146192)
|
||||||
|
|
||||||
|
(cherry picked from commit af85627)
|
||||||
|
---
|
||||||
|
salt/modules/aptpkg.py | 2 +-
|
||||||
|
1 file changed, 1 insertion(+), 1 deletion(-)
|
||||||
|
|
||||||
|
diff --git a/salt/modules/aptpkg.py b/salt/modules/aptpkg.py
|
||||||
|
index b7c1a342ef..d49a48310e 100644
|
||||||
|
--- a/salt/modules/aptpkg.py
|
||||||
|
+++ b/salt/modules/aptpkg.py
|
||||||
|
@@ -2354,7 +2354,7 @@ def mod_repo(repo, saltenv='base', **kwargs):
|
||||||
|
# and the resulting source line. The idea here is to ensure
|
||||||
|
# we are not returning bogus data because the source line
|
||||||
|
# has already been modified on a previous run.
|
||||||
|
- repo_matches = source.type == repo_type and source.uri == repo_uri and source.dist == repo_dist
|
||||||
|
+ repo_matches = source.type == repo_type and source.uri.rstrip('/') == repo_uri.rstrip('/') and source.dist == repo_dist
|
||||||
|
kw_matches = source.dist == kw_dist and source.type == kw_type
|
||||||
|
|
||||||
|
if repo_matches or kw_matches:
|
||||||
|
--
|
||||||
|
2.20.1
|
||||||
|
|
||||||
|
|
83
virt.volume_infos-needs-to-ignore-inactive-pools-174.patch
Normal file
83
virt.volume_infos-needs-to-ignore-inactive-pools-174.patch
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
From df1caa8fa6551f880202649a7f4133343da5da0f Mon Sep 17 00:00:00 2001
|
||||||
|
From: Cedric Bosdonnat <cbosdonnat@suse.com>
|
||||||
|
Date: Tue, 3 Sep 2019 15:17:38 +0200
|
||||||
|
Subject: [PATCH] virt.volume_infos needs to ignore inactive pools (#174)
|
||||||
|
|
||||||
|
libvirt raises an error when getting the list of volumes of a pool that
|
||||||
|
is not active. Rule out those pools from virt.volume_infos since we
|
||||||
|
still need to give infos on the other pools' volumes.
|
||||||
|
---
|
||||||
|
salt/modules/virt.py | 7 +++++--
|
||||||
|
tests/unit/modules/test_virt.py | 9 +++++++++
|
||||||
|
2 files changed, 14 insertions(+), 2 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/salt/modules/virt.py b/salt/modules/virt.py
|
||||||
|
index 953064cc2c..0353e6a1f5 100644
|
||||||
|
--- a/salt/modules/virt.py
|
||||||
|
+++ b/salt/modules/virt.py
|
||||||
|
@@ -5021,7 +5021,9 @@ def _get_all_volumes_paths(conn):
|
||||||
|
|
||||||
|
:param conn: libvirt connection to use
|
||||||
|
'''
|
||||||
|
- volumes = [vol for l in [obj.listAllVolumes() for obj in conn.listAllStoragePools()] for vol in l]
|
||||||
|
+ volumes = [vol for l in
|
||||||
|
+ [obj.listAllVolumes() for obj in conn.listAllStoragePools()
|
||||||
|
+ if obj.info()[0] == libvirt.VIR_STORAGE_POOL_RUNNING] for vol in l]
|
||||||
|
return {vol.path(): [path.text for path in ElementTree.fromstring(vol.XMLDesc()).findall('.//backingStore/path')]
|
||||||
|
for vol in volumes if _is_valid_volume(vol)}
|
||||||
|
|
||||||
|
@@ -5086,7 +5088,8 @@ def volume_infos(pool=None, volume=None, **kwargs):
|
||||||
|
'used_by': used_by,
|
||||||
|
}
|
||||||
|
|
||||||
|
- pools = [obj for obj in conn.listAllStoragePools() if pool is None or obj.name() == pool]
|
||||||
|
+ pools = [obj for obj in conn.listAllStoragePools()
|
||||||
|
+ if (pool is None or obj.name() == pool) and obj.info()[0] == libvirt.VIR_STORAGE_POOL_RUNNING]
|
||||||
|
vols = {pool_obj.name(): {vol.name(): _volume_extract_infos(vol)
|
||||||
|
for vol in pool_obj.listAllVolumes()
|
||||||
|
if (volume is None or vol.name() == volume) and _is_valid_volume(vol)}
|
||||||
|
diff --git a/tests/unit/modules/test_virt.py b/tests/unit/modules/test_virt.py
|
||||||
|
index b343b9bc31..e644e62452 100644
|
||||||
|
--- a/tests/unit/modules/test_virt.py
|
||||||
|
+++ b/tests/unit/modules/test_virt.py
|
||||||
|
@@ -2743,6 +2743,7 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
mock_pool_data = [
|
||||||
|
{
|
||||||
|
'name': 'pool0',
|
||||||
|
+ 'state': self.mock_libvirt.VIR_STORAGE_POOL_RUNNING,
|
||||||
|
'volumes': [
|
||||||
|
{
|
||||||
|
'key': '/key/of/vol0',
|
||||||
|
@@ -2755,6 +2756,7 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'name': 'pool1',
|
||||||
|
+ 'state': self.mock_libvirt.VIR_STORAGE_POOL_RUNNING,
|
||||||
|
'volumes': [
|
||||||
|
{
|
||||||
|
'key': '/key/of/vol0bad',
|
||||||
|
@@ -2784,6 +2786,7 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
for pool_data in mock_pool_data:
|
||||||
|
mock_pool = MagicMock()
|
||||||
|
mock_pool.name.return_value = pool_data['name'] # pylint: disable=no-member
|
||||||
|
+ mock_pool.info.return_value = [pool_data['state']]
|
||||||
|
mock_volumes = []
|
||||||
|
for vol_data in pool_data['volumes']:
|
||||||
|
mock_volume = MagicMock()
|
||||||
|
@@ -2817,6 +2820,12 @@ class VirtTestCase(TestCase, LoaderModuleMockMixin):
|
||||||
|
mock_pool.listAllVolumes.return_value = mock_volumes # pylint: disable=no-member
|
||||||
|
mock_pools.append(mock_pool)
|
||||||
|
|
||||||
|
+ inactive_pool = MagicMock()
|
||||||
|
+ inactive_pool.name.return_value = 'pool2'
|
||||||
|
+ inactive_pool.info.return_value = [self.mock_libvirt.VIR_STORAGE_POOL_INACTIVE]
|
||||||
|
+ inactive_pool.listAllVolumes.side_effect = self.mock_libvirt.libvirtError('pool is inactive')
|
||||||
|
+ mock_pools.append(inactive_pool)
|
||||||
|
+
|
||||||
|
self.mock_conn.listAllStoragePools.return_value = mock_pools # pylint: disable=no-member
|
||||||
|
|
||||||
|
with patch('salt.modules.virt._get_domain', MagicMock(return_value=mock_vms)):
|
||||||
|
--
|
||||||
|
2.20.1
|
||||||
|
|
||||||
|
|
37
virt.volume_infos-silence-libvirt-error-message-175.patch
Normal file
37
virt.volume_infos-silence-libvirt-error-message-175.patch
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
From fa621d17371ea6c8eff75460755c0040fcbf13de Mon Sep 17 00:00:00 2001
|
||||||
|
From: Cedric Bosdonnat <cbosdonnat@suse.com>
|
||||||
|
Date: Tue, 3 Sep 2019 15:17:46 +0200
|
||||||
|
Subject: [PATCH] virt.volume_infos: silence libvirt error message (#175)
|
||||||
|
|
||||||
|
Even though the volume_infos handles the libvirt exception when a volume
|
||||||
|
is missing, libvirt was still outputting the error message in the log.
|
||||||
|
Since this can add noise to the log only record the libvirt error
|
||||||
|
message in debug level.
|
||||||
|
---
|
||||||
|
salt/modules/virt.py | 8 +++++++-
|
||||||
|
1 file changed, 7 insertions(+), 1 deletion(-)
|
||||||
|
|
||||||
|
diff --git a/salt/modules/virt.py b/salt/modules/virt.py
|
||||||
|
index 0353e6a1f5..96c17bd60b 100644
|
||||||
|
--- a/salt/modules/virt.py
|
||||||
|
+++ b/salt/modules/virt.py
|
||||||
|
@@ -5008,8 +5008,14 @@ def _is_valid_volume(vol):
|
||||||
|
the last pool refresh.
|
||||||
|
'''
|
||||||
|
try:
|
||||||
|
- # Getting info on an invalid volume raises error
|
||||||
|
+ # Getting info on an invalid volume raises error and libvirt logs an error
|
||||||
|
+ def discarder(ctxt, error): # pylint: disable=unused-argument
|
||||||
|
+ log.debug("Ignore libvirt error: %s", error[2])
|
||||||
|
+ # Disable the libvirt error logging
|
||||||
|
+ libvirt.registerErrorHandler(discarder, None)
|
||||||
|
vol.info()
|
||||||
|
+ # Reenable the libvirt error logging
|
||||||
|
+ libvirt.registerErrorHandler(None, None)
|
||||||
|
return True
|
||||||
|
except libvirt.libvirtError as err:
|
||||||
|
return False
|
||||||
|
--
|
||||||
|
2.20.1
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user