qvm_volume.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  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.1 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 qubesadmin.tests
  21. import qubesadmin.tests.tools
  22. import qubesadmin.tools.qvm_volume
  23. class TC_00_qvm_volume(qubesadmin.tests.QubesTestCase):
  24. def setup_expected_calls_for_list(self, vms=('vm1', 'sys-net')):
  25. self.app.expected_calls[
  26. ('dom0', 'admin.vm.List', None, None)] = \
  27. b'0\x00vm1 class=AppVM state=Running\n' \
  28. b'sys-net class=AppVM state=Running\n'
  29. for vm in vms:
  30. for vol in ('root', 'private'):
  31. self.app.expected_calls[
  32. (vm, 'admin.vm.volume.Info', vol, None)] = \
  33. b'0\x00' + \
  34. (b'pool=pool-file\n' if vol == 'root' else
  35. b'pool=other-pool\n') + \
  36. b'vid=' + vm.encode() + b'-' + vol.encode() + b'\n' \
  37. b'size=10737418240\n'
  38. self.app.expected_calls[
  39. (vm, 'admin.vm.volume.ListSnapshots', vol, None)] = \
  40. b'0\x00snap1\n' if vol == 'private' else b'0\x00'
  41. self.app.expected_calls[
  42. (vm, 'admin.vm.volume.List', None, None)] = \
  43. b'0\x00root\nprivate\n'
  44. def test_000_list(self):
  45. self.setup_expected_calls_for_list()
  46. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  47. self.assertEqual(0,
  48. qubesadmin.tools.qvm_volume.main(['ls'], app=self.app))
  49. self.assertEqual(stdout.getvalue(),
  50. 'POOL:VOLUME VMNAME VOLUME_NAME '
  51. 'REVERT_POSSIBLE\n'
  52. 'other-pool:sys-net-private sys-net private Yes\n'
  53. 'other-pool:vm1-private vm1 private Yes\n'
  54. 'pool-file:sys-net-root sys-net root No\n'
  55. 'pool-file:vm1-root vm1 root No\n'
  56. )
  57. self.assertAllCalled()
  58. def test_001_list_domain(self):
  59. self.setup_expected_calls_for_list(vms=('vm1',))
  60. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  61. self.assertEqual(0,
  62. qubesadmin.tools.qvm_volume.main(['ls', 'vm1'],
  63. app=self.app))
  64. self.assertEqual(stdout.getvalue(),
  65. 'POOL:VOLUME VMNAME VOLUME_NAME REVERT_POSSIBLE\n'
  66. 'other-pool:vm1-private vm1 private Yes\n'
  67. 'pool-file:vm1-root vm1 root No\n'
  68. )
  69. self.assertAllCalled()
  70. def test_002_list_domain_pool(self):
  71. self.setup_expected_calls_for_list(vms=('vm1',))
  72. self.app.expected_calls[('dom0', 'admin.pool.List', None, None)] = \
  73. b'0\x00pool-file\nother-pool\n'
  74. del self.app.expected_calls[
  75. ('vm1', 'admin.vm.volume.ListSnapshots', 'private', None)]
  76. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  77. self.assertEqual(0,
  78. qubesadmin.tools.qvm_volume.main(
  79. ['ls', '-p', 'pool-file', 'vm1'],
  80. app=self.app))
  81. self.assertEqual(stdout.getvalue(),
  82. 'POOL:VOLUME VMNAME VOLUME_NAME REVERT_POSSIBLE\n'
  83. 'pool-file:vm1-root vm1 root No\n'
  84. )
  85. self.assertAllCalled()
  86. def test_003_list_pool(self):
  87. self.setup_expected_calls_for_list()
  88. self.app.expected_calls[('dom0', 'admin.pool.List', None, None)] = \
  89. b'0\x00pool-file\nother-pool\n'
  90. del self.app.expected_calls[
  91. ('vm1', 'admin.vm.volume.ListSnapshots', 'private', None)]
  92. del self.app.expected_calls[
  93. ('sys-net', 'admin.vm.volume.ListSnapshots', 'private', None)]
  94. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  95. self.assertEqual(0,
  96. qubesadmin.tools.qvm_volume.main(
  97. ['ls', '-p', 'pool-file'],
  98. app=self.app))
  99. self.assertEqual(stdout.getvalue(),
  100. 'POOL:VOLUME VMNAME VOLUME_NAME REVERT_POSSIBLE\n'
  101. 'pool-file:sys-net-root sys-net root No\n'
  102. 'pool-file:vm1-root vm1 root No\n'
  103. )
  104. self.assertAllCalled()
  105. def test_004_list_multiple_domains(self):
  106. self.setup_expected_calls_for_list(vms=('vm1', 'vm2'))
  107. self.app.expected_calls[
  108. ('dom0', 'admin.vm.List', None, None)] = \
  109. b'0\x00vm1 class=AppVM state=Running\n' \
  110. b'vm2 class=AppVM state=Running\n' \
  111. b'vm3 class=AppVM state=Running\n'
  112. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  113. self.assertEqual(0,
  114. qubesadmin.tools.qvm_volume.main(
  115. ['ls', 'vm1', 'vm2'], app=self.app))
  116. self.assertEqual(stdout.getvalue(),
  117. 'POOL:VOLUME VMNAME VOLUME_NAME REVERT_POSSIBLE\n'
  118. 'other-pool:vm1-private vm1 private Yes\n'
  119. 'other-pool:vm2-private vm2 private Yes\n'
  120. 'pool-file:vm1-root vm1 root No\n'
  121. 'pool-file:vm2-root vm2 root No\n'
  122. )
  123. self.assertAllCalled()
  124. def test_005_list_default_action(self):
  125. self.setup_expected_calls_for_list()
  126. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  127. self.assertEqual(0,
  128. qubesadmin.tools.qvm_volume.main([], app=self.app))
  129. self.assertEqual(stdout.getvalue(),
  130. 'POOL:VOLUME VMNAME VOLUME_NAME '
  131. 'REVERT_POSSIBLE\n'
  132. 'other-pool:sys-net-private sys-net private Yes\n'
  133. 'other-pool:vm1-private vm1 private Yes\n'
  134. 'pool-file:sys-net-root sys-net root No\n'
  135. 'pool-file:vm1-root vm1 root No\n'
  136. )
  137. self.assertAllCalled()
  138. def test_010_extend(self):
  139. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  140. b'0\x00testvm class=AppVM state=Running\n'
  141. self.app.expected_calls[
  142. ('testvm', 'admin.vm.volume.List', None, None)] = \
  143. b'0\x00root\nprivate\n'
  144. self.app.expected_calls[
  145. ('testvm', 'admin.vm.volume.Info', 'private', None)] = \
  146. b'0\x00pool=lvm\n' \
  147. b'vid=qubes_dom0/vm-testvm-private\n' \
  148. b'size=2147483648\n' \
  149. b'usage=10000000\n' \
  150. b'rw=True\n' \
  151. b'source=\n' \
  152. b'save_on_stop=True\n' \
  153. b'snap_on_start=False\n' \
  154. b'revisions_to_keep=3\n' \
  155. b'is_outdated=False\n'
  156. self.app.expected_calls[
  157. ('testvm', 'admin.vm.volume.Resize', 'private', b'10737418240')] = \
  158. b'0\x00'
  159. self.assertEqual(0,
  160. qubesadmin.tools.qvm_volume.main(
  161. ['extend', 'testvm:private', '10GiB'],
  162. app=self.app))
  163. self.assertAllCalled()
  164. def test_011_extend_error(self):
  165. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  166. b'0\x00testvm class=AppVM state=Running\n'
  167. self.app.expected_calls[
  168. ('testvm', 'admin.vm.volume.List', None, None)] = \
  169. b'0\x00root\nprivate\n'
  170. self.app.expected_calls[
  171. ('testvm', 'admin.vm.volume.Info', 'private', None)] = \
  172. b'0\x00pool=lvm\n' \
  173. b'vid=qubes_dom0/vm-testvm-private\n' \
  174. b'size=2147483648\n' \
  175. b'usage=10000000\n' \
  176. b'rw=True\n' \
  177. b'source=\n' \
  178. b'save_on_stop=True\n' \
  179. b'snap_on_start=False\n' \
  180. b'revisions_to_keep=3\n' \
  181. b'is_outdated=False\n'
  182. self.app.expected_calls[
  183. ('testvm', 'admin.vm.volume.Resize', 'private', b'10737418240')] = \
  184. b'2\x00StoragePoolException\x00\x00Failed to resize volume: ' \
  185. b'error: success\x00'
  186. with qubesadmin.tests.tools.StderrBuffer() as stderr:
  187. self.assertEqual(1,
  188. qubesadmin.tools.qvm_volume.main(
  189. ['extend', 'testvm:private', '10GiB'],
  190. app=self.app))
  191. self.assertIn('error: success', stderr.getvalue())
  192. self.assertAllCalled()
  193. def test_012_extend_deny_shrink(self):
  194. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  195. b'0\x00testvm class=AppVM state=Running\n'
  196. self.app.expected_calls[
  197. ('testvm', 'admin.vm.volume.List', None, None)] = \
  198. b'0\x00root\nprivate\n'
  199. self.app.expected_calls[
  200. ('testvm', 'admin.vm.volume.Info', 'private', None)] = \
  201. b'0\x00pool=lvm\n' \
  202. b'vid=qubes_dom0/vm-testvm-private\n' \
  203. b'size=2147483648\n' \
  204. b'usage=10000000\n' \
  205. b'rw=True\n' \
  206. b'source=\n' \
  207. b'save_on_stop=True\n' \
  208. b'snap_on_start=False\n' \
  209. b'revisions_to_keep=3\n' \
  210. b'is_outdated=False\n'
  211. with qubesadmin.tests.tools.StderrBuffer() as stderr:
  212. self.assertEqual(1,
  213. qubesadmin.tools.qvm_volume.main(
  214. ['resize', 'testvm:private', '1GiB'],
  215. app=self.app))
  216. self.assertIn('shrinking of private is disabled', stderr.getvalue())
  217. self.assertAllCalled()
  218. def test_013_resize_force_shrink(self):
  219. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  220. b'0\x00testvm class=AppVM state=Running\n'
  221. self.app.expected_calls[
  222. ('testvm', 'admin.vm.volume.List', None, None)] = \
  223. b'0\x00root\nprivate\n'
  224. self.app.expected_calls[
  225. ('testvm', 'admin.vm.volume.Resize', 'private', b'1073741824')] = \
  226. b'0\x00'
  227. self.assertEqual(0,
  228. qubesadmin.tools.qvm_volume.main(
  229. ['resize', '-f', 'testvm:private', '1GiB'],
  230. app=self.app))
  231. self.assertAllCalled()
  232. def test_020_revert(self):
  233. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  234. b'0\x00testvm class=AppVM state=Running\n'
  235. self.app.expected_calls[
  236. ('testvm', 'admin.vm.volume.List', None, None)] = \
  237. b'0\x00root\nprivate\n'
  238. self.app.expected_calls[
  239. ('testvm', 'admin.vm.volume.ListSnapshots', 'private', None)] = \
  240. b'0\x00200101010000\n200201010000\n200301010000\n'
  241. self.app.expected_calls[
  242. ('testvm', 'admin.vm.volume.Revert', 'private', b'200301010000')] = \
  243. b'0\x00'
  244. self.assertEqual(0,
  245. qubesadmin.tools.qvm_volume.main(
  246. ['revert', 'testvm:private'],
  247. app=self.app))
  248. self.assertAllCalled()
  249. def test_021_revert_error(self):
  250. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  251. b'0\x00testvm class=AppVM state=Running\n'
  252. self.app.expected_calls[
  253. ('testvm', 'admin.vm.volume.List', None, None)] = \
  254. b'0\x00root\nprivate\n'
  255. self.app.expected_calls[
  256. ('testvm', 'admin.vm.volume.ListSnapshots', 'private', None)] = \
  257. b'0\x00200101010000\n200201010000\n200301010000\n'
  258. self.app.expected_calls[
  259. ('testvm', 'admin.vm.volume.Revert', 'private', b'200301010000')] = \
  260. b'2\x00StoragePoolException\x00\x00Failed to revert volume: ' \
  261. b'some error\x00'
  262. with qubesadmin.tests.tools.StderrBuffer() as stderr:
  263. self.assertEqual(1,
  264. qubesadmin.tools.qvm_volume.main(
  265. ['revert', 'testvm:private'],
  266. app=self.app))
  267. self.assertIn('some error', stderr.getvalue())
  268. self.assertAllCalled()
  269. def test_022_revert_no_snapshots(self):
  270. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  271. b'0\x00testvm class=AppVM state=Running\n'
  272. self.app.expected_calls[
  273. ('testvm', 'admin.vm.volume.List', None, None)] = \
  274. b'0\x00root\nprivate\n'
  275. self.app.expected_calls[
  276. ('testvm', 'admin.vm.volume.ListSnapshots', 'private', None)] = \
  277. b'0\x00'
  278. with qubesadmin.tests.tools.StderrBuffer() as stderr:
  279. self.assertEqual(1,
  280. qubesadmin.tools.qvm_volume.main(
  281. ['revert', 'testvm:private'],
  282. app=self.app))
  283. self.assertIn('No snapshots', stderr.getvalue())
  284. self.assertAllCalled()
  285. def test_023_revert_specific(self):
  286. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  287. b'0\x00testvm class=AppVM state=Running\n'
  288. self.app.expected_calls[
  289. ('testvm', 'admin.vm.volume.List', None, None)] = \
  290. b'0\x00root\nprivate\n'
  291. self.app.expected_calls[
  292. ('testvm', 'admin.vm.volume.Revert', 'private', b'20050101')] = \
  293. b'0\x00'
  294. self.assertEqual(0,
  295. qubesadmin.tools.qvm_volume.main(
  296. ['revert', 'testvm:private', '20050101'],
  297. app=self.app))
  298. self.assertAllCalled()
  299. def test_030_set_revisions_to_keep(self):
  300. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  301. b'0\x00testvm class=AppVM state=Running\n'
  302. self.app.expected_calls[
  303. ('testvm', 'admin.vm.volume.List', None, None)] = \
  304. b'0\x00root\nprivate\n'
  305. self.app.expected_calls[
  306. ('testvm', 'admin.vm.volume.Set.revisions_to_keep', 'private',
  307. b'3')] = b'0\x00'
  308. self.assertEqual(0,
  309. qubesadmin.tools.qvm_volume.main(
  310. ['set', 'testvm:private', 'revisions_to_keep', '3'],
  311. app=self.app))
  312. self.assertAllCalled()
  313. def test_031_set_rw(self):
  314. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  315. b'0\x00testvm class=AppVM state=Running\n'
  316. self.app.expected_calls[
  317. ('testvm', 'admin.vm.volume.List', None, None)] = \
  318. b'0\x00root\nprivate\n'
  319. self.app.expected_calls[
  320. ('testvm', 'admin.vm.volume.Set.rw', 'private',
  321. b'True')] = b'0\x00'
  322. self.assertEqual(0,
  323. qubesadmin.tools.qvm_volume.main(
  324. ['set', 'testvm:private', 'rw', 'True'],
  325. app=self.app))
  326. self.assertAllCalled()
  327. def test_032_set_invalid(self):
  328. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  329. b'0\x00testvm class=AppVM state=Running\n'
  330. self.app.expected_calls[
  331. ('testvm', 'admin.vm.volume.List', None, None)] = \
  332. b'0\x00root\nprivate\n'
  333. self.assertNotEqual(0,
  334. qubesadmin.tools.qvm_volume.main(
  335. ['set', 'testvm:private', 'invalid', 'True'],
  336. app=self.app))
  337. self.assertAllCalled()
  338. def test_040_info(self):
  339. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  340. b'0\x00testvm class=AppVM state=Running\n'
  341. self.app.expected_calls[
  342. ('testvm', 'admin.vm.volume.List', None, None)] = \
  343. b'0\x00root\nprivate\n'
  344. self.app.expected_calls[
  345. ('testvm', 'admin.vm.volume.Info', 'private', None)] = \
  346. b'0\x00pool=lvm\n' \
  347. b'vid=qubes_dom0/vm-testvm-private\n' \
  348. b'size=2147483648\n' \
  349. b'usage=10000000\n' \
  350. b'rw=True\n' \
  351. b'source=\n' \
  352. b'save_on_stop=True\n' \
  353. b'snap_on_start=False\n' \
  354. b'revisions_to_keep=3\n' \
  355. b'is_outdated=False\n'
  356. self.app.expected_calls[
  357. ('testvm', 'admin.vm.volume.ListSnapshots', 'private', None)] = \
  358. b'0\x00200101010000\n200201010000\n200301010000\n'
  359. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  360. self.assertEqual(0,
  361. qubesadmin.tools.qvm_volume.main(['info', 'testvm:private'],
  362. app=self.app))
  363. output = stdout.getvalue()
  364. # travis...
  365. output = output.replace('\nsource\n', '\nsource \n')
  366. self.assertEqual(output,
  367. 'pool lvm\n'
  368. 'vid qubes_dom0/vm-testvm-private\n'
  369. 'rw True\n'
  370. 'source \n'
  371. 'save_on_stop True\n'
  372. 'snap_on_start False\n'
  373. 'size 2147483648\n'
  374. 'usage 10000000\n'
  375. 'revisions_to_keep 3\n'
  376. 'is_outdated False\n'
  377. 'Available revisions (for revert):\n'
  378. ' 200101010000\n'
  379. ' 200201010000\n'
  380. ' 200301010000\n')
  381. self.assertAllCalled()
  382. def test_041_info_no_revisions(self):
  383. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  384. b'0\x00testvm class=AppVM state=Running\n'
  385. self.app.expected_calls[
  386. ('testvm', 'admin.vm.volume.List', None, None)] = \
  387. b'0\x00root\nprivate\n'
  388. self.app.expected_calls[
  389. ('testvm', 'admin.vm.volume.Info', 'root', None)] = \
  390. b'0\x00pool=lvm\n' \
  391. b'vid=qubes_dom0/vm-testvm-root\n' \
  392. b'size=2147483648\n' \
  393. b'usage=10000000\n' \
  394. b'rw=True\n' \
  395. b'source=qubes_dom0/vm-fedora-26-root\n' \
  396. b'save_on_stop=False\n' \
  397. b'snap_on_start=True\n' \
  398. b'revisions_to_keep=0\n' \
  399. b'is_outdated=False\n'
  400. self.app.expected_calls[
  401. ('testvm', 'admin.vm.volume.ListSnapshots', 'root', None)] = \
  402. b'0\x00'
  403. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  404. self.assertEqual(0,
  405. qubesadmin.tools.qvm_volume.main(['info', 'testvm:root'],
  406. app=self.app))
  407. self.assertEqual(stdout.getvalue(),
  408. 'pool lvm\n'
  409. 'vid qubes_dom0/vm-testvm-root\n'
  410. 'rw True\n'
  411. 'source qubes_dom0/vm-fedora-26-root\n'
  412. 'save_on_stop False\n'
  413. 'snap_on_start True\n'
  414. 'size 2147483648\n'
  415. 'usage 10000000\n'
  416. 'revisions_to_keep 0\n'
  417. 'is_outdated False\n'
  418. 'Available revisions (for revert): none\n')
  419. self.assertAllCalled()
  420. def test_042_info_single_prop(self):
  421. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  422. b'0\x00testvm class=AppVM state=Running\n'
  423. self.app.expected_calls[
  424. ('testvm', 'admin.vm.volume.List', None, None)] = \
  425. b'0\x00root\nprivate\n'
  426. self.app.expected_calls[
  427. ('testvm', 'admin.vm.volume.Info', 'root', None)] = \
  428. b'0\x00pool=lvm\n' \
  429. b'vid=qubes_dom0/vm-testvm-root\n' \
  430. b'size=2147483648\n' \
  431. b'usage=10000000\n' \
  432. b'rw=True\n' \
  433. b'source=qubes_dom0/vm-fedora-26-root\n' \
  434. b'save_on_stop=False\n' \
  435. b'snap_on_start=True\n' \
  436. b'revisions_to_keep=0\n' \
  437. b'is_outdated=False\n'
  438. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  439. self.assertEqual(0,
  440. qubesadmin.tools.qvm_volume.main(
  441. ['info', 'testvm:root', 'usage'],
  442. app=self.app))
  443. self.assertEqual(stdout.getvalue(), '10000000\n')
  444. self.assertAllCalled()
  445. def test_043_info_revisions_only(self):
  446. self.app.expected_calls[('dom0', 'admin.vm.List', None, None)] = \
  447. b'0\x00testvm class=AppVM state=Running\n'
  448. self.app.expected_calls[
  449. ('testvm', 'admin.vm.volume.List', None, None)] = \
  450. b'0\x00root\nprivate\n'
  451. self.app.expected_calls[
  452. ('testvm', 'admin.vm.volume.ListSnapshots', 'private', None)] = \
  453. b'0\x00200101010000\n200201010000\n200301010000\n'
  454. with qubesadmin.tests.tools.StdoutBuffer() as stdout:
  455. self.assertEqual(0,
  456. qubesadmin.tools.qvm_volume.main(
  457. ['info', 'testvm:private', 'revisions'],
  458. app=self.app))
  459. self.assertEqual(stdout.getvalue(),
  460. '200101010000\n'
  461. '200201010000\n'
  462. '200301010000\n')
  463. self.assertAllCalled()