qubesvm.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. # pylint: disable=protected-access
  2. #
  3. # The Qubes OS Project, https://www.qubes-os.org/
  4. #
  5. # Copyright (C) 2014-2015 Joanna Rutkowska <joanna@invisiblethingslab.com>
  6. # Copyright (C) 2014-2015 Wojtek Porczyk <woju@invisiblethingslab.com>
  7. #
  8. # This program is free software; you can redistribute it and/or modify
  9. # it under the terms of the GNU General Public License as published by
  10. # the Free Software Foundation; either version 2 of the License, or
  11. # (at your option) any later version.
  12. #
  13. # This program is distributed in the hope that it will be useful,
  14. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. # GNU General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU General Public License along
  19. # with this program; if not, write to the Free Software Foundation, Inc.,
  20. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  21. #
  22. import os
  23. import unittest
  24. import uuid
  25. import datetime
  26. import qubes
  27. import qubes.exc
  28. import qubes.config
  29. import qubes.vm
  30. import qubes.vm.qubesvm
  31. import qubes.tests
  32. import qubes.tests.vm
  33. class TestApp(object):
  34. labels = {1: qubes.Label(1, '0xcc0000', 'red')}
  35. def __init__(self):
  36. self.domains = {}
  37. class TestProp(object):
  38. # pylint: disable=too-few-public-methods
  39. __name__ = 'testprop'
  40. class TestVM(object):
  41. # pylint: disable=too-few-public-methods
  42. app = TestApp()
  43. def __init__(self, **kwargs):
  44. self.running = False
  45. self.installed_by_rpm = False
  46. for k, v in kwargs.items():
  47. setattr(self, k, v)
  48. def is_running(self):
  49. return self.running
  50. class TC_00_setters(qubes.tests.QubesTestCase):
  51. def setUp(self):
  52. super().setUp()
  53. self.vm = TestVM()
  54. self.prop = TestProp()
  55. def test_000_setter_qid(self):
  56. self.assertEqual(
  57. qubes.vm.qubesvm._setter_qid(self.vm, self.prop, 5), 5)
  58. def test_001_setter_qid_lt_0(self):
  59. with self.assertRaises(ValueError):
  60. qubes.vm.qubesvm._setter_qid(self.vm, self.prop, -1)
  61. def test_002_setter_qid_gt_max(self):
  62. with self.assertRaises(ValueError):
  63. qubes.vm.qubesvm._setter_qid(self.vm,
  64. self.prop, qubes.config.max_qid + 5)
  65. def test_010_setter_name(self):
  66. self.assertEqual(
  67. qubes.vm.qubesvm._setter_name(self.vm, self.prop, 'test_name-1'),
  68. 'test_name-1')
  69. def test_011_setter_name_not_a_string(self):
  70. # pylint: disable=invalid-name
  71. with self.assertRaises(TypeError):
  72. qubes.vm.qubesvm._setter_name(self.vm, self.prop, False)
  73. def test_012_setter_name_longer_than_31(self):
  74. # pylint: disable=invalid-name
  75. with self.assertRaises(ValueError):
  76. qubes.vm.qubesvm._setter_name(self.vm, self.prop, 't' * 32)
  77. def test_013_setter_name_illegal_character(self):
  78. # pylint: disable=invalid-name
  79. with self.assertRaises(ValueError):
  80. qubes.vm.qubesvm._setter_name(self.vm, self.prop, 'test#')
  81. def test_014_setter_name_first_not_letter(self):
  82. # pylint: disable=invalid-name
  83. with self.assertRaises(ValueError):
  84. qubes.vm.qubesvm._setter_name(self.vm, self.prop, '1test')
  85. def test_015_setter_name_running(self):
  86. self.vm.running = True
  87. with self.assertRaises(qubes.exc.QubesVMNotHaltedError):
  88. qubes.vm.qubesvm._setter_name(self.vm, self.prop, 'testname')
  89. def test_016_setter_name_installed_by_rpm(self):
  90. # pylint: disable=invalid-name
  91. self.vm.installed_by_rpm = True
  92. with self.assertRaises(qubes.exc.QubesException):
  93. qubes.vm.qubesvm._setter_name(self.vm, self.prop, 'testname')
  94. def test_017_setter_name_duplicate(self):
  95. # pylint: disable=invalid-name
  96. self.vm.app.domains['duplicate'] = TestVM(name='duplicate')
  97. with self.assertRaises(qubes.exc.QubesException):
  98. qubes.vm.qubesvm._setter_name(self.vm, self.prop, 'duplicate')
  99. @unittest.skip('test not implemented')
  100. def test_020_setter_kernel(self):
  101. pass
  102. def test_030_setter_label_object(self):
  103. label = TestApp.labels[1]
  104. self.assertIs(label,
  105. qubes.vm.setter_label(self.vm, self.prop, label))
  106. def test_031_setter_label_getitem(self):
  107. label = TestApp.labels[1]
  108. self.assertIs(label,
  109. qubes.vm.setter_label(self.vm, self.prop, 'label-1'))
  110. # there is no check for self.app.get_label()
  111. class QubesVMTestsMixin(object):
  112. property_no_default = object()
  113. def setUp(self):
  114. super(QubesVMTestsMixin, self).setUp()
  115. self.app = qubes.tests.vm.TestApp()
  116. self.app.vmm.offline_mode = True
  117. def get_vm(self, **kwargs):
  118. return qubes.vm.qubesvm.QubesVM(self.app, None,
  119. qid=1, name=qubes.tests.VMPREFIX + 'test',
  120. **kwargs)
  121. def assertPropertyValue(self, vm, prop_name, set_value, expected_value,
  122. expected_xml_content=None):
  123. # FIXME: any better exception list? or maybe all of that should be a
  124. # single exception?
  125. with self.assertNotRaises((ValueError, TypeError, KeyError)):
  126. setattr(vm, prop_name, set_value)
  127. self.assertEqual(getattr(vm, prop_name), expected_value)
  128. if expected_xml_content is not None:
  129. xml = vm.__xml__()
  130. prop_xml = xml.xpath(
  131. './properties/property[@name=\'{}\']'.format(prop_name))
  132. self.assertEqual(len(prop_xml), 1, "Property not found in XML")
  133. self.assertEqual(prop_xml[0].text, expected_xml_content)
  134. def assertPropertyInvalidValue(self, vm, prop_name, set_value):
  135. orig_value_set = True
  136. orig_value = None
  137. try:
  138. orig_value = getattr(vm, prop_name)
  139. except AttributeError:
  140. orig_value_set = False
  141. # FIXME: any better exception list? or maybe all of that should be a
  142. # single exception?
  143. with self.assertRaises((ValueError, TypeError, KeyError)):
  144. setattr(vm, prop_name, set_value)
  145. if orig_value_set:
  146. self.assertEqual(getattr(vm, prop_name), orig_value)
  147. else:
  148. with self.assertRaises(AttributeError):
  149. getattr(vm, prop_name)
  150. def assertPropertyDefaultValue(self, vm, prop_name,
  151. expected_default=property_no_default):
  152. if expected_default is self.property_no_default:
  153. with self.assertRaises(AttributeError):
  154. getattr(vm, prop_name)
  155. else:
  156. with self.assertNotRaises(AttributeError):
  157. self.assertEqual(getattr(vm, prop_name), expected_default)
  158. xml = vm.__xml__()
  159. prop_xml = xml.xpath(
  160. './properties/property[@name=\'{}\']'.format(prop_name))
  161. self.assertEqual(len(prop_xml), 0, "Property still found in XML")
  162. def _test_generic_bool_property(self, vm, prop_name, default=False):
  163. self.assertPropertyDefaultValue(vm, prop_name, default)
  164. self.assertPropertyValue(vm, prop_name, False, False, 'False')
  165. self.assertPropertyValue(vm, prop_name, True, True, 'True')
  166. delattr(vm, prop_name)
  167. self.assertPropertyDefaultValue(vm, prop_name, default)
  168. self.assertPropertyValue(vm, prop_name, 'True', True, 'True')
  169. self.assertPropertyValue(vm, prop_name, 'False', False, 'False')
  170. self.assertPropertyInvalidValue(vm, prop_name, 'xxx')
  171. self.assertPropertyValue(vm, prop_name, 123, True)
  172. self.assertPropertyInvalidValue(vm, prop_name, '')
  173. class TC_90_QubesVM(QubesVMTestsMixin, qubes.tests.QubesTestCase):
  174. def test_000_init(self):
  175. self.get_vm()
  176. def test_001_init_no_qid_or_name(self):
  177. with self.assertRaises(AssertionError):
  178. qubes.vm.qubesvm.QubesVM(self.app, None,
  179. name=qubes.tests.VMPREFIX + 'test')
  180. with self.assertRaises(AssertionError):
  181. qubes.vm.qubesvm.QubesVM(self.app, None,
  182. qid=1)
  183. def test_003_init_fire_domain_init(self):
  184. class TestVM2(qubes.vm.qubesvm.QubesVM):
  185. event_fired = False
  186. @qubes.events.handler('domain-init')
  187. def on_domain_init(self, event): # pylint: disable=unused-argument
  188. self.__class__.event_fired = True
  189. TestVM2(self.app, None, qid=1, name=qubes.tests.VMPREFIX + 'test')
  190. self.assertTrue(TestVM2.event_fired)
  191. def test_004_uuid_autogen(self):
  192. vm = self.get_vm()
  193. self.assertTrue(hasattr(vm, 'uuid'))
  194. def test_100_qid(self):
  195. vm = self.get_vm()
  196. self.assertIsInstance(vm.qid, int)
  197. with self.assertRaises(AttributeError):
  198. vm.qid = 2
  199. def test_110_name(self):
  200. vm = self.get_vm()
  201. self.assertIsInstance(vm.name, str)
  202. def test_120_uuid(self):
  203. my_uuid = uuid.uuid4()
  204. vm = self.get_vm(uuid=my_uuid)
  205. self.assertIsInstance(vm.uuid, uuid.UUID)
  206. self.assertIs(vm.uuid, my_uuid)
  207. with self.assertRaises(AttributeError):
  208. vm.uuid = uuid.uuid4()
  209. @unittest.skip('TODO: how to not fail on making an icon symlink here?')
  210. def test_130_label(self):
  211. vm = self.get_vm()
  212. self.assertPropertyDefaultValue(vm, 'label')
  213. self.assertPropertyValue(vm, 'label', self.app.labels[1],
  214. self.app.labels[1], 'label-1')
  215. del vm.label
  216. self.assertPropertyDefaultValue(vm, 'label')
  217. self.assertPropertyValue(vm, 'label', 'red',
  218. self.app.labels[1], 'label-1')
  219. self.assertPropertyValue(vm, 'label', 'label-1',
  220. self.app.labels[1], 'label-1')
  221. def test_131_label_invalid(self):
  222. vm = self.get_vm()
  223. self.assertPropertyInvalidValue(vm, 'label', 'invalid')
  224. self.assertPropertyInvalidValue(vm, 'label', 123)
  225. def test_150_hvm(self):
  226. vm = self.get_vm()
  227. self._test_generic_bool_property(vm, 'hvm', default=True)
  228. def test_160_memory(self):
  229. vm = self.get_vm()
  230. self.assertPropertyDefaultValue(vm, 'memory', 400)
  231. self.assertPropertyValue(vm, 'memory', 500, 500, '500')
  232. del vm.memory
  233. self.assertPropertyDefaultValue(vm, 'memory', 400)
  234. self.assertPropertyValue(vm, 'memory', '500', 500, '500')
  235. def test_161_memory_invalid(self):
  236. vm = self.get_vm()
  237. self.assertPropertyInvalidValue(vm, 'memory', -100)
  238. self.assertPropertyInvalidValue(vm, 'memory', '-100')
  239. self.assertPropertyInvalidValue(vm, 'memory', '')
  240. # TODO: higher than maxmem
  241. # TODO: human readable setter (500M, 4G)?
  242. def test_170_maxmem(self):
  243. vm = self.get_vm()
  244. self.assertPropertyDefaultValue(vm, 'maxmem',
  245. self.app.host.memory_total / 1024 / 2)
  246. self.assertPropertyValue(vm, 'maxmem', 500, 500, '500')
  247. del vm.maxmem
  248. self.assertPropertyDefaultValue(vm, 'maxmem',
  249. self.app.host.memory_total / 1024 / 2)
  250. self.assertPropertyValue(vm, 'maxmem', '500', 500, '500')
  251. def test_171_maxmem_invalid(self):
  252. vm = self.get_vm()
  253. self.assertPropertyInvalidValue(vm, 'maxmem', -100)
  254. self.assertPropertyInvalidValue(vm, 'maxmem', '-100')
  255. self.assertPropertyInvalidValue(vm, 'maxmem', '')
  256. # TODO: lower than memory
  257. # TODO: human readable setter (500M, 4G)?
  258. def test_190_vcpus(self):
  259. vm = self.get_vm()
  260. self.assertPropertyDefaultValue(vm, 'vcpus', self.app.host.no_cpus)
  261. self.assertPropertyValue(vm, 'vcpus', 3, 3, '3')
  262. del vm.vcpus
  263. self.assertPropertyDefaultValue(vm, 'vcpus', self.app.host.no_cpus)
  264. self.assertPropertyValue(vm, 'vcpus', '3', 3, '3')
  265. def test_191_vcpus_invalid(self):
  266. vm = self.get_vm()
  267. self.assertPropertyInvalidValue(vm, 'vcpus', 0)
  268. self.assertPropertyInvalidValue(vm, 'vcpus', -2)
  269. self.assertPropertyInvalidValue(vm, 'vcpus', '-2')
  270. self.assertPropertyInvalidValue(vm, 'vcpus', '')
  271. def test_200_debug(self):
  272. vm = self.get_vm()
  273. self._test_generic_bool_property(vm, 'debug', False)
  274. def test_210_installed_by_rpm(self):
  275. vm = self.get_vm()
  276. self._test_generic_bool_property(vm, 'installed_by_rpm', False)
  277. def test_220_include_in_backups(self):
  278. vm = self.get_vm()
  279. self._test_generic_bool_property(vm, 'include_in_backups', True)
  280. @qubes.tests.skipUnlessDom0
  281. def test_250_kernel(self):
  282. kernels = os.listdir(os.path.join(
  283. qubes.config.qubes_base_dir,
  284. qubes.config.system_path['qubes_kernels_base_dir']))
  285. if not len(kernels):
  286. self.skipTest('Needs at least one kernel installed')
  287. self.app.default_kernel = kernels[0]
  288. vm = self.get_vm()
  289. self.assertPropertyDefaultValue(vm, 'kernel', kernels[0])
  290. self.assertPropertyValue(vm, 'kernel', kernels[-1], kernels[-1],
  291. kernels[-1])
  292. del vm.kernel
  293. self.assertPropertyDefaultValue(vm, 'kernel', kernels[0])
  294. @qubes.tests.skipUnlessDom0
  295. def test_251_kernel_invalid(self):
  296. vm = self.get_vm()
  297. self.assertPropertyInvalidValue(vm, 'kernel', 123)
  298. self.assertPropertyInvalidValue(vm, 'kernel', 'invalid')
  299. def test_260_kernelopts(self):
  300. vm = self.get_vm()
  301. self.assertPropertyDefaultValue(vm, 'kernelopts',
  302. qubes.config.defaults['kernelopts'])
  303. self.assertPropertyValue(vm, 'kernelopts', 'some options',
  304. 'some options', 'some options')
  305. del vm.kernelopts
  306. self.assertPropertyDefaultValue(vm, 'kernelopts',
  307. qubes.config.defaults['kernelopts'])
  308. self.assertPropertyValue(vm, 'kernelopts', '',
  309. '', '')
  310. # TODO?
  311. # self.assertPropertyInvalidValue(vm, 'kernelopts', None),
  312. @unittest.skip('test not implemented')
  313. def test_261_kernelopts_pcidevs(self):
  314. vm = self.get_vm()
  315. # how to do that here? use dummy DeviceManager/DeviceCollection?
  316. # Disable events?
  317. vm.devices['pci'].attach('something')
  318. self.assertPropertyDefaultValue(vm, 'kernelopts',
  319. qubes.config.defaults['kernelopts_pcidevs'])
  320. def test_270_qrexec_timeout(self):
  321. vm = self.get_vm()
  322. self.assertPropertyDefaultValue(vm, 'qrexec_timeout', 60)
  323. self.assertPropertyValue(vm, 'qrexec_timeout', 3, 3, '3')
  324. del vm.qrexec_timeout
  325. self.assertPropertyDefaultValue(vm, 'qrexec_timeout', 60)
  326. self.assertPropertyValue(vm, 'qrexec_timeout', '3', 3, '3')
  327. def test_271_qrexec_timeout_invalid(self):
  328. vm = self.get_vm()
  329. self.assertPropertyInvalidValue(vm, 'qrexec_timeout', -2)
  330. self.assertPropertyInvalidValue(vm, 'qrexec_timeout', '-2')
  331. self.assertPropertyInvalidValue(vm, 'qrexec_timeout', '')
  332. def test_280_autostart(self):
  333. vm = self.get_vm()
  334. # FIXME any better idea to not involve systemctl call at this stage?
  335. vm.events_enabled = False
  336. self._test_generic_bool_property(vm, 'autostart', False)
  337. @qubes.tests.skipUnlessDom0
  338. def test_281_autostart_systemd(self):
  339. vm = self.get_vm()
  340. self.assertFalse(os.path.exists(
  341. '/etc/systemd/system/multi-user.target.wants/'
  342. 'qubes-vm@{}.service'.format(vm.name)),
  343. "systemd service enabled before setting autostart")
  344. vm.autostart = True
  345. self.assertTrue(os.path.exists(
  346. '/etc/systemd/system/multi-user.target.wants/'
  347. 'qubes-vm@{}.service'.format(vm.name)),
  348. "systemd service not enabled by autostart=True")
  349. vm.autostart = False
  350. self.assertFalse(os.path.exists(
  351. '/etc/systemd/system/multi-user.target.wants/'
  352. 'qubes-vm@{}.service'.format(vm.name)),
  353. "systemd service not disabled by autostart=False")
  354. vm.autostart = True
  355. del vm.autostart
  356. self.assertFalse(os.path.exists(
  357. '/etc/systemd/system/multi-user.target.wants/'
  358. 'qubes-vm@{}.service'.format(vm.name)),
  359. "systemd service not disabled by resetting autostart")
  360. @unittest.skip('TODO')
  361. def test_320_seamless_gui_mode(self):
  362. vm = self.get_vm()
  363. self._test_generic_bool_property(vm, 'seamless_gui_mode')
  364. # TODO: reject setting to True when guiagent_installed is false
  365. def test_330_mac(self):
  366. vm = self.get_vm()
  367. # TODO: calculate proper default here
  368. default_mac = vm.mac
  369. self.assertIsNotNone(default_mac)
  370. self.assertPropertyDefaultValue(vm, 'mac', default_mac)
  371. self.assertPropertyValue(vm, 'mac', '00:11:22:33:44:55',
  372. '00:11:22:33:44:55', '00:11:22:33:44:55')
  373. del vm.mac
  374. self.assertPropertyDefaultValue(vm, 'mac', default_mac)
  375. def test_331_mac_invalid(self):
  376. vm = self.get_vm()
  377. self.assertPropertyInvalidValue(vm, 'mac', 123)
  378. self.assertPropertyInvalidValue(vm, 'mac', 'invalid')
  379. self.assertPropertyInvalidValue(vm, 'mac', '00:11:22:33:44:55:66')
  380. def test_340_default_user(self):
  381. vm = self.get_vm()
  382. self.assertPropertyDefaultValue(vm, 'default_user', 'user')
  383. self.assertPropertyValue(vm, 'default_user', 'someuser', 'someuser',
  384. 'someuser')
  385. del vm.default_user
  386. self.assertPropertyDefaultValue(vm, 'default_user', 'user')
  387. self.assertPropertyValue(vm, 'default_user', 123, '123', '123')
  388. # TODO: check propagation for template-based VMs
  389. @unittest.skip('TODO')
  390. def test_350_timezone(self):
  391. vm = self.get_vm()
  392. self.assertPropertyDefaultValue(vm, 'timezone', 'localtime')
  393. self.assertPropertyValue(vm, 'timezone', 0, 0, '0')
  394. del vm.timezone
  395. self.assertPropertyDefaultValue(vm, 'timezone', 'localtime')
  396. self.assertPropertyValue(vm, 'timezone', '0', 0, '0')
  397. self.assertPropertyValue(vm, 'timezone', -3600, -3600, '-3600')
  398. self.assertPropertyValue(vm, 'timezone', 7200, 7200, '7200')
  399. @unittest.skip('TODO')
  400. def test_350_timezone_invalid(self):
  401. vm = self.get_vm()
  402. self.assertPropertyInvalidValue(vm, 'timezone', 'xxx')
  403. @unittest.skip('TODO')
  404. def test_360_drive(self):
  405. vm = self.get_vm()
  406. self.assertPropertyDefaultValue(vm, 'drive', None)
  407. # self.execute_tests('drive', [
  408. # ('hd:dom0:/tmp/drive.img', 'hd:dom0:/tmp/drive.img', True),
  409. # ('hd:/tmp/drive.img', 'hd:dom0:/tmp/drive.img', True),
  410. # ('cdrom:dom0:/tmp/drive.img', 'cdrom:dom0:/tmp/drive.img', True),
  411. # ('cdrom:/tmp/drive.img', 'cdrom:dom0:/tmp/drive.img', True),
  412. # ('/tmp/drive.img', 'cdrom:dom0:/tmp/drive.img', True),
  413. # ('hd:drive.img', '', False),
  414. # ('drive.img', '', False),
  415. # ])
  416. def test_400_backup_timestamp(self):
  417. vm = self.get_vm()
  418. timestamp = datetime.datetime(2016, 1, 1, 12, 14, 2)
  419. timestamp_str = timestamp.strftime('%s')
  420. self.assertPropertyDefaultValue(vm, 'backup_timestamp', None)
  421. self.assertPropertyValue(vm, 'backup_timestamp', timestamp,
  422. timestamp, timestamp_str)
  423. del vm.backup_timestamp
  424. self.assertPropertyDefaultValue(vm, 'backup_timestamp', None)
  425. self.assertPropertyValue(vm, 'backup_timestamp', timestamp_str,
  426. timestamp)
  427. def test_401_backup_timestamp_invalid(self):
  428. vm = self.get_vm()
  429. self.assertPropertyInvalidValue(vm, 'backup_timestamp', 'xxx')
  430. self.assertPropertyInvalidValue(vm, 'backup_timestamp', None)