test_create.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  1. import pytest
  2. from flexmock import flexmock
  3. from borgmatic.actions import create as module
  4. def test_expand_directory_with_basic_path_passes_it_through():
  5. flexmock(module.os.path).should_receive('expanduser').and_return('foo')
  6. flexmock(module.glob).should_receive('glob').and_return([])
  7. paths = module.expand_directory('foo', None)
  8. assert paths == ['foo']
  9. def test_expand_directory_with_glob_expands():
  10. flexmock(module.os.path).should_receive('expanduser').and_return('foo*')
  11. flexmock(module.glob).should_receive('glob').and_return(['foo', 'food'])
  12. paths = module.expand_directory('foo*', None)
  13. assert paths == ['foo', 'food']
  14. def test_expand_directory_strips_off_working_directory():
  15. flexmock(module.os.path).should_receive('expanduser').and_return('foo')
  16. flexmock(module.glob).should_receive('glob').with_args('/working/dir/foo').and_return([]).once()
  17. paths = module.expand_directory('foo', working_directory='/working/dir')
  18. assert paths == ['foo']
  19. def test_expand_directory_globs_working_directory_and_strips_it_off():
  20. flexmock(module.os.path).should_receive('expanduser').and_return('foo*')
  21. flexmock(module.glob).should_receive('glob').with_args('/working/dir/foo*').and_return(
  22. ['/working/dir/foo', '/working/dir/food']
  23. ).once()
  24. paths = module.expand_directory('foo*', working_directory='/working/dir')
  25. assert paths == ['foo', 'food']
  26. def test_expand_directory_with_slashdot_hack_globs_working_directory_and_strips_it_off():
  27. flexmock(module.os.path).should_receive('expanduser').and_return('./foo*')
  28. flexmock(module.glob).should_receive('glob').with_args('/working/dir/./foo*').and_return(
  29. ['/working/dir/./foo', '/working/dir/./food']
  30. ).once()
  31. paths = module.expand_directory('./foo*', working_directory='/working/dir')
  32. assert paths == ['./foo', './food']
  33. def test_expand_directories_flattens_expanded_directories():
  34. flexmock(module).should_receive('expand_directory').with_args('~/foo', None).and_return(
  35. ['/root/foo']
  36. )
  37. flexmock(module).should_receive('expand_directory').with_args('bar*', None).and_return(
  38. ['bar', 'barf']
  39. )
  40. paths = module.expand_directories(('~/foo', 'bar*'))
  41. assert paths == ('/root/foo', 'bar', 'barf')
  42. def test_expand_directories_with_working_directory_passes_it_through():
  43. flexmock(module).should_receive('expand_directory').with_args('foo', '/working/dir').and_return(
  44. ['/working/dir/foo']
  45. )
  46. paths = module.expand_directories(('foo',), working_directory='/working/dir')
  47. assert paths == ('/working/dir/foo',)
  48. def test_expand_directories_considers_none_as_no_directories():
  49. paths = module.expand_directories(None, None)
  50. assert paths == ()
  51. def test_map_directories_to_devices_gives_device_id_per_path():
  52. flexmock(module.os.path).should_receive('exists').and_return(True)
  53. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  54. flexmock(module.os).should_receive('stat').with_args('/bar').and_return(flexmock(st_dev=66))
  55. device_map = module.map_directories_to_devices(('/foo', '/bar'))
  56. assert device_map == {
  57. '/foo': 55,
  58. '/bar': 66,
  59. }
  60. def test_map_directories_to_devices_with_missing_path_does_not_error():
  61. flexmock(module.os.path).should_receive('exists').and_return(True).and_return(False)
  62. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  63. flexmock(module.os).should_receive('stat').with_args('/bar').never()
  64. device_map = module.map_directories_to_devices(('/foo', '/bar'))
  65. assert device_map == {
  66. '/foo': 55,
  67. '/bar': None,
  68. }
  69. def test_map_directories_to_devices_uses_working_directory_to_construct_path():
  70. flexmock(module.os.path).should_receive('exists').and_return(True)
  71. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  72. flexmock(module.os).should_receive('stat').with_args('/working/dir/bar').and_return(
  73. flexmock(st_dev=66)
  74. )
  75. device_map = module.map_directories_to_devices(
  76. ('/foo', 'bar'), working_directory='/working/dir'
  77. )
  78. assert device_map == {
  79. '/foo': 55,
  80. 'bar': 66,
  81. }
  82. @pytest.mark.parametrize(
  83. 'directories,additional_directories,expected_directories',
  84. (
  85. ({'/': 1, '/root': 1}, {}, ['/']),
  86. ({'/': 1, '/root/': 1}, {}, ['/']),
  87. ({'/': 1, '/root': 2}, {}, ['/', '/root']),
  88. ({'/root': 1, '/': 1}, {}, ['/']),
  89. ({'/root': 1, '/root/foo': 1}, {}, ['/root']),
  90. ({'/root/': 1, '/root/foo': 1}, {}, ['/root/']),
  91. ({'/root': 1, '/root/foo/': 1}, {}, ['/root']),
  92. ({'/root': 1, '/root/foo': 2}, {}, ['/root', '/root/foo']),
  93. ({'/root/foo': 1, '/root': 1}, {}, ['/root']),
  94. ({'/root': None, '/root/foo': None}, {}, ['/root', '/root/foo']),
  95. ({'/root': 1, '/etc': 1, '/root/foo/bar': 1}, {}, ['/etc', '/root']),
  96. ({'/root': 1, '/root/foo': 1, '/root/foo/bar': 1}, {}, ['/root']),
  97. ({'/dup': 1, '/dup': 1}, {}, ['/dup']),
  98. ({'/foo': 1, '/bar': 1}, {}, ['/bar', '/foo']),
  99. ({'/foo': 1, '/bar': 2}, {}, ['/bar', '/foo']),
  100. ({'/root/foo': 1}, {'/root': 1}, []),
  101. ({'/root/foo': 1}, {'/root': 2}, ['/root/foo']),
  102. ({'/root/foo': 1}, {}, ['/root/foo']),
  103. ),
  104. )
  105. def test_deduplicate_directories_removes_child_paths_on_the_same_filesystem(
  106. directories, additional_directories, expected_directories
  107. ):
  108. assert (
  109. module.deduplicate_directories(directories, additional_directories) == expected_directories
  110. )
  111. def test_pattern_root_directories_deals_with_none_patterns():
  112. assert module.pattern_root_directories(patterns=None) == []
  113. def test_pattern_root_directories_parses_roots_and_ignores_others():
  114. assert module.pattern_root_directories(
  115. ['R /root', '+ /root/foo', '- /root/foo/bar', 'R /baz']
  116. ) == ['/root', '/baz']
  117. def test_process_source_directories_includes_source_directories():
  118. flexmock(module.borgmatic.config.paths).should_receive('get_working_directory').and_return(
  119. '/working'
  120. )
  121. flexmock(module).should_receive('deduplicate_directories').and_return(('foo', 'bar'))
  122. flexmock(module).should_receive('map_directories_to_devices').and_return({})
  123. flexmock(module).should_receive('expand_directories').with_args(
  124. ('foo', 'bar'), working_directory='/working'
  125. ).and_return(()).once()
  126. flexmock(module).should_receive('pattern_root_directories').and_return(())
  127. flexmock(module).should_receive('expand_directories').with_args(
  128. (), working_directory='/working'
  129. ).and_return(())
  130. assert module.process_source_directories(
  131. config={'source_directories': ['foo', 'bar']},
  132. ) == ('foo', 'bar')
  133. def test_process_source_directories_prefers_source_directory_argument_to_config():
  134. flexmock(module.borgmatic.config.paths).should_receive('get_working_directory').and_return(
  135. '/working'
  136. )
  137. flexmock(module).should_receive('deduplicate_directories').and_return(('foo', 'bar'))
  138. flexmock(module).should_receive('map_directories_to_devices').and_return({})
  139. flexmock(module).should_receive('expand_directories').with_args(
  140. ('foo', 'bar'), working_directory='/working'
  141. ).and_return(()).once()
  142. flexmock(module).should_receive('pattern_root_directories').and_return(())
  143. flexmock(module).should_receive('expand_directories').with_args(
  144. (), working_directory='/working'
  145. ).and_return(())
  146. assert module.process_source_directories(
  147. config={'source_directories': ['nope']},
  148. source_directories=['foo', 'bar'],
  149. ) == ('foo', 'bar')
  150. def test_run_create_executes_and_calls_hooks_for_configured_repository():
  151. flexmock(module.logger).answer = lambda message: None
  152. flexmock(module.borgmatic.config.validate).should_receive('repositories_match').never()
  153. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  154. flexmock()
  155. )
  156. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once()
  157. flexmock(module.borgmatic.hooks.command).should_receive('execute_hook').times(2)
  158. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  159. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  160. 'call_hooks_even_if_unconfigured'
  161. ).and_return({})
  162. flexmock(module).should_receive('process_source_directories').and_return([])
  163. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  164. create_arguments = flexmock(
  165. repository=None,
  166. progress=flexmock(),
  167. stats=flexmock(),
  168. json=False,
  169. list_files=flexmock(),
  170. )
  171. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  172. list(
  173. module.run_create(
  174. config_filename='test.yaml',
  175. repository={'path': 'repo'},
  176. config={},
  177. config_paths=['/tmp/test.yaml'],
  178. hook_context={},
  179. local_borg_version=None,
  180. create_arguments=create_arguments,
  181. global_arguments=global_arguments,
  182. dry_run_label='',
  183. local_path=None,
  184. remote_path=None,
  185. )
  186. )
  187. def test_run_create_runs_with_selected_repository():
  188. flexmock(module.logger).answer = lambda message: None
  189. flexmock(module.borgmatic.config.validate).should_receive(
  190. 'repositories_match'
  191. ).once().and_return(True)
  192. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  193. flexmock()
  194. )
  195. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once()
  196. flexmock(module.borgmatic.hooks.command).should_receive('execute_hook').times(2)
  197. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  198. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  199. 'call_hooks_even_if_unconfigured'
  200. ).and_return({})
  201. flexmock(module).should_receive('process_source_directories').and_return([])
  202. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  203. create_arguments = flexmock(
  204. repository=flexmock(),
  205. progress=flexmock(),
  206. stats=flexmock(),
  207. json=False,
  208. list_files=flexmock(),
  209. )
  210. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  211. list(
  212. module.run_create(
  213. config_filename='test.yaml',
  214. repository={'path': 'repo'},
  215. config={},
  216. config_paths=['/tmp/test.yaml'],
  217. hook_context={},
  218. local_borg_version=None,
  219. create_arguments=create_arguments,
  220. global_arguments=global_arguments,
  221. dry_run_label='',
  222. local_path=None,
  223. remote_path=None,
  224. )
  225. )
  226. def test_run_create_bails_if_repository_does_not_match():
  227. flexmock(module.logger).answer = lambda message: None
  228. flexmock(module.borgmatic.config.validate).should_receive(
  229. 'repositories_match'
  230. ).once().and_return(False)
  231. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').never()
  232. flexmock(module.borgmatic.borg.create).should_receive('create_archive').never()
  233. create_arguments = flexmock(
  234. repository=flexmock(),
  235. progress=flexmock(),
  236. stats=flexmock(),
  237. json=False,
  238. list_files=flexmock(),
  239. )
  240. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  241. list(
  242. module.run_create(
  243. config_filename='test.yaml',
  244. repository='repo',
  245. config={},
  246. config_paths=['/tmp/test.yaml'],
  247. hook_context={},
  248. local_borg_version=None,
  249. create_arguments=create_arguments,
  250. global_arguments=global_arguments,
  251. dry_run_label='',
  252. local_path=None,
  253. remote_path=None,
  254. )
  255. )
  256. def test_run_create_produces_json():
  257. flexmock(module.logger).answer = lambda message: None
  258. flexmock(module.borgmatic.config.validate).should_receive(
  259. 'repositories_match'
  260. ).once().and_return(True)
  261. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  262. flexmock()
  263. )
  264. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once().and_return(
  265. flexmock()
  266. )
  267. parsed_json = flexmock()
  268. flexmock(module.borgmatic.actions.json).should_receive('parse_json').and_return(parsed_json)
  269. flexmock(module.borgmatic.hooks.command).should_receive('execute_hook').times(2)
  270. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  271. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  272. 'call_hooks_even_if_unconfigured'
  273. ).and_return({})
  274. flexmock(module).should_receive('process_source_directories').and_return([])
  275. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  276. create_arguments = flexmock(
  277. repository=flexmock(),
  278. progress=flexmock(),
  279. stats=flexmock(),
  280. json=True,
  281. list_files=flexmock(),
  282. )
  283. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  284. assert list(
  285. module.run_create(
  286. config_filename='test.yaml',
  287. repository={'path': 'repo'},
  288. config={},
  289. config_paths=['/tmp/test.yaml'],
  290. hook_context={},
  291. local_borg_version=None,
  292. create_arguments=create_arguments,
  293. global_arguments=global_arguments,
  294. dry_run_label='',
  295. local_path=None,
  296. remote_path=None,
  297. )
  298. ) == [parsed_json]