tests: basic salt integration tests
Test base functions of dom0 module (creating VM, setting property) and configuring system inside of VM (through DispVM). The later is done for each available template (the process use salt installed in that template, not copied from dom0). QubesOS/qubes-issues#3316
This commit is contained in:
parent
3065e0de94
commit
962742880f
@ -1198,6 +1198,7 @@ def load_tests(loader, tests, pattern): # pylint: disable=unused-argument
|
|||||||
'qubes.tests.integ.network',
|
'qubes.tests.integ.network',
|
||||||
'qubes.tests.integ.dispvm',
|
'qubes.tests.integ.dispvm',
|
||||||
'qubes.tests.integ.vm_qrexec_gui',
|
'qubes.tests.integ.vm_qrexec_gui',
|
||||||
|
'qubes.tests.integ.salt',
|
||||||
'qubes.tests.integ.backup',
|
'qubes.tests.integ.backup',
|
||||||
'qubes.tests.integ.backupcompatibility',
|
'qubes.tests.integ.backupcompatibility',
|
||||||
# 'qubes.tests.regressions',
|
# 'qubes.tests.regressions',
|
||||||
|
401
qubes/tests/integ/salt.py
Normal file
401
qubes/tests/integ/salt.py
Normal file
@ -0,0 +1,401 @@
|
|||||||
|
# -*- encoding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# The Qubes OS Project, http://www.qubes-os.org
|
||||||
|
#
|
||||||
|
# Copyright (C) 2017 Marek Marczykowski-Górecki
|
||||||
|
# <marmarek@invisiblethingslab.com>
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 2 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along
|
||||||
|
# with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||||
|
import os
|
||||||
|
import subprocess
|
||||||
|
import json
|
||||||
|
|
||||||
|
import shutil
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
import qubes.tests
|
||||||
|
|
||||||
|
class SaltTestMixin(object):
|
||||||
|
def setUp(self):
|
||||||
|
super().setUp()
|
||||||
|
self.salt_testdir = '/srv/salt/test_salt'
|
||||||
|
os.makedirs(self.salt_testdir, exist_ok=True)
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
shutil.rmtree(self.salt_testdir)
|
||||||
|
try:
|
||||||
|
tops = '/srv/salt/_tops/base'
|
||||||
|
for top_link in os.listdir(tops):
|
||||||
|
path = os.path.join(tops, top_link)
|
||||||
|
target = os.readlink(path)
|
||||||
|
if target.startswith(self.salt_testdir):
|
||||||
|
os.unlink(path)
|
||||||
|
except FileNotFoundError:
|
||||||
|
pass
|
||||||
|
super().tearDown()
|
||||||
|
|
||||||
|
def salt_call(self, cmd):
|
||||||
|
full_cmd = ['qubesctl']
|
||||||
|
if '--dom0-only' in cmd:
|
||||||
|
full_cmd.insert(1, '--dom0-only')
|
||||||
|
cmd.remove('--dom0-only')
|
||||||
|
full_cmd.extend(cmd)
|
||||||
|
full_cmd.append('--out=json')
|
||||||
|
p = self.loop.run_until_complete(asyncio.create_subprocess_exec(
|
||||||
|
*full_cmd, stdout=subprocess.PIPE))
|
||||||
|
output, _ = self.loop.run_until_complete(p.communicate())
|
||||||
|
if p.returncode != 0:
|
||||||
|
raise subprocess.CalledProcessError(p.returncode, full_cmd, output)
|
||||||
|
return output.decode()
|
||||||
|
|
||||||
|
def dom0_salt_call_json(self, cmd):
|
||||||
|
return json.loads(self.salt_call(['--dom0-only'] + cmd))
|
||||||
|
|
||||||
|
|
||||||
|
class TC_00_Dom0(SaltTestMixin, qubes.tests.SystemTestCase):
|
||||||
|
def test_000_top_enable_disable(self):
|
||||||
|
with open(os.path.join(self.salt_testdir, 'something.sls'), 'w') as f:
|
||||||
|
f.write('test-top-enable:\n')
|
||||||
|
f.write(' test.succeed_without_changes: []\n')
|
||||||
|
with open(os.path.join(self.salt_testdir, 'something.top'), 'w') as f:
|
||||||
|
f.write('base:\n')
|
||||||
|
f.write(' dom0:\n')
|
||||||
|
f.write(' - test_salt.something\n')
|
||||||
|
|
||||||
|
cmd_output = self.dom0_salt_call_json(
|
||||||
|
['top.enable', 'test_salt.something'])
|
||||||
|
self.assertEqual(cmd_output,
|
||||||
|
{'local': {'test_salt.something.top': {'status': 'enabled'}}})
|
||||||
|
|
||||||
|
cmd_output = self.dom0_salt_call_json(['state.show_top'])
|
||||||
|
self.assertIn('local', cmd_output)
|
||||||
|
self.assertIn('base', cmd_output['local'])
|
||||||
|
self.assertIn('test_salt.something', cmd_output['local']['base'])
|
||||||
|
|
||||||
|
cmd_output = self.dom0_salt_call_json(
|
||||||
|
['top.disable', 'test_salt.something'])
|
||||||
|
#self.assertEqual(cmd_output,
|
||||||
|
# {'local': {'test_salt.something.top': {'status': 'disabled'}}})
|
||||||
|
|
||||||
|
cmd_output = self.dom0_salt_call_json(['state.show_top'])
|
||||||
|
self.assertIn('local', cmd_output)
|
||||||
|
self.assertIn('base', cmd_output['local'])
|
||||||
|
self.assertNotIn('test_salt.something', cmd_output['local']['base'])
|
||||||
|
|
||||||
|
def test_001_state_sls(self):
|
||||||
|
with open(os.path.join(self.salt_testdir, 'something.sls'), 'w') as f:
|
||||||
|
f.write('test-top-enable:\n')
|
||||||
|
f.write(' test.succeed_without_changes: []\n')
|
||||||
|
|
||||||
|
cmd_output = self.dom0_salt_call_json(
|
||||||
|
['state.sls', 'test_salt.something'])
|
||||||
|
state_id = 'test_|-test-top-enable_|-test-top-enable_|-succeed_without_changes'
|
||||||
|
self.assertIn('local', cmd_output)
|
||||||
|
self.assertIn(state_id, cmd_output['local'])
|
||||||
|
self.assertIn('start_time', cmd_output['local'][state_id])
|
||||||
|
del cmd_output['local'][state_id]['start_time']
|
||||||
|
self.assertIn('duration', cmd_output['local'][state_id])
|
||||||
|
del cmd_output['local'][state_id]['duration']
|
||||||
|
self.assertEqual(cmd_output,
|
||||||
|
{'local': {state_id: {
|
||||||
|
'name': 'test-top-enable',
|
||||||
|
'comment': 'Success!',
|
||||||
|
'result': True,
|
||||||
|
'__run_num__': 0,
|
||||||
|
'__sls__': 'test_salt.something',
|
||||||
|
'changes': {},
|
||||||
|
'__id__': 'test-top-enable'
|
||||||
|
}}})
|
||||||
|
|
||||||
|
def test_010_create_vm(self):
|
||||||
|
vmname = self.make_vm_name('appvm')
|
||||||
|
with open(os.path.join(self.salt_testdir, 'create_vm.sls'), 'w') as f:
|
||||||
|
f.write(vmname + ':\n')
|
||||||
|
f.write(' qvm.vm:\n')
|
||||||
|
f.write(' - present:\n')
|
||||||
|
f.write(' - label: orange\n')
|
||||||
|
f.write(' - prefs:\n')
|
||||||
|
f.write(' - vcpus: 1\n')
|
||||||
|
cmd_output = self.dom0_salt_call_json(
|
||||||
|
['state.sls', 'test_salt.create_vm'])
|
||||||
|
state_out = list(cmd_output['local'].values())[0]
|
||||||
|
del state_out['start_time']
|
||||||
|
del state_out['duration']
|
||||||
|
self.assertEqual(state_out, {
|
||||||
|
'comment': '====== [\'present\'] ======\n'
|
||||||
|
'/usr/bin/qvm-create {} --class=AppVM --label=orange \n'
|
||||||
|
'\n'
|
||||||
|
'====== [\'prefs\'] ======\n'.format(vmname),
|
||||||
|
'name': vmname,
|
||||||
|
'result': True,
|
||||||
|
'changes': {
|
||||||
|
'qvm.prefs': {'qvm.create': {
|
||||||
|
'vcpus': {'new': 1, 'old': '*default*'}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'__sls__': 'test_salt.create_vm',
|
||||||
|
'__run_num__': 0,
|
||||||
|
'__id__': vmname,
|
||||||
|
})
|
||||||
|
|
||||||
|
self.assertIn(vmname, self.app.domains)
|
||||||
|
vm = self.app.domains[vmname]
|
||||||
|
self.assertEqual(str(vm.label), 'orange')
|
||||||
|
self.assertEqual(vm.vcpus, 1)
|
||||||
|
|
||||||
|
def test_011_set_prefs(self):
|
||||||
|
vmname = self.make_vm_name('appvm')
|
||||||
|
|
||||||
|
vm = self.app.add_new_vm('AppVM', label='red',
|
||||||
|
name=vmname)
|
||||||
|
self.loop.run_until_complete(vm.create_on_disk())
|
||||||
|
|
||||||
|
with open(os.path.join(self.salt_testdir, 'create_vm.sls'), 'w') as f:
|
||||||
|
f.write(vmname + ':\n')
|
||||||
|
f.write(' qvm.vm:\n')
|
||||||
|
f.write(' - present:\n')
|
||||||
|
f.write(' - label: orange\n')
|
||||||
|
f.write(' - prefs:\n')
|
||||||
|
f.write(' - vcpus: 1\n')
|
||||||
|
cmd_output = self.dom0_salt_call_json(
|
||||||
|
['state.sls', 'test_salt.create_vm'])
|
||||||
|
state_out = list(cmd_output['local'].values())[0]
|
||||||
|
del state_out['start_time']
|
||||||
|
del state_out['duration']
|
||||||
|
self.assertEqual(state_out, {
|
||||||
|
'comment': '====== [\'present\'] ======\n'
|
||||||
|
'[SKIP] A VM with the name \'{}\' already exists.\n'
|
||||||
|
'\n'
|
||||||
|
'====== [\'prefs\'] ======\n'.format(vmname),
|
||||||
|
'name': vmname,
|
||||||
|
'result': True,
|
||||||
|
'changes': {
|
||||||
|
'qvm.prefs': {'qvm.create': {
|
||||||
|
'vcpus': {'new': 1, 'old': '*default*'}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'__sls__': 'test_salt.create_vm',
|
||||||
|
'__run_num__': 0,
|
||||||
|
'__id__': vmname,
|
||||||
|
})
|
||||||
|
|
||||||
|
self.assertIn(vmname, self.app.domains)
|
||||||
|
vm = self.app.domains[vmname]
|
||||||
|
self.assertEqual(str(vm.label), 'red')
|
||||||
|
self.assertEqual(vm.vcpus, 1)
|
||||||
|
|
||||||
|
def test_012_tags(self):
|
||||||
|
vmname = self.make_vm_name('appvm')
|
||||||
|
|
||||||
|
vm = self.app.add_new_vm('AppVM', label='red',
|
||||||
|
name=vmname)
|
||||||
|
self.loop.run_until_complete(vm.create_on_disk())
|
||||||
|
vm.tags.add('tag1')
|
||||||
|
vm.tags.add('tag3')
|
||||||
|
|
||||||
|
with open(os.path.join(self.salt_testdir, 'test_state.sls'), 'w') as f:
|
||||||
|
f.write(vmname + ':\n')
|
||||||
|
f.write(' qvm.vm:\n')
|
||||||
|
f.write(' - tags:\n')
|
||||||
|
f.write(' - present:\n')
|
||||||
|
f.write(' - tag1\n')
|
||||||
|
f.write(' - tag2\n')
|
||||||
|
f.write(' - absent:\n')
|
||||||
|
f.write(' - tag3\n')
|
||||||
|
f.write(' - tag4\n')
|
||||||
|
cmd_output = self.dom0_salt_call_json(
|
||||||
|
['state.sls', 'test_salt.test_state'])
|
||||||
|
state_out = list(cmd_output['local'].values())[0]
|
||||||
|
del state_out['start_time']
|
||||||
|
del state_out['duration']
|
||||||
|
self.assertEqual(state_out, {
|
||||||
|
'comment': '====== [\'tags\'] ======\n',
|
||||||
|
'name': vmname,
|
||||||
|
'result': True,
|
||||||
|
'changes': {
|
||||||
|
'qvm.tags': {'qvm.tags': {
|
||||||
|
'new': ['tag1', 'tag2'], 'old': ['tag1', 'tag3'],
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'__sls__': 'test_salt.test_state',
|
||||||
|
'__run_num__': 0,
|
||||||
|
'__id__': vmname,
|
||||||
|
})
|
||||||
|
|
||||||
|
self.assertIn(vmname, self.app.domains)
|
||||||
|
vm = self.app.domains[vmname]
|
||||||
|
self.assertEqual(set(vm.tags), {'tag1', 'tag2'})
|
||||||
|
|
||||||
|
def test_020_qubes_pillar(self):
|
||||||
|
vmname = self.make_vm_name('appvm')
|
||||||
|
|
||||||
|
vm = self.app.add_new_vm('AppVM', label='red',
|
||||||
|
name=vmname)
|
||||||
|
self.loop.run_until_complete(vm.create_on_disk())
|
||||||
|
|
||||||
|
cmd_output = self.dom0_salt_call_json(
|
||||||
|
['pillar.items', '--id=' + vmname])
|
||||||
|
self.assertIn('local', cmd_output)
|
||||||
|
self.assertIn('qubes', cmd_output['local'])
|
||||||
|
qubes_pillar = cmd_output['local']['qubes']
|
||||||
|
self.assertEqual(qubes_pillar, {
|
||||||
|
'type': 'app',
|
||||||
|
'netvm': str(vm.netvm),
|
||||||
|
'template': str(vm.template),
|
||||||
|
})
|
||||||
|
|
||||||
|
class SaltVMTestMixin(SaltTestMixin):
|
||||||
|
template = None
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
if self.template.startswith('whonix'):
|
||||||
|
self.skipTest('Whonix not supported as salt VM')
|
||||||
|
super(SaltVMTestMixin, self).setUp()
|
||||||
|
self.init_default_template(self.template)
|
||||||
|
|
||||||
|
dispvm_tpl_name = self.make_vm_name('disp-tpl')
|
||||||
|
dispvm_tpl = self.app.add_new_vm('AppVM', label='red',
|
||||||
|
template_for_dispvms=True, name=dispvm_tpl_name)
|
||||||
|
self.loop.run_until_complete(dispvm_tpl.create_on_disk())
|
||||||
|
self.app.default_dispvm = dispvm_tpl
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
self.app.default_dispvm = None
|
||||||
|
super(SaltVMTestMixin, self).tearDown()
|
||||||
|
|
||||||
|
def test_000_simple_sls(self):
|
||||||
|
vmname = self.make_vm_name('target')
|
||||||
|
self.vm = self.app.add_new_vm('AppVM', name=vmname, label='red')
|
||||||
|
self.loop.run_until_complete(self.vm.create_on_disk())
|
||||||
|
# start the VM manually, so it stays running after applying salt state
|
||||||
|
self.loop.run_until_complete(self.vm.start())
|
||||||
|
with open(os.path.join(self.salt_testdir, 'something.sls'), 'w') as f:
|
||||||
|
f.write('/home/user/testfile:\n')
|
||||||
|
f.write(' file.managed:\n')
|
||||||
|
f.write(' - contents: |\n')
|
||||||
|
f.write(' this is test\n')
|
||||||
|
with open(os.path.join(self.salt_testdir, 'something.top'), 'w') as f:
|
||||||
|
f.write('base:\n')
|
||||||
|
f.write(' {}:\n'.format(vmname))
|
||||||
|
f.write(' - test_salt.something\n')
|
||||||
|
|
||||||
|
# enable so state.show_top will not be empty, otherwise qubesctl will
|
||||||
|
# skip the VM; but we don't use state.highstate
|
||||||
|
self.dom0_salt_call_json(['top.enable', 'test_salt.something'])
|
||||||
|
state_output = self.salt_call(
|
||||||
|
['--skip-dom0', '--show-output', '--targets=' + vmname,
|
||||||
|
'state.sls', 'test_salt.something'])
|
||||||
|
expected_output = vmname + ':\n'
|
||||||
|
self.assertTrue(state_output.startswith(expected_output),
|
||||||
|
'Full output: ' + state_output)
|
||||||
|
state_id = 'file_|-/home/user/testfile_|-/home/user/testfile_|-managed'
|
||||||
|
# drop the header
|
||||||
|
state_output_json = json.loads(state_output[len(expected_output):])
|
||||||
|
state_output_json = state_output_json[vmname][state_id]
|
||||||
|
try:
|
||||||
|
del state_output_json['duration']
|
||||||
|
del state_output_json['start_time']
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
del state_output_json['pchanges']
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
# older salt do not report this
|
||||||
|
self.assertEqual(state_output_json['__id__'], '/home/user/testfile')
|
||||||
|
del state_output_json['__id__']
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
# or sls file
|
||||||
|
self.assertEqual(state_output_json['__sls__'],
|
||||||
|
'test_salt.something')
|
||||||
|
del state_output_json['__sls__']
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
# different output depending on salt version
|
||||||
|
expected_output = {
|
||||||
|
'__run_num__': 0,
|
||||||
|
'changes': {
|
||||||
|
'diff': 'New file',
|
||||||
|
},
|
||||||
|
'name': '/home/user/testfile',
|
||||||
|
'comment': 'File /home/user/testfile updated',
|
||||||
|
'result': True,
|
||||||
|
}
|
||||||
|
self.assertEqual(state_output_json, expected_output)
|
||||||
|
stdout, stderr = self.loop.run_until_complete(self.vm.run_for_stdio(
|
||||||
|
'cat /home/user/testfile'))
|
||||||
|
self.assertEqual(stdout, b'this is test\n')
|
||||||
|
self.assertEqual(stderr, b'')
|
||||||
|
|
||||||
|
def test_001_multi_state_highstate(self):
|
||||||
|
vmname = self.make_vm_name('target')
|
||||||
|
self.vm = self.app.add_new_vm('AppVM', name=vmname, label='red')
|
||||||
|
self.loop.run_until_complete(self.vm.create_on_disk())
|
||||||
|
# start the VM manually, so it stays running after applying salt state
|
||||||
|
self.loop.run_until_complete(self.vm.start())
|
||||||
|
states = ('something', 'something2')
|
||||||
|
for state in states:
|
||||||
|
with open(os.path.join(self.salt_testdir, state + '.sls'), 'w') as f:
|
||||||
|
f.write('/home/user/{}:\n'.format(state))
|
||||||
|
f.write(' file.managed:\n')
|
||||||
|
f.write(' - contents: |\n')
|
||||||
|
f.write(' this is test\n')
|
||||||
|
with open(os.path.join(self.salt_testdir, state + '.top'), 'w') as f:
|
||||||
|
f.write('base:\n')
|
||||||
|
f.write(' {}:\n'.format(vmname))
|
||||||
|
f.write(' - test_salt.{}\n'.format(state))
|
||||||
|
|
||||||
|
self.dom0_salt_call_json(['top.enable', 'test_salt.something'])
|
||||||
|
self.dom0_salt_call_json(['top.enable', 'test_salt.something2'])
|
||||||
|
state_output = self.salt_call(
|
||||||
|
['--skip-dom0', '--show-output', '--targets=' + vmname,
|
||||||
|
'state.highstate'])
|
||||||
|
expected_output = vmname + ':\n'
|
||||||
|
self.assertTrue(state_output.startswith(expected_output),
|
||||||
|
'Full output: ' + state_output)
|
||||||
|
state_output_json = json.loads(state_output[len(expected_output):])
|
||||||
|
for state in states:
|
||||||
|
state_id = \
|
||||||
|
'file_|-/home/user/{0}_|-/home/user/{0}_|-managed'.format(state)
|
||||||
|
# drop the header
|
||||||
|
self.assertIn(state_id, state_output_json[vmname])
|
||||||
|
state_output_single = state_output_json[vmname][state_id]
|
||||||
|
|
||||||
|
self.assertTrue(state_output_single['result'])
|
||||||
|
self.assertNotEqual(state_output_single['changes'], {})
|
||||||
|
|
||||||
|
stdout, stderr = self.loop.run_until_complete(self.vm.run_for_stdio(
|
||||||
|
'cat /home/user/' + state))
|
||||||
|
self.assertEqual(stdout, b'this is test\n')
|
||||||
|
self.assertEqual(stderr, b'')
|
||||||
|
|
||||||
|
|
||||||
|
def load_tests(loader, tests, pattern):
|
||||||
|
for template in qubes.tests.list_templates():
|
||||||
|
tests.addTests(loader.loadTestsFromTestCase(
|
||||||
|
type(
|
||||||
|
'TC_10_VMSalt_' + template,
|
||||||
|
(SaltVMTestMixin, qubes.tests.SystemTestCase),
|
||||||
|
{'template': template})))
|
||||||
|
return tests
|
@ -348,6 +348,7 @@ fi
|
|||||||
%{python3_sitelib}/qubes/tests/integ/dispvm.py
|
%{python3_sitelib}/qubes/tests/integ/dispvm.py
|
||||||
%{python3_sitelib}/qubes/tests/integ/dom0_update.py
|
%{python3_sitelib}/qubes/tests/integ/dom0_update.py
|
||||||
%{python3_sitelib}/qubes/tests/integ/network.py
|
%{python3_sitelib}/qubes/tests/integ/network.py
|
||||||
|
%{python3_sitelib}/qubes/tests/integ/salt.py
|
||||||
%{python3_sitelib}/qubes/tests/integ/storage.py
|
%{python3_sitelib}/qubes/tests/integ/storage.py
|
||||||
%{python3_sitelib}/qubes/tests/integ/vm_qrexec_gui.py
|
%{python3_sitelib}/qubes/tests/integ/vm_qrexec_gui.py
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user