test_create.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. import sys
  2. import pytest
  3. from flexmock import flexmock
  4. from borgmatic.actions import create as module
  5. def test_create_borgmatic_manifest_creates_manifest_file():
  6. flexmock(module.os.path).should_receive('join').with_args(
  7. '/run/borgmatic', 'bootstrap', 'manifest.json'
  8. ).and_return('/run/borgmatic/bootstrap/manifest.json')
  9. flexmock(module.os.path).should_receive('exists').and_return(False)
  10. flexmock(module.os).should_receive('makedirs').and_return(True)
  11. flexmock(module.importlib.metadata).should_receive('version').and_return('1.0.0')
  12. flexmock(sys.modules['builtins']).should_receive('open').with_args(
  13. '/run/borgmatic/bootstrap/manifest.json', 'w'
  14. ).and_return(
  15. flexmock(
  16. __enter__=lambda *args: flexmock(write=lambda *args: None, close=lambda *args: None),
  17. __exit__=lambda *args: None,
  18. )
  19. )
  20. flexmock(module.json).should_receive('dump').and_return(True).once()
  21. module.create_borgmatic_manifest({}, 'test.yaml', '/run/borgmatic', False)
  22. def test_create_borgmatic_manifest_creates_manifest_file_with_custom_borgmatic_runtime_directory():
  23. flexmock(module.os.path).should_receive('join').with_args(
  24. '/run/borgmatic', 'bootstrap', 'manifest.json'
  25. ).and_return('/run/borgmatic/bootstrap/manifest.json')
  26. flexmock(module.os.path).should_receive('exists').and_return(False)
  27. flexmock(module.os).should_receive('makedirs').and_return(True)
  28. flexmock(module.importlib.metadata).should_receive('version').and_return('1.0.0')
  29. flexmock(sys.modules['builtins']).should_receive('open').with_args(
  30. '/run/borgmatic/bootstrap/manifest.json', 'w'
  31. ).and_return(
  32. flexmock(
  33. __enter__=lambda *args: flexmock(write=lambda *args: None, close=lambda *args: None),
  34. __exit__=lambda *args: None,
  35. )
  36. )
  37. flexmock(module.json).should_receive('dump').and_return(True).once()
  38. module.create_borgmatic_manifest(
  39. {'borgmatic_runtime_directory': '/borgmatic'}, 'test.yaml', '/run/borgmatic', False
  40. )
  41. def test_create_borgmatic_manifest_does_not_create_manifest_file_on_dry_run():
  42. flexmock(module.json).should_receive('dump').never()
  43. module.create_borgmatic_manifest({}, 'test.yaml', '/run/borgmatic', True)
  44. def test_expand_directory_with_basic_path_passes_it_through():
  45. flexmock(module.os.path).should_receive('expanduser').and_return('foo')
  46. flexmock(module.glob).should_receive('glob').and_return([])
  47. paths = module.expand_directory('foo', None)
  48. assert paths == ['foo']
  49. def test_expand_directory_with_glob_expands():
  50. flexmock(module.os.path).should_receive('expanduser').and_return('foo*')
  51. flexmock(module.glob).should_receive('glob').and_return(['foo', 'food'])
  52. paths = module.expand_directory('foo*', None)
  53. assert paths == ['foo', 'food']
  54. def test_expand_directory_with_working_directory_passes_it_through():
  55. flexmock(module.os.path).should_receive('expanduser').and_return('foo')
  56. flexmock(module.glob).should_receive('glob').with_args('/working/dir/foo').and_return([]).once()
  57. paths = module.expand_directory('foo', working_directory='/working/dir')
  58. assert paths == ['/working/dir/foo']
  59. def test_expand_directory_with_glob_passes_through_working_directory():
  60. flexmock(module.os.path).should_receive('expanduser').and_return('foo*')
  61. flexmock(module.glob).should_receive('glob').with_args('/working/dir/foo*').and_return(
  62. ['/working/dir/foo', '/working/dir/food']
  63. ).once()
  64. paths = module.expand_directory('foo*', working_directory='/working/dir')
  65. assert paths == ['/working/dir/foo', '/working/dir/food']
  66. def test_expand_directories_flattens_expanded_directories():
  67. flexmock(module).should_receive('expand_directory').with_args('~/foo', None).and_return(
  68. ['/root/foo']
  69. )
  70. flexmock(module).should_receive('expand_directory').with_args('bar*', None).and_return(
  71. ['bar', 'barf']
  72. )
  73. paths = module.expand_directories(('~/foo', 'bar*'))
  74. assert paths == ('/root/foo', 'bar', 'barf')
  75. def test_expand_directories_with_working_directory_passes_it_through():
  76. flexmock(module).should_receive('expand_directory').with_args('foo', '/working/dir').and_return(
  77. ['/working/dir/foo']
  78. )
  79. paths = module.expand_directories(('foo',), working_directory='/working/dir')
  80. assert paths == ('/working/dir/foo',)
  81. def test_expand_directories_considers_none_as_no_directories():
  82. paths = module.expand_directories(None, None)
  83. assert paths == ()
  84. def test_map_directories_to_devices_gives_device_id_per_path():
  85. flexmock(module.os.path).should_receive('exists').and_return(True)
  86. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  87. flexmock(module.os).should_receive('stat').with_args('/bar').and_return(flexmock(st_dev=66))
  88. device_map = module.map_directories_to_devices(('/foo', '/bar'))
  89. assert device_map == {
  90. '/foo': 55,
  91. '/bar': 66,
  92. }
  93. def test_map_directories_to_devices_with_missing_path_does_not_error():
  94. flexmock(module.os.path).should_receive('exists').and_return(True).and_return(False)
  95. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  96. flexmock(module.os).should_receive('stat').with_args('/bar').never()
  97. device_map = module.map_directories_to_devices(('/foo', '/bar'))
  98. assert device_map == {
  99. '/foo': 55,
  100. '/bar': None,
  101. }
  102. def test_map_directories_to_devices_uses_working_directory_to_construct_path():
  103. flexmock(module.os.path).should_receive('exists').and_return(True)
  104. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  105. flexmock(module.os).should_receive('stat').with_args('/working/dir/bar').and_return(
  106. flexmock(st_dev=66)
  107. )
  108. device_map = module.map_directories_to_devices(
  109. ('/foo', 'bar'), working_directory='/working/dir'
  110. )
  111. assert device_map == {
  112. '/foo': 55,
  113. 'bar': 66,
  114. }
  115. @pytest.mark.parametrize(
  116. 'directories,additional_directories,expected_directories',
  117. (
  118. ({'/': 1, '/root': 1}, {}, ['/']),
  119. ({'/': 1, '/root/': 1}, {}, ['/']),
  120. ({'/': 1, '/root': 2}, {}, ['/', '/root']),
  121. ({'/root': 1, '/': 1}, {}, ['/']),
  122. ({'/root': 1, '/root/foo': 1}, {}, ['/root']),
  123. ({'/root/': 1, '/root/foo': 1}, {}, ['/root/']),
  124. ({'/root': 1, '/root/foo/': 1}, {}, ['/root']),
  125. ({'/root': 1, '/root/foo': 2}, {}, ['/root', '/root/foo']),
  126. ({'/root/foo': 1, '/root': 1}, {}, ['/root']),
  127. ({'/root': None, '/root/foo': None}, {}, ['/root', '/root/foo']),
  128. ({'/root': 1, '/etc': 1, '/root/foo/bar': 1}, {}, ['/etc', '/root']),
  129. ({'/root': 1, '/root/foo': 1, '/root/foo/bar': 1}, {}, ['/root']),
  130. ({'/dup': 1, '/dup': 1}, {}, ['/dup']),
  131. ({'/foo': 1, '/bar': 1}, {}, ['/bar', '/foo']),
  132. ({'/foo': 1, '/bar': 2}, {}, ['/bar', '/foo']),
  133. ({'/root/foo': 1}, {'/root': 1}, []),
  134. ({'/root/foo': 1}, {'/root': 2}, ['/root/foo']),
  135. ({'/root/foo': 1}, {}, ['/root/foo']),
  136. ),
  137. )
  138. def test_deduplicate_directories_removes_child_paths_on_the_same_filesystem(
  139. directories, additional_directories, expected_directories
  140. ):
  141. assert (
  142. module.deduplicate_directories(directories, additional_directories) == expected_directories
  143. )
  144. def test_pattern_root_directories_deals_with_none_patterns():
  145. assert module.pattern_root_directories(patterns=None) == []
  146. def test_pattern_root_directories_parses_roots_and_ignores_others():
  147. assert module.pattern_root_directories(
  148. ['R /root', '+ /root/foo', '- /root/foo/bar', 'R /baz']
  149. ) == ['/root', '/baz']
  150. def test_process_source_directories_includes_source_directories_and_config_paths():
  151. flexmock(module.borgmatic.config.paths).should_receive('get_working_directory').and_return(
  152. '/working'
  153. )
  154. flexmock(module).should_receive('deduplicate_directories').and_return(
  155. ('foo', 'bar', 'test.yaml')
  156. )
  157. flexmock(module).should_receive('map_directories_to_devices').and_return({})
  158. flexmock(module).should_receive('expand_directories').with_args(
  159. ('foo', 'bar', 'test.yaml'), working_directory='/working'
  160. ).and_return(()).once()
  161. flexmock(module).should_receive('pattern_root_directories').and_return(())
  162. flexmock(module).should_receive('expand_directories').with_args(
  163. (), working_directory='/working'
  164. ).and_return(())
  165. assert module.process_source_directories(
  166. config={'source_directories': ['foo', 'bar']}, config_paths=('test.yaml',)
  167. ) == ('foo', 'bar', 'test.yaml')
  168. def test_process_source_directories_does_not_include_config_paths_when_store_config_files_is_false():
  169. flexmock(module.borgmatic.config.paths).should_receive('get_working_directory').and_return(
  170. '/working'
  171. )
  172. flexmock(module).should_receive('deduplicate_directories').and_return(('foo', 'bar'))
  173. flexmock(module).should_receive('map_directories_to_devices').and_return({})
  174. flexmock(module).should_receive('expand_directories').with_args(
  175. ('foo', 'bar'), working_directory='/working'
  176. ).and_return(()).once()
  177. flexmock(module).should_receive('pattern_root_directories').and_return(())
  178. flexmock(module).should_receive('expand_directories').with_args(
  179. (), working_directory='/working'
  180. ).and_return(())
  181. assert module.process_source_directories(
  182. config={'source_directories': ['foo', 'bar'], 'store_config_files': False},
  183. config_paths=('test.yaml',),
  184. ) == ('foo', 'bar')
  185. def test_process_source_directories_prefers_source_directory_argument_to_config():
  186. flexmock(module.borgmatic.config.paths).should_receive('get_working_directory').and_return(
  187. '/working'
  188. )
  189. flexmock(module).should_receive('deduplicate_directories').and_return(('foo', 'bar'))
  190. flexmock(module).should_receive('map_directories_to_devices').and_return({})
  191. flexmock(module).should_receive('expand_directories').with_args(
  192. ('foo', 'bar'), working_directory='/working'
  193. ).and_return(()).once()
  194. flexmock(module).should_receive('pattern_root_directories').and_return(())
  195. flexmock(module).should_receive('expand_directories').with_args(
  196. (), working_directory='/working'
  197. ).and_return(())
  198. assert module.process_source_directories(
  199. config={'source_directories': ['nope']},
  200. config_paths=('test.yaml',),
  201. source_directories=['foo', 'bar'],
  202. ) == ('foo', 'bar')
  203. def test_run_create_executes_and_calls_hooks_for_configured_repository():
  204. flexmock(module.logger).answer = lambda message: None
  205. flexmock(module.borgmatic.config.validate).should_receive('repositories_match').never()
  206. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  207. flexmock()
  208. )
  209. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once()
  210. flexmock(module).should_receive('create_borgmatic_manifest').once()
  211. flexmock(module.borgmatic.hooks.command).should_receive('execute_hook').times(2)
  212. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  213. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  214. 'call_hooks_even_if_unconfigured'
  215. ).and_return({})
  216. flexmock(module).should_receive('process_source_directories').and_return([])
  217. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  218. create_arguments = flexmock(
  219. repository=None,
  220. progress=flexmock(),
  221. stats=flexmock(),
  222. json=False,
  223. list_files=flexmock(),
  224. )
  225. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  226. list(
  227. module.run_create(
  228. config_filename='test.yaml',
  229. repository={'path': 'repo'},
  230. config={},
  231. config_paths=['/tmp/test.yaml'],
  232. hook_context={},
  233. local_borg_version=None,
  234. create_arguments=create_arguments,
  235. global_arguments=global_arguments,
  236. dry_run_label='',
  237. local_path=None,
  238. remote_path=None,
  239. )
  240. )
  241. def test_run_create_with_store_config_files_false_does_not_create_borgmatic_manifest():
  242. flexmock(module.logger).answer = lambda message: None
  243. flexmock(module.borgmatic.config.validate).should_receive('repositories_match').never()
  244. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  245. flexmock()
  246. )
  247. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once()
  248. flexmock(module).should_receive('create_borgmatic_manifest').never()
  249. flexmock(module.borgmatic.hooks.command).should_receive('execute_hook').times(2)
  250. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  251. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  252. 'call_hooks_even_if_unconfigured'
  253. ).and_return({})
  254. flexmock(module).should_receive('process_source_directories').and_return([])
  255. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  256. create_arguments = flexmock(
  257. repository=None,
  258. progress=flexmock(),
  259. stats=flexmock(),
  260. json=False,
  261. list_files=flexmock(),
  262. )
  263. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  264. list(
  265. module.run_create(
  266. config_filename='test.yaml',
  267. repository={'path': 'repo'},
  268. config={'store_config_files': False},
  269. config_paths=['/tmp/test.yaml'],
  270. hook_context={},
  271. local_borg_version=None,
  272. create_arguments=create_arguments,
  273. global_arguments=global_arguments,
  274. dry_run_label='',
  275. local_path=None,
  276. remote_path=None,
  277. )
  278. )
  279. def test_run_create_runs_with_selected_repository():
  280. flexmock(module.logger).answer = lambda message: None
  281. flexmock(module.borgmatic.config.validate).should_receive(
  282. 'repositories_match'
  283. ).once().and_return(True)
  284. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  285. flexmock()
  286. )
  287. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once()
  288. flexmock(module).should_receive('create_borgmatic_manifest').once()
  289. flexmock(module.borgmatic.hooks.command).should_receive('execute_hook').times(2)
  290. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  291. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  292. 'call_hooks_even_if_unconfigured'
  293. ).and_return({})
  294. flexmock(module).should_receive('process_source_directories').and_return([])
  295. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  296. create_arguments = flexmock(
  297. repository=flexmock(),
  298. progress=flexmock(),
  299. stats=flexmock(),
  300. json=False,
  301. list_files=flexmock(),
  302. )
  303. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  304. list(
  305. module.run_create(
  306. config_filename='test.yaml',
  307. repository={'path': 'repo'},
  308. config={},
  309. config_paths=['/tmp/test.yaml'],
  310. hook_context={},
  311. local_borg_version=None,
  312. create_arguments=create_arguments,
  313. global_arguments=global_arguments,
  314. dry_run_label='',
  315. local_path=None,
  316. remote_path=None,
  317. )
  318. )
  319. def test_run_create_bails_if_repository_does_not_match():
  320. flexmock(module.logger).answer = lambda message: None
  321. flexmock(module.borgmatic.config.validate).should_receive(
  322. 'repositories_match'
  323. ).once().and_return(False)
  324. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').never()
  325. flexmock(module.borgmatic.borg.create).should_receive('create_archive').never()
  326. flexmock(module).should_receive('create_borgmatic_manifest').never()
  327. create_arguments = flexmock(
  328. repository=flexmock(),
  329. progress=flexmock(),
  330. stats=flexmock(),
  331. json=False,
  332. list_files=flexmock(),
  333. )
  334. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  335. list(
  336. module.run_create(
  337. config_filename='test.yaml',
  338. repository='repo',
  339. config={},
  340. config_paths=['/tmp/test.yaml'],
  341. hook_context={},
  342. local_borg_version=None,
  343. create_arguments=create_arguments,
  344. global_arguments=global_arguments,
  345. dry_run_label='',
  346. local_path=None,
  347. remote_path=None,
  348. )
  349. )
  350. def test_run_create_produces_json():
  351. flexmock(module.logger).answer = lambda message: None
  352. flexmock(module.borgmatic.config.validate).should_receive(
  353. 'repositories_match'
  354. ).once().and_return(True)
  355. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  356. flexmock()
  357. )
  358. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once().and_return(
  359. flexmock()
  360. )
  361. parsed_json = flexmock()
  362. flexmock(module.borgmatic.actions.json).should_receive('parse_json').and_return(parsed_json)
  363. flexmock(module).should_receive('create_borgmatic_manifest').once()
  364. flexmock(module.borgmatic.hooks.command).should_receive('execute_hook').times(2)
  365. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  366. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  367. 'call_hooks_even_if_unconfigured'
  368. ).and_return({})
  369. flexmock(module).should_receive('process_source_directories').and_return([])
  370. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  371. create_arguments = flexmock(
  372. repository=flexmock(),
  373. progress=flexmock(),
  374. stats=flexmock(),
  375. json=True,
  376. list_files=flexmock(),
  377. )
  378. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  379. assert list(
  380. module.run_create(
  381. config_filename='test.yaml',
  382. repository={'path': 'repo'},
  383. config={},
  384. config_paths=['/tmp/test.yaml'],
  385. hook_context={},
  386. local_borg_version=None,
  387. create_arguments=create_arguments,
  388. global_arguments=global_arguments,
  389. dry_run_label='',
  390. local_path=None,
  391. remote_path=None,
  392. )
  393. ) == [parsed_json]