test_create.py 13 KB

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