app.py 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941
  1. # -*- encoding: utf8 -*-
  2. #
  3. # The Qubes OS Project, http://www.qubes-os.org
  4. #
  5. # Copyright (C) 2017 Marek Marczykowski-Górecki
  6. # <marmarek@invisiblethingslab.com>
  7. #
  8. # This program is free software; you can redistribute it and/or modify
  9. # it under the terms of the GNU Lesser 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 Lesser General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU Lesser General Public License along
  19. # with this program; if not, see <http://www.gnu.org/licenses/>.
  20. import os
  21. import shutil
  22. import socket
  23. import subprocess
  24. import unittest
  25. import multiprocessing
  26. try:
  27. import unittest.mock as mock
  28. except ImportError:
  29. import mock
  30. from mock import call
  31. import tempfile
  32. import qubesadmin.exc
  33. import qubesadmin.tests
  34. class TC_00_VMCollection(qubesadmin.tests.QubesTestCase):
  35. def test_000_list(self):
  36. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  37. b'0\x00test-vm class=AppVM state=Running\n'
  38. self.assertEqual(
  39. list(self.app.domains.keys()),
  40. ['test-vm'])
  41. self.assertAllCalled()
  42. def test_001_getitem(self):
  43. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  44. b'0\x00test-vm class=AppVM state=Running\n'
  45. try:
  46. vm = self.app.domains['test-vm']
  47. self.assertEqual(vm.name, 'test-vm')
  48. except KeyError:
  49. self.fail('VM not found in collection')
  50. self.assertAllCalled()
  51. with self.assertRaises(KeyError):
  52. vm = self.app.domains['test-non-existent']
  53. self.assertAllCalled()
  54. def test_002_in(self):
  55. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  56. b'0\x00test-vm class=AppVM state=Running\n'
  57. self.assertIn('test-vm', self.app.domains)
  58. self.assertAllCalled()
  59. self.assertNotIn('test-non-existent', self.app.domains)
  60. self.assertAllCalled()
  61. def test_003_iter(self):
  62. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  63. b'0\x00test-vm class=AppVM state=Running\n'
  64. self.assertEqual([vm.name for vm in self.app.domains], ['test-vm'])
  65. self.assertAllCalled()
  66. def test_004_delitem(self):
  67. self.app.expected_calls[('test-vm', 'admin.vm.Remove', None, None)] = \
  68. b'0\x00'
  69. del self.app.domains['test-vm']
  70. self.assertAllCalled()
  71. def test_005_keys(self):
  72. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  73. b'0\x00test-vm class=AppVM state=Running\n' \
  74. b'test-vm2 class=AppVM state=Running\n'
  75. self.assertEqual(set(self.app.domains.keys()),
  76. set(['test-vm', 'test-vm2']))
  77. self.assertAllCalled()
  78. def test_006_values(self):
  79. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  80. b'0\x00test-vm class=AppVM state=Running\n' \
  81. b'test-vm2 class=AppVM state=Running\n'
  82. values = self.app.domains.values()
  83. for obj in values:
  84. self.assertIsInstance(obj, qubesadmin.vm.QubesVM)
  85. self.assertEqual(set([vm.name for vm in values]),
  86. set(['test-vm', 'test-vm2']))
  87. self.assertAllCalled()
  88. def test_007_getitem_blind_mode(self):
  89. self.app.blind_mode = True
  90. try:
  91. vm = self.app.domains['test-vm']
  92. self.assertEqual(vm.name, 'test-vm')
  93. except KeyError:
  94. self.fail('VM not found in collection')
  95. self.assertAllCalled()
  96. with self.assertNotRaises(KeyError):
  97. vm = self.app.domains['test-non-existent']
  98. self.assertAllCalled()
  99. def test_008_in_blind_mode(self):
  100. self.app.blind_mode = True
  101. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  102. b'0\x00test-vm class=AppVM state=Running\n'
  103. self.assertIn('test-vm', self.app.domains)
  104. self.assertAllCalled()
  105. self.assertNotIn('test-non-existent', self.app.domains)
  106. self.assertAllCalled()
  107. def test_009_getitem_cache_class(self):
  108. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  109. b'0\x00test-vm class=AppVM state=Running\n'
  110. try:
  111. vm = self.app.domains['test-vm']
  112. self.assertEqual(vm.name, 'test-vm')
  113. self.assertEqual(vm.klass, 'AppVM')
  114. except KeyError:
  115. self.fail('VM not found in collection')
  116. self.assertAllCalled()
  117. class TC_10_QubesBase(qubesadmin.tests.QubesTestCase):
  118. def setUp(self):
  119. super(TC_10_QubesBase, self).setUp()
  120. self.check_output_patch = mock.patch(
  121. 'subprocess.check_output')
  122. self.check_output_mock = self.check_output_patch.start()
  123. def tearDown(self):
  124. self.check_output_patch.stop()
  125. super(TC_10_QubesBase, self).tearDown()
  126. def test_010_new_simple(self):
  127. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM', None,
  128. b'name=new-vm label=red')] = b'0\x00'
  129. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  130. b'0\x00new-vm class=AppVM state=Running\n'
  131. vm = self.app.add_new_vm('AppVM', 'new-vm', 'red')
  132. self.assertEqual(vm.name, 'new-vm')
  133. self.assertEqual(vm.klass, 'AppVM')
  134. self.assertAllCalled()
  135. def test_011_new_template(self):
  136. self.app.expected_calls[('dom0', 'admin.vm.Create.TemplateVM', None,
  137. b'name=new-template label=red')] = b'0\x00'
  138. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  139. b'0\x00new-template class=TemplateVM state=Running\n'
  140. vm = self.app.add_new_vm('TemplateVM', 'new-template', 'red')
  141. self.assertEqual(vm.name, 'new-template')
  142. self.assertEqual(vm.klass, 'TemplateVM')
  143. self.assertAllCalled()
  144. def test_012_new_template_based(self):
  145. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  146. 'some-template', b'name=new-vm label=red')] = b'0\x00'
  147. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  148. b'0\x00new-vm class=AppVM state=Running\n'
  149. vm = self.app.add_new_vm('AppVM', 'new-vm', 'red', 'some-template')
  150. self.assertEqual(vm.name, 'new-vm')
  151. self.assertEqual(vm.klass, 'AppVM')
  152. self.assertAllCalled()
  153. def test_013_new_objects_params(self):
  154. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  155. 'some-template', b'name=new-vm label=red')] = b'0\x00'
  156. self.app.expected_calls[('dom0', 'admin.label.List', None, None)] = \
  157. b'0\x00red\nblue\n'
  158. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  159. b'0\x00new-vm class=AppVM state=Running\n' \
  160. b'some-template class=TemplateVM state=Running\n'
  161. vm = self.app.add_new_vm(self.app.get_vm_class('AppVM'), 'new-vm',
  162. self.app.get_label('red'), self.app.domains['some-template'])
  163. self.assertEqual(vm.name, 'new-vm')
  164. self.assertEqual(vm.klass, 'AppVM')
  165. self.assertAllCalled()
  166. def test_014_new_pool(self):
  167. self.app.expected_calls[('dom0', 'admin.vm.CreateInPool.AppVM', None,
  168. b'name=new-vm label=red pool=some-pool')] = b'0\x00'
  169. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  170. b'0\x00new-vm class=AppVM state=Running\n'
  171. vm = self.app.add_new_vm('AppVM', 'new-vm', 'red', pool='some-pool')
  172. self.assertEqual(vm.name, 'new-vm')
  173. self.assertEqual(vm.klass, 'AppVM')
  174. self.assertAllCalled()
  175. def test_015_new_pools(self):
  176. self.app.expected_calls[('dom0', 'admin.vm.CreateInPool.AppVM', None,
  177. b'name=new-vm label=red pool:private=some-pool '
  178. b'pool:volatile=other-pool')] = b'0\x00'
  179. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  180. b'0\x00new-vm class=AppVM state=Running\n'
  181. vm = self.app.add_new_vm('AppVM', 'new-vm', 'red',
  182. pools={'private': 'some-pool', 'volatile': 'other-pool'})
  183. self.assertEqual(vm.name, 'new-vm')
  184. self.assertEqual(vm.klass, 'AppVM')
  185. self.assertAllCalled()
  186. def test_016_new_template_based_default(self):
  187. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  188. None, b'name=new-vm label=red')] = b'0\x00'
  189. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  190. b'0\x00new-vm class=AppVM state=Running\n'
  191. vm = self.app.add_new_vm('AppVM', 'new-vm', 'red',
  192. template=qubesadmin.DEFAULT)
  193. self.assertEqual(vm.name, 'new-vm')
  194. self.assertEqual(vm.klass, 'AppVM')
  195. self.assertAllCalled()
  196. def test_020_get_label(self):
  197. self.app.expected_calls[('dom0', 'admin.label.List', None, None)] = \
  198. b'0\x00red\nblue\n'
  199. label = self.app.get_label('red')
  200. self.assertEqual(label.name, 'red')
  201. self.assertAllCalled()
  202. def clone_setup_common_calls(self, src, dst):
  203. # have each property type with default=no, each special-cased,
  204. # and some with default=yes
  205. properties = {
  206. 'label': 'default=False type=label red',
  207. 'template': 'default=False type=vm test-template',
  208. 'memory': 'default=False type=int 400',
  209. 'kernel': 'default=False type=str 4.9.31',
  210. 'netvm': 'default=False type=vm test-net',
  211. 'virt_mode': 'default=False type=str hvm',
  212. 'default_user': 'default=True type=str user',
  213. }
  214. self.app.expected_calls[
  215. (src, 'admin.vm.property.List', None, None)] = \
  216. b'0\0qid\nname\n' + \
  217. b'\n'.join(prop.encode() for prop in properties.keys()) + \
  218. b'\n'
  219. for prop, value in properties.items():
  220. self.app.expected_calls[
  221. (src, 'admin.vm.property.Get', prop, None)] = \
  222. b'0\0' + value.encode()
  223. # special cases handled by admin.vm.Create call
  224. if prop in ('label', 'template'):
  225. continue
  226. # default properties should not be set
  227. if 'default=True' in value:
  228. continue
  229. self.app.expected_calls[
  230. (dst, 'admin.vm.property.Set', prop,
  231. value.split()[-1].encode())] = b'0\0'
  232. # tags
  233. self.app.expected_calls[
  234. (src, 'admin.vm.tag.List', None, None)] = \
  235. b'0\0tag1\ntag2\n'
  236. self.app.expected_calls[
  237. (dst, 'admin.vm.tag.Set', 'tag1', None)] = b'0\0'
  238. self.app.expected_calls[
  239. (dst, 'admin.vm.tag.Set', 'tag2', None)] = b'0\0'
  240. # features
  241. self.app.expected_calls[
  242. (src, 'admin.vm.feature.List', None, None)] = \
  243. b'0\0feat1\nfeat2\n'
  244. self.app.expected_calls[
  245. (src, 'admin.vm.feature.Get', 'feat1', None)] = \
  246. b'0\0feat1-value with spaces'
  247. self.app.expected_calls[
  248. (src, 'admin.vm.feature.Get', 'feat2', None)] = \
  249. b'0\x001'
  250. self.app.expected_calls[
  251. (dst, 'admin.vm.feature.Set', 'feat1',
  252. b'feat1-value with spaces')] = b'0\0'
  253. self.app.expected_calls[
  254. (dst, 'admin.vm.feature.Set', 'feat2', b'1')] = b'0\0'
  255. # firewall
  256. rules = (
  257. b'action=drop dst4=192.168.0.0/24\n'
  258. b'action=accept\n'
  259. )
  260. self.app.expected_calls[
  261. (src, 'admin.vm.firewall.Get', None, None)] = \
  262. b'0\x00' + rules
  263. self.app.expected_calls[
  264. (dst, 'admin.vm.firewall.Set', None, rules)] = \
  265. b'0\x00'
  266. # storage
  267. for vm in (src, dst):
  268. self.app.expected_calls[
  269. (vm, 'admin.vm.volume.Info', 'root', None)] = \
  270. b'0\x00pool=lvm\n' \
  271. b'vid=vm-' + vm.encode() + b'/root\n' \
  272. b'size=10737418240\n' \
  273. b'usage=2147483648\n' \
  274. b'rw=False\n' \
  275. b'internal=True\n' \
  276. b'source=vm-test-template/root\n' \
  277. b'save_on_stop=False\n' \
  278. b'snap_on_start=True\n'
  279. self.app.expected_calls[
  280. (vm, 'admin.vm.volume.Info', 'private', None)] = \
  281. b'0\x00pool=lvm\n' \
  282. b'vid=vm-' + vm.encode() + b'/private\n' \
  283. b'size=2147483648\n' \
  284. b'usage=214748364\n' \
  285. b'rw=True\n' \
  286. b'internal=True\n' \
  287. b'save_on_stop=True\n' \
  288. b'snap_on_start=False\n'
  289. self.app.expected_calls[
  290. (vm, 'admin.vm.volume.Info', 'volatile', None)] = \
  291. b'0\x00pool=lvm\n' \
  292. b'vid=vm-' + vm.encode() + b'/volatile\n' \
  293. b'size=10737418240\n' \
  294. b'usage=0\n' \
  295. b'rw=True\n' \
  296. b'internal=True\n' \
  297. b'source=None\n' \
  298. b'save_on_stop=False\n' \
  299. b'snap_on_start=False\n'
  300. self.app.expected_calls[
  301. (vm, 'admin.vm.volume.Info', 'kernel', None)] = \
  302. b'0\x00pool=linux-kernel\n' \
  303. b'vid=\n' \
  304. b'size=0\n' \
  305. b'usage=0\n' \
  306. b'rw=False\n' \
  307. b'internal=True\n' \
  308. b'source=None\n' \
  309. b'save_on_stop=False\n' \
  310. b'snap_on_start=False\n'
  311. self.app.expected_calls[
  312. (vm, 'admin.vm.volume.List', None, None)] = \
  313. b'0\x00root\nprivate\nvolatile\nkernel\n'
  314. self.app.expected_calls[
  315. (src, 'admin.vm.volume.CloneFrom', 'private', None)] = \
  316. b'0\x00token-private'
  317. self.app.expected_calls[
  318. (dst, 'admin.vm.volume.CloneTo', 'private', b'token-private')] = \
  319. b'0\x00'
  320. self.app.expected_calls[
  321. ('dom0', 'admin.property.Get', 'default_pool_private', None)] = \
  322. b'0\0default=True type=str lvm'
  323. def test_030_clone(self):
  324. self.clone_setup_common_calls('test-vm', 'new-name')
  325. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  326. b'0\x00new-name class=AppVM state=Halted\n' \
  327. b'test-vm class=AppVM state=Halted\n' \
  328. b'test-template class=TemplateVM state=Halted\n' \
  329. b'test-net class=AppVM state=Halted\n'
  330. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  331. 'test-template', b'name=new-name label=red')] = b'0\x00'
  332. new_vm = self.app.clone_vm('test-vm', 'new-name')
  333. self.assertEqual(new_vm.name, 'new-name')
  334. self.check_output_mock.assert_called_once_with(
  335. ['qvm-appmenus', '--init', '--update',
  336. '--source', 'test-vm', 'new-name'],
  337. stderr=subprocess.STDOUT
  338. )
  339. self.assertAllCalled()
  340. def test_031_clone_object(self):
  341. self.clone_setup_common_calls('test-vm', 'new-name')
  342. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  343. b'0\x00new-name class=AppVM state=Halted\n' \
  344. b'test-vm class=AppVM state=Halted\n' \
  345. b'test-template class=TemplateVM state=Halted\n' \
  346. b'test-net class=AppVM state=Halted\n'
  347. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  348. 'test-template', b'name=new-name label=red')] = b'0\x00'
  349. new_vm = self.app.clone_vm(self.app.domains['test-vm'], 'new-name')
  350. self.assertEqual(new_vm.name, 'new-name')
  351. self.assertAllCalled()
  352. def test_032_clone_pool(self):
  353. self.clone_setup_common_calls('test-vm', 'new-name')
  354. for volume in ('root', 'private', 'volatile', 'kernel'):
  355. del self.app.expected_calls[
  356. 'test-vm', 'admin.vm.volume.Info', volume, None]
  357. del self.app.expected_calls[
  358. 'dom0', 'admin.property.Get', 'default_pool_private', None]
  359. self.app.expected_calls[('dom0', 'admin.vm.CreateInPool.AppVM',
  360. 'test-template',
  361. b'name=new-name label=red pool=some-pool')] = b'0\x00'
  362. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  363. b'0\x00new-name class=AppVM state=Halted\n' \
  364. b'test-vm class=AppVM state=Halted\n' \
  365. b'test-template class=TemplateVM state=Halted\n' \
  366. b'test-net class=AppVM state=Halted\n'
  367. new_vm = self.app.clone_vm('test-vm', 'new-name', pool='some-pool')
  368. self.assertEqual(new_vm.name, 'new-name')
  369. self.assertAllCalled()
  370. def test_033_clone_pools(self):
  371. self.clone_setup_common_calls('test-vm', 'new-name')
  372. for volume in ('root', 'private', 'volatile', 'kernel'):
  373. del self.app.expected_calls[
  374. 'test-vm', 'admin.vm.volume.Info', volume, None]
  375. del self.app.expected_calls[
  376. 'dom0', 'admin.property.Get', 'default_pool_private', None]
  377. self.app.expected_calls[('dom0', 'admin.vm.CreateInPool.AppVM',
  378. 'test-template',
  379. b'name=new-name label=red pool:private=some-pool '
  380. b'pool:volatile=other-pool')] = b'0\x00'
  381. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  382. b'0\x00new-name class=AppVM state=Halted\n' \
  383. b'test-vm class=AppVM state=Halted\n' \
  384. b'test-template class=TemplateVM state=Halted\n' \
  385. b'test-net class=AppVM state=Halted\n'
  386. new_vm = self.app.clone_vm('test-vm', 'new-name',
  387. pools={'private': 'some-pool', 'volatile': 'other-pool'})
  388. self.assertEqual(new_vm.name, 'new-name')
  389. self.assertAllCalled()
  390. def test_034_clone_class_change(self):
  391. self.clone_setup_common_calls('test-vm', 'new-name')
  392. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  393. b'0\x00new-name class=StandaloneVM state=Halted\n' \
  394. b'test-vm class=AppVM state=Halted\n' \
  395. b'test-template class=TemplateVM state=Halted\n' \
  396. b'test-net class=AppVM state=Halted\n'
  397. self.app.expected_calls[('dom0', 'admin.vm.Create.StandaloneVM',
  398. 'test-template', b'name=new-name label=red')] = b'0\x00'
  399. self.app.expected_calls[
  400. ('new-name', 'admin.vm.volume.Info', 'root', None)] = \
  401. b'0\x00pool=lvm\n' \
  402. b'vid=vm-new-name/root\n' \
  403. b'size=10737418240\n' \
  404. b'usage=2147483648\n' \
  405. b'rw=True\n' \
  406. b'internal=True\n' \
  407. b'source=None\n' \
  408. b'save_on_stop=True\n' \
  409. b'snap_on_start=False\n'
  410. self.app.expected_calls[
  411. ('test-vm', 'admin.vm.volume.CloneFrom', 'root', None)] = \
  412. b'0\x00token-root'
  413. self.app.expected_calls[
  414. ('new-name', 'admin.vm.volume.CloneTo', 'root', b'token-root')] = \
  415. b'0\x00'
  416. new_vm = self.app.clone_vm('test-vm', 'new-name',
  417. new_cls='StandaloneVM')
  418. self.assertEqual(new_vm.name, 'new-name')
  419. self.assertEqual(new_vm.klass, 'StandaloneVM')
  420. self.assertAllCalled()
  421. def test_035_clone_fail(self):
  422. self.app.expected_calls[
  423. ('test-vm', 'admin.vm.property.List', None, None)] = \
  424. b'0\0qid\nname\ntemplate\nlabel\nmemory\n'
  425. # simplify it a little, shouldn't get this far anyway
  426. self.app.expected_calls[
  427. ('test-vm', 'admin.vm.volume.List', None, None)] = \
  428. b'0\x00'
  429. self.app.expected_calls[
  430. ('test-vm', 'admin.vm.property.Get', 'label', None)] = \
  431. b'0\0default=False type=label red'
  432. self.app.expected_calls[
  433. ('test-vm', 'admin.vm.property.Get', 'template', None)] = \
  434. b'0\0default=False type=vm test-template'
  435. self.app.expected_calls[
  436. ('test-vm', 'admin.vm.property.Get', 'memory', None)] = \
  437. b'0\0default=False type=int 400'
  438. self.app.expected_calls[
  439. ('new-name', 'admin.vm.property.Set', 'memory', b'400')] = \
  440. b'2\0QubesException\0\0something happened\0'
  441. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  442. b'0\x00new-name class=AppVM state=Halted\n' \
  443. b'test-vm class=AppVM state=Halted\n' \
  444. b'test-template class=TemplateVM state=Halted\n' \
  445. b'test-net class=AppVM state=Halted\n'
  446. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  447. 'test-template', b'name=new-name label=red')] = b'0\x00'
  448. self.app.expected_calls[('new-name', 'admin.vm.Remove', None, None)] = \
  449. b'0\x00'
  450. with self.assertRaises(qubesadmin.exc.QubesException):
  451. self.app.clone_vm('test-vm', 'new-name')
  452. self.assertAllCalled()
  453. def test_036_clone_ignore_errors_prop(self):
  454. self.clone_setup_common_calls('test-vm', 'new-name')
  455. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  456. b'0\x00new-name class=AppVM state=Halted\n' \
  457. b'test-vm class=AppVM state=Halted\n' \
  458. b'test-template class=TemplateVM state=Halted\n' \
  459. b'test-net class=AppVM state=Halted\n'
  460. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  461. 'test-template', b'name=new-name label=red')] = b'0\x00'
  462. self.app.expected_calls[
  463. ('new-name', 'admin.vm.property.Set', 'memory', b'400')] = \
  464. b'2\0QubesException\0\0something happened\0'
  465. new_vm = self.app.clone_vm('test-vm', 'new-name', ignore_errors=True)
  466. self.assertEqual(new_vm.name, 'new-name')
  467. self.assertAllCalled()
  468. def test_037_clone_ignore_errors_feature(self):
  469. self.clone_setup_common_calls('test-vm', 'new-name')
  470. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  471. b'0\x00new-name class=AppVM state=Halted\n' \
  472. b'test-vm class=AppVM state=Halted\n' \
  473. b'test-template class=TemplateVM state=Halted\n' \
  474. b'test-net class=AppVM state=Halted\n'
  475. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  476. 'test-template', b'name=new-name label=red')] = b'0\x00'
  477. self.app.expected_calls[
  478. ('new-name', 'admin.vm.feature.Set', 'feat2', b'1')] = \
  479. b'2\0QubesException\0\0something happened\0'
  480. new_vm = self.app.clone_vm('test-vm', 'new-name', ignore_errors=True)
  481. self.assertEqual(new_vm.name, 'new-name')
  482. self.assertAllCalled()
  483. def test_038_clone_ignore_errors_tag(self):
  484. self.clone_setup_common_calls('test-vm', 'new-name')
  485. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  486. b'0\x00new-name class=AppVM state=Halted\n' \
  487. b'test-vm class=AppVM state=Halted\n' \
  488. b'test-template class=TemplateVM state=Halted\n' \
  489. b'test-net class=AppVM state=Halted\n'
  490. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  491. 'test-template', b'name=new-name label=red')] = b'0\x00'
  492. self.app.expected_calls[
  493. ('new-name', 'admin.vm.tag.Set', 'tag1', None)] = \
  494. b'2\0QubesException\0\0something happened\0'
  495. new_vm = self.app.clone_vm('test-vm', 'new-name', ignore_errors=True)
  496. self.assertEqual(new_vm.name, 'new-name')
  497. self.assertAllCalled()
  498. def test_039_clone_ignore_errors_firewall(self):
  499. self.clone_setup_common_calls('test-vm', 'new-name')
  500. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  501. b'0\x00new-name class=AppVM state=Halted\n' \
  502. b'test-vm class=AppVM state=Halted\n' \
  503. b'test-template class=TemplateVM state=Halted\n' \
  504. b'test-net class=AppVM state=Halted\n'
  505. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  506. 'test-template', b'name=new-name label=red')] = b'0\x00'
  507. self.app.expected_calls[
  508. ('new-name', 'admin.vm.firewall.Set', None,
  509. b'action=drop dst4=192.168.0.0/24\naction=accept\n')] = \
  510. b'2\0QubesException\0\0something happened\0'
  511. new_vm = self.app.clone_vm('test-vm', 'new-name', ignore_errors=True)
  512. self.assertEqual(new_vm.name, 'new-name')
  513. self.assertAllCalled()
  514. def test_040_clone_ignore_errors_storage(self):
  515. self.clone_setup_common_calls('test-vm', 'new-name')
  516. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  517. b'0\x00new-name class=AppVM state=Halted\n' \
  518. b'test-vm class=AppVM state=Halted\n' \
  519. b'test-template class=TemplateVM state=Halted\n' \
  520. b'test-net class=AppVM state=Halted\n'
  521. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  522. 'test-template', b'name=new-name label=red')] = b'0\x00'
  523. self.app.expected_calls[
  524. ('new-name', 'admin.vm.volume.CloneTo', 'private',
  525. b'token-private')] = \
  526. b'2\0QubesException\0\0something happened\0'
  527. self.app.expected_calls[('new-name', 'admin.vm.Remove', None, None)] = \
  528. b'0\x00'
  529. del self.app.expected_calls[
  530. ('new-name', 'admin.vm.volume.Info', 'root', None)]
  531. del self.app.expected_calls[
  532. ('new-name', 'admin.vm.volume.Info', 'volatile', None)]
  533. with self.assertRaises(qubesadmin.exc.QubesException):
  534. self.app.clone_vm('test-vm', 'new-name', ignore_errors=True)
  535. self.assertAllCalled()
  536. def test_041_clone_fail_storage(self):
  537. self.clone_setup_common_calls('test-vm', 'new-name')
  538. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  539. b'0\x00new-name class=AppVM state=Halted\n' \
  540. b'test-vm class=AppVM state=Halted\n' \
  541. b'test-template class=TemplateVM state=Halted\n' \
  542. b'test-net class=AppVM state=Halted\n'
  543. self.app.expected_calls[('dom0', 'admin.vm.Create.AppVM',
  544. 'test-template', b'name=new-name label=red')] = b'0\x00'
  545. self.app.expected_calls[
  546. ('new-name', 'admin.vm.volume.CloneTo', 'private',
  547. b'token-private')] = \
  548. b'2\0QubesException\0\0something happened\0'
  549. self.app.expected_calls[('new-name', 'admin.vm.Remove', None, None)] = \
  550. b'0\x00'
  551. del self.app.expected_calls[
  552. ('new-name', 'admin.vm.volume.Info', 'root', None)]
  553. del self.app.expected_calls[
  554. ('new-name', 'admin.vm.volume.Info', 'volatile', None)]
  555. with self.assertRaises(qubesadmin.exc.QubesException):
  556. self.app.clone_vm('test-vm', 'new-name')
  557. self.assertAllCalled()
  558. def test_042_clone_nondefault_pool(self):
  559. self.clone_setup_common_calls('test-vm', 'new-name')
  560. self.app.expected_calls[
  561. ('test-vm', 'admin.vm.volume.Info', 'private', None)] = \
  562. b'0\x00pool=another\n' \
  563. b'vid=vm-test-vm/private\n' \
  564. b'size=2147483648\n' \
  565. b'usage=214748364\n' \
  566. b'rw=True\n' \
  567. b'internal=True\n' \
  568. b'save_on_stop=True\n' \
  569. b'snap_on_start=False\n'
  570. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  571. b'0\x00new-name class=AppVM state=Halted\n' \
  572. b'test-vm class=AppVM state=Halted\n' \
  573. b'test-template class=TemplateVM state=Halted\n' \
  574. b'test-net class=AppVM state=Halted\n'
  575. self.app.expected_calls[('dom0', 'admin.vm.CreateInPool.AppVM',
  576. 'test-template', b'name=new-name label=red pool:private=another')]\
  577. = b'0\x00'
  578. new_vm = self.app.clone_vm('test-vm', 'new-name')
  579. self.assertEqual(new_vm.name, 'new-name')
  580. self.check_output_mock.assert_called_once_with(
  581. ['qvm-appmenus', '--init', '--update',
  582. '--source', 'test-vm', 'new-name'],
  583. stderr=subprocess.STDOUT
  584. )
  585. self.assertAllCalled()
  586. class TC_20_QubesLocal(unittest.TestCase):
  587. def setUp(self):
  588. super(TC_20_QubesLocal, self).setUp()
  589. self.socket_dir = tempfile.mkdtemp()
  590. self.orig_sock = qubesadmin.config.QUBESD_SOCKET
  591. qubesadmin.config.QUBESD_SOCKET = os.path.join(self.socket_dir, 'sock')
  592. self.proc = None
  593. self.app = qubesadmin.app.QubesLocal()
  594. def listen_and_send(self, send_data):
  595. '''Listen on socket and send data in response.
  596. :param bytes send_data: data to send
  597. '''
  598. self.socket_pipe, child_pipe = multiprocessing.Pipe()
  599. self.socket = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
  600. self.socket.bind(os.path.join(self.socket_dir, 'sock'))
  601. self.socket.listen(1)
  602. def worker(sock, pipe, send_data_):
  603. conn, addr = sock.accept()
  604. pipe.send(conn.makefile('rb').read())
  605. conn.sendall(send_data_)
  606. conn.close()
  607. self.proc = multiprocessing.Process(target=worker,
  608. args=(self.socket, child_pipe, send_data))
  609. self.proc.start()
  610. self.socket.close()
  611. def get_request(self):
  612. '''Get request sent to "qubesd" mock'''
  613. return self.socket_pipe.recv()
  614. def tearDown(self):
  615. qubesadmin.config.QUBESD_SOCKET = self.orig_sock
  616. if self.proc is not None:
  617. try:
  618. self.proc.terminate()
  619. except OSError:
  620. pass
  621. shutil.rmtree(self.socket_dir)
  622. super(TC_20_QubesLocal, self).tearDown()
  623. def test_000_qubesd_call(self):
  624. self.listen_and_send(b'0\0')
  625. self.app.qubesd_call('test-vm', 'some.method', 'arg1', b'payload')
  626. self.assertEqual(self.get_request(),
  627. b'dom0\0some.method\0test-vm\0arg1\0payload')
  628. def test_001_qubesd_call_none_arg(self):
  629. self.listen_and_send(b'0\0')
  630. self.app.qubesd_call('test-vm', 'some.method', None, b'payload')
  631. self.assertEqual(self.get_request(),
  632. b'dom0\0some.method\0test-vm\0\0payload')
  633. def test_002_qubesd_call_none_payload(self):
  634. self.listen_and_send(b'0\0')
  635. self.app.qubesd_call('test-vm', 'some.method', None, None)
  636. self.assertEqual(self.get_request(),
  637. b'dom0\0some.method\0test-vm\0\0')
  638. def test_003_qubesd_call_payload_stream(self):
  639. # this should really be in setUp()...
  640. tmpdir = tempfile.mkdtemp()
  641. self.addCleanup(shutil.rmtree, tmpdir)
  642. service_path = os.path.join(tmpdir, 'test.service')
  643. payload_input = os.path.join(tmpdir, 'payload-input')
  644. with open(service_path, 'w') as f:
  645. f.write('#!/bin/bash\n'
  646. 'env > {dir}/env\n'
  647. 'echo "$@" > {dir}/args\n'
  648. 'cat > {dir}/payload\n'
  649. 'echo -en \'0\\0return-value\'\n'.format(dir=tmpdir))
  650. os.chmod(service_path, 0o755)
  651. with open(payload_input, 'w+') as payload_file:
  652. payload_file.write('some payload\n')
  653. payload_file.seek(0)
  654. with mock.patch('qubesadmin.config.QREXEC_SERVICES_DIR',
  655. tmpdir):
  656. value = self.app.qubesd_call('test-vm', 'test.service',
  657. 'some-arg', payload_stream=payload_file)
  658. self.assertEqual(value, b'return-value')
  659. self.assertTrue(os.path.exists(tmpdir + '/env'))
  660. with open(tmpdir + '/env') as env:
  661. self.assertIn('QREXEC_REMOTE_DOMAIN=dom0\n', env)
  662. self.assertIn('QREXEC_REQUESTED_TARGET=test-vm\n', env)
  663. self.assertTrue(os.path.exists(tmpdir + '/args'))
  664. with open(tmpdir + '/args') as args:
  665. self.assertEqual(args.read(), 'some-arg\n')
  666. self.assertTrue(os.path.exists(tmpdir + '/payload'))
  667. with open(tmpdir + '/payload') as payload:
  668. self.assertEqual(payload.read(), 'some payload\n')
  669. def test_004_qubesd_call_payload_stream_proc(self):
  670. # this should really be in setUp()...
  671. tmpdir = tempfile.mkdtemp()
  672. self.addCleanup(shutil.rmtree, tmpdir)
  673. service_path = os.path.join(tmpdir, 'test.service')
  674. echo = subprocess.Popen(['echo', 'some payload'],
  675. stdout=subprocess.PIPE)
  676. with open(service_path, 'w') as f:
  677. f.write('#!/bin/bash\n'
  678. 'env > {dir}/env\n'
  679. 'echo "$@" > {dir}/args\n'
  680. 'cat > {dir}/payload\n'
  681. 'echo -en \'0\\0return-value\'\n'.format(dir=tmpdir))
  682. os.chmod(service_path, 0o755)
  683. with mock.patch('qubesadmin.config.QREXEC_SERVICES_DIR',
  684. tmpdir):
  685. value = self.app.qubesd_call('test-vm', 'test.service',
  686. 'some-arg', payload_stream=echo.stdout)
  687. echo.stdout.close()
  688. self.assertEqual(value, b'return-value')
  689. self.assertTrue(os.path.exists(tmpdir + '/env'))
  690. with open(tmpdir + '/env') as env:
  691. self.assertIn('QREXEC_REMOTE_DOMAIN=dom0\n', env)
  692. self.assertIn('QREXEC_REQUESTED_TARGET=test-vm\n', env)
  693. self.assertTrue(os.path.exists(tmpdir + '/args'))
  694. with open(tmpdir + '/args') as args:
  695. self.assertEqual(args.read(), 'some-arg\n')
  696. self.assertTrue(os.path.exists(tmpdir + '/payload'))
  697. with open(tmpdir + '/payload') as payload:
  698. self.assertEqual(payload.read(), 'some payload\n')
  699. @mock.patch('os.isatty', lambda fd: fd == 2)
  700. def test_010_run_service(self):
  701. self.listen_and_send(b'0\0')
  702. with mock.patch('subprocess.Popen') as mock_proc:
  703. p = self.app.run_service('some-vm', 'service.name')
  704. mock_proc.assert_called_once_with([
  705. qubesadmin.config.QREXEC_CLIENT,
  706. '-d', 'some-vm', '-T', 'DEFAULT:QUBESRPC service.name dom0'],
  707. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  708. stderr=subprocess.PIPE)
  709. self.assertEqual(self.get_request(),
  710. b'dom0\0admin.vm.Start\0some-vm\0\0')
  711. def test_011_run_service_filter_esc(self):
  712. self.listen_and_send(b'0\0')
  713. with mock.patch('subprocess.Popen') as mock_proc:
  714. p = self.app.run_service('some-vm', 'service.name', filter_esc=True)
  715. mock_proc.assert_called_once_with([
  716. qubesadmin.config.QREXEC_CLIENT,
  717. '-d', 'some-vm', '-t', '-T',
  718. 'DEFAULT:QUBESRPC service.name dom0'],
  719. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  720. stderr=subprocess.PIPE)
  721. self.assertEqual(self.get_request(),
  722. b'dom0\0admin.vm.Start\0some-vm\0\0')
  723. @mock.patch('os.isatty', lambda fd: fd == 2)
  724. def test_012_run_service_user(self):
  725. self.listen_and_send(b'0\0')
  726. with mock.patch('subprocess.Popen') as mock_proc:
  727. p = self.app.run_service('some-vm', 'service.name', user='user')
  728. mock_proc.assert_called_once_with([
  729. qubesadmin.config.QREXEC_CLIENT,
  730. '-d', 'some-vm', '-T',
  731. 'user:QUBESRPC service.name dom0'],
  732. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  733. stderr=subprocess.PIPE)
  734. self.assertEqual(self.get_request(),
  735. b'dom0\0admin.vm.Start\0some-vm\0\0')
  736. def test_013_run_service_default_target(self):
  737. with self.assertRaises(ValueError):
  738. self.app.run_service('', 'service.name')
  739. class TC_30_QubesRemote(unittest.TestCase):
  740. def setUp(self):
  741. super(TC_30_QubesRemote, self).setUp()
  742. self.proc_mock = mock.Mock()
  743. self.proc_mock.configure_mock(**{
  744. 'return_value.returncode': 0
  745. })
  746. self.proc_patch = mock.patch('subprocess.Popen', self.proc_mock)
  747. self.proc_patch.start()
  748. self.app = qubesadmin.app.QubesRemote()
  749. def set_proc_stdout(self, send_data):
  750. self.proc_mock.configure_mock(**{
  751. 'return_value.communicate.return_value': (send_data, None)
  752. })
  753. def tearDown(self):
  754. self.proc_patch.stop()
  755. super(TC_30_QubesRemote, self).tearDown()
  756. def test_000_qubesd_call(self):
  757. self.set_proc_stdout(b'0\0')
  758. self.app.qubesd_call('test-vm', 'some.method', 'arg1', b'payload')
  759. self.assertEqual(self.proc_mock.mock_calls, [
  760. mock.call([qubesadmin.config.QREXEC_CLIENT_VM, 'test-vm',
  761. 'some.method+arg1'],
  762. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  763. stderr=subprocess.PIPE),
  764. mock.call().communicate(b'payload')
  765. ])
  766. def test_001_qubesd_call_none_arg(self):
  767. self.set_proc_stdout(b'0\0')
  768. self.app.qubesd_call('test-vm', 'some.method', None, b'payload')
  769. self.assertEqual(self.proc_mock.mock_calls, [
  770. mock.call([qubesadmin.config.QREXEC_CLIENT_VM, 'test-vm',
  771. 'some.method'],
  772. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  773. stderr=subprocess.PIPE),
  774. mock.call().communicate(b'payload')
  775. ])
  776. def test_002_qubesd_call_none_payload(self):
  777. self.set_proc_stdout(b'0\0')
  778. self.app.qubesd_call('test-vm', 'some.method', None, None)
  779. self.assertEqual(self.proc_mock.mock_calls, [
  780. mock.call([qubesadmin.config.QREXEC_CLIENT_VM, 'test-vm',
  781. 'some.method'],
  782. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  783. stderr=subprocess.PIPE),
  784. mock.call().communicate(None)
  785. ])
  786. def test_003_qubesd_call_payload_stream(self):
  787. self.set_proc_stdout(b'0\0return-value')
  788. tmpdir = tempfile.mkdtemp()
  789. self.addCleanup(shutil.rmtree, tmpdir)
  790. payload_input = os.path.join(tmpdir, 'payload-input')
  791. with open(payload_input, 'w+') as payload_file:
  792. payload_file.write('some payload\n')
  793. payload_file.seek(0)
  794. value = self.app.qubesd_call('test-vm', 'some.method',
  795. 'some-arg', payload_stream=payload_file)
  796. self.assertEqual(self.proc_mock.mock_calls, [
  797. mock.call([qubesadmin.config.QREXEC_CLIENT_VM, 'test-vm',
  798. 'some.method+some-arg'],
  799. stdin=payload_file, stdout=subprocess.PIPE,
  800. stderr=subprocess.PIPE),
  801. mock.call().communicate(None)
  802. ])
  803. self.assertEqual(value, b'return-value')
  804. @mock.patch('os.isatty', lambda fd: fd == 2)
  805. def test_010_run_service(self):
  806. self.app.run_service('some-vm', 'service.name')
  807. self.proc_mock.assert_called_once_with([
  808. qubesadmin.config.QREXEC_CLIENT_VM,
  809. '-T', 'some-vm', 'service.name'],
  810. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  811. stderr=subprocess.PIPE)
  812. @mock.patch('os.isatty', lambda fd: fd == 2)
  813. def test_011_run_service_filter_esc(self):
  814. self.app.run_service('some-vm', 'service.name', filter_esc=True)
  815. self.proc_mock.assert_called_once_with([
  816. qubesadmin.config.QREXEC_CLIENT_VM,
  817. '-t', '-T', 'some-vm', 'service.name'],
  818. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  819. stderr=subprocess.PIPE)
  820. @mock.patch('os.isatty', lambda fd: fd == 2)
  821. def test_012_run_service_user(self):
  822. with self.assertRaises(ValueError):
  823. p = self.app.run_service('some-vm', 'service.name', user='user')
  824. @mock.patch('os.isatty', lambda fd: fd == 2)
  825. def test_013_run_service_default_target(self):
  826. self.app.run_service('', 'service.name')
  827. self.proc_mock.assert_called_once_with([
  828. qubesadmin.config.QREXEC_CLIENT_VM,
  829. '-T', '', 'service.name'],
  830. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  831. stderr=subprocess.PIPE)
  832. @mock.patch('os.isatty', lambda fd: fd == 2)
  833. def test_014_run_service_no_autostart1(self):
  834. self.set_proc_stdout( b'0\x00some-vm class=AppVM state=Running\n')
  835. self.app.run_service('some-vm', 'service.name', autostart=False)
  836. self.proc_mock.assert_has_calls([
  837. call([qubesadmin.config.QREXEC_CLIENT_VM,
  838. 'some-vm', 'admin.vm.List'],
  839. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  840. stderr=subprocess.PIPE),
  841. call().communicate(None),
  842. call([qubesadmin.config.QREXEC_CLIENT_VM,
  843. '-T', 'some-vm', 'service.name'],
  844. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  845. stderr=subprocess.PIPE),
  846. ])
  847. @mock.patch('os.isatty', lambda fd: fd == 2)
  848. def test_015_run_service_no_autostart2(self):
  849. self.set_proc_stdout( b'0\x00some-vm class=AppVM state=Halted\n')
  850. with self.assertRaises(qubesadmin.exc.QubesVMNotRunningError):
  851. self.app.run_service('some-vm', 'service.name', autostart=False)
  852. self.proc_mock.assert_called_once_with([
  853. qubesadmin.config.QREXEC_CLIENT_VM,
  854. 'some-vm', 'admin.vm.List'],
  855. stdin=subprocess.PIPE, stdout=subprocess.PIPE,
  856. stderr=subprocess.PIPE)