test_create.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649
  1. import io
  2. import sys
  3. import pytest
  4. from flexmock import flexmock
  5. from borgmatic.actions import create as module
  6. from borgmatic.borg.pattern import Pattern, Pattern_source, Pattern_style, Pattern_type
  7. @pytest.mark.parametrize(
  8. 'pattern_line,expected_pattern',
  9. (
  10. ('R /foo', Pattern('/foo', source=Pattern_source.CONFIG)),
  11. ('P sh', Pattern('sh', Pattern_type.PATTERN_STYLE, source=Pattern_source.CONFIG)),
  12. ('+ /foo*', Pattern('/foo*', Pattern_type.INCLUDE, source=Pattern_source.CONFIG)),
  13. (
  14. '+ sh:/foo*',
  15. Pattern(
  16. '/foo*', Pattern_type.INCLUDE, Pattern_style.SHELL, source=Pattern_source.CONFIG
  17. ),
  18. ),
  19. ),
  20. )
  21. def test_parse_pattern_transforms_pattern_line_to_instance(pattern_line, expected_pattern):
  22. module.parse_pattern(pattern_line) == expected_pattern
  23. def test_parse_pattern_with_invalid_pattern_line_errors():
  24. with pytest.raises(ValueError):
  25. module.parse_pattern('/foo')
  26. def test_collect_patterns_converts_source_directories():
  27. assert module.collect_patterns({'source_directories': ['/foo', '/bar']}) == (
  28. Pattern('/foo', source=Pattern_source.CONFIG),
  29. Pattern('/bar', source=Pattern_source.CONFIG),
  30. )
  31. def test_collect_patterns_parses_config_patterns():
  32. flexmock(module).should_receive('parse_pattern').with_args('R /foo').and_return(Pattern('/foo'))
  33. flexmock(module).should_receive('parse_pattern').with_args('# comment').never()
  34. flexmock(module).should_receive('parse_pattern').with_args('').never()
  35. flexmock(module).should_receive('parse_pattern').with_args(' ').never()
  36. flexmock(module).should_receive('parse_pattern').with_args('R /bar').and_return(Pattern('/bar'))
  37. assert module.collect_patterns({'patterns': ['R /foo', '# comment', '', ' ', 'R /bar']}) == (
  38. Pattern('/foo'),
  39. Pattern('/bar'),
  40. )
  41. def test_collect_patterns_converts_exclude_patterns():
  42. assert module.collect_patterns({'exclude_patterns': ['/foo', '/bar', 'sh:**/baz']}) == (
  43. Pattern(
  44. '/foo', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH, source=Pattern_source.CONFIG
  45. ),
  46. Pattern(
  47. '/bar', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH, source=Pattern_source.CONFIG
  48. ),
  49. Pattern(
  50. '**/baz', Pattern_type.NO_RECURSE, Pattern_style.SHELL, source=Pattern_source.CONFIG
  51. ),
  52. )
  53. def test_collect_patterns_reads_config_patterns_from_file():
  54. builtins = flexmock(sys.modules['builtins'])
  55. builtins.should_receive('open').with_args('file1.txt').and_return(io.StringIO('R /foo'))
  56. builtins.should_receive('open').with_args('file2.txt').and_return(
  57. io.StringIO('R /bar\n# comment\n\n \nR /baz')
  58. )
  59. flexmock(module).should_receive('parse_pattern').with_args('R /foo').and_return(Pattern('/foo'))
  60. flexmock(module).should_receive('parse_pattern').with_args('# comment').never()
  61. flexmock(module).should_receive('parse_pattern').with_args('').never()
  62. flexmock(module).should_receive('parse_pattern').with_args(' ').never()
  63. flexmock(module).should_receive('parse_pattern').with_args('R /bar').and_return(Pattern('/bar'))
  64. flexmock(module).should_receive('parse_pattern').with_args('R /baz').and_return(Pattern('/baz'))
  65. assert module.collect_patterns({'patterns_from': ['file1.txt', 'file2.txt']}) == (
  66. Pattern('/foo'),
  67. Pattern('/bar'),
  68. Pattern('/baz'),
  69. )
  70. def test_collect_patterns_errors_on_missing_config_patterns_from_file():
  71. builtins = flexmock(sys.modules['builtins'])
  72. builtins.should_receive('open').with_args('file1.txt').and_raise(FileNotFoundError)
  73. flexmock(module).should_receive('parse_pattern').never()
  74. with pytest.raises(ValueError):
  75. module.collect_patterns({'patterns_from': ['file1.txt', 'file2.txt']})
  76. def test_collect_patterns_reads_config_exclude_from_file():
  77. builtins = flexmock(sys.modules['builtins'])
  78. builtins.should_receive('open').with_args('file1.txt').and_return(io.StringIO('/foo'))
  79. builtins.should_receive('open').with_args('file2.txt').and_return(
  80. io.StringIO('/bar\n# comment\n\n \n/baz')
  81. )
  82. flexmock(module).should_receive('parse_pattern').with_args(
  83. '! /foo', default_style=Pattern_style.FNMATCH
  84. ).and_return(Pattern('/foo', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH))
  85. flexmock(module).should_receive('parse_pattern').with_args(
  86. '! /bar', default_style=Pattern_style.FNMATCH
  87. ).and_return(Pattern('/bar', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH))
  88. flexmock(module).should_receive('parse_pattern').with_args('# comment').never()
  89. flexmock(module).should_receive('parse_pattern').with_args('').never()
  90. flexmock(module).should_receive('parse_pattern').with_args(' ').never()
  91. flexmock(module).should_receive('parse_pattern').with_args(
  92. '! /baz', default_style=Pattern_style.FNMATCH
  93. ).and_return(Pattern('/baz', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH))
  94. assert module.collect_patterns({'exclude_from': ['file1.txt', 'file2.txt']}) == (
  95. Pattern('/foo', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH),
  96. Pattern('/bar', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH),
  97. Pattern('/baz', Pattern_type.NO_RECURSE, Pattern_style.FNMATCH),
  98. )
  99. def test_collect_patterns_errors_on_missing_config_exclude_from_file():
  100. builtins = flexmock(sys.modules['builtins'])
  101. builtins.should_receive('open').with_args('file1.txt').and_raise(OSError)
  102. flexmock(module).should_receive('parse_pattern').never()
  103. with pytest.raises(ValueError):
  104. module.collect_patterns({'exclude_from': ['file1.txt', 'file2.txt']})
  105. def test_expand_directory_with_basic_path_passes_it_through():
  106. flexmock(module.os.path).should_receive('expanduser').and_return('foo')
  107. flexmock(module.glob).should_receive('glob').and_return([])
  108. paths = module.expand_directory('foo', None)
  109. assert paths == ['foo']
  110. def test_expand_directory_with_glob_expands():
  111. flexmock(module.os.path).should_receive('expanduser').and_return('foo*')
  112. flexmock(module.glob).should_receive('glob').and_return(['foo', 'food'])
  113. paths = module.expand_directory('foo*', None)
  114. assert paths == ['foo', 'food']
  115. def test_expand_directory_strips_off_working_directory():
  116. flexmock(module.os.path).should_receive('expanduser').and_return('foo')
  117. flexmock(module.glob).should_receive('glob').with_args('/working/dir/foo').and_return([]).once()
  118. paths = module.expand_directory('foo', working_directory='/working/dir')
  119. assert paths == ['foo']
  120. def test_expand_directory_globs_working_directory_and_strips_it_off():
  121. flexmock(module.os.path).should_receive('expanduser').and_return('foo*')
  122. flexmock(module.glob).should_receive('glob').with_args('/working/dir/foo*').and_return(
  123. ['/working/dir/foo', '/working/dir/food']
  124. ).once()
  125. paths = module.expand_directory('foo*', working_directory='/working/dir')
  126. assert paths == ['foo', 'food']
  127. def test_expand_directory_with_slashdot_hack_globs_working_directory_and_strips_it_off():
  128. flexmock(module.os.path).should_receive('expanduser').and_return('./foo*')
  129. flexmock(module.glob).should_receive('glob').with_args('/working/dir/./foo*').and_return(
  130. ['/working/dir/./foo', '/working/dir/./food']
  131. ).once()
  132. paths = module.expand_directory('./foo*', working_directory='/working/dir')
  133. assert paths == ['./foo', './food']
  134. def test_expand_directory_with_working_directory_matching_start_of_directory_does_not_strip_it_off():
  135. flexmock(module.os.path).should_receive('expanduser').and_return('/working/dir/foo')
  136. flexmock(module.glob).should_receive('glob').with_args('/working/dir/foo').and_return(
  137. ['/working/dir/foo']
  138. ).once()
  139. paths = module.expand_directory('/working/dir/foo', working_directory='/working/dir')
  140. assert paths == ['/working/dir/foo']
  141. def test_expand_patterns_flattens_expanded_directories():
  142. flexmock(module).should_receive('expand_directory').with_args('~/foo', None).and_return(
  143. ['/root/foo']
  144. )
  145. flexmock(module).should_receive('expand_directory').with_args('bar*', None).and_return(
  146. ['bar', 'barf']
  147. )
  148. paths = module.expand_patterns((Pattern('~/foo'), Pattern('bar*')))
  149. assert paths == (Pattern('/root/foo'), Pattern('bar'), Pattern('barf'))
  150. def test_expand_patterns_with_working_directory_passes_it_through():
  151. flexmock(module).should_receive('expand_directory').with_args('foo', '/working/dir').and_return(
  152. ['/working/dir/foo']
  153. )
  154. patterns = module.expand_patterns((Pattern('foo'),), working_directory='/working/dir')
  155. assert patterns == (Pattern('/working/dir/foo'),)
  156. def test_expand_patterns_does_not_expand_skip_paths():
  157. flexmock(module).should_receive('expand_directory').with_args('/foo', None).and_return(['/foo'])
  158. flexmock(module).should_receive('expand_directory').with_args('/bar*', None).never()
  159. patterns = module.expand_patterns((Pattern('/foo'), Pattern('/bar*')), skip_paths=('/bar*',))
  160. assert patterns == (Pattern('/foo'), Pattern('/bar*'))
  161. def test_expand_patterns_considers_none_as_no_patterns():
  162. assert module.expand_patterns(None) == ()
  163. def test_expand_patterns_expands_tildes_and_globs_in_root_patterns():
  164. flexmock(module.os.path).should_receive('expanduser').never()
  165. flexmock(module).should_receive('expand_directory').and_return(
  166. ['/root/foo/one', '/root/foo/two']
  167. )
  168. paths = module.expand_patterns((Pattern('~/foo/*'),))
  169. assert paths == (Pattern('/root/foo/one'), Pattern('/root/foo/two'))
  170. def test_expand_patterns_expands_only_tildes_in_non_root_patterns():
  171. flexmock(module).should_receive('expand_directory').never()
  172. flexmock(module.os.path).should_receive('expanduser').and_return('/root/bar/*')
  173. paths = module.expand_patterns((Pattern('~/bar/*', Pattern_type.INCLUDE),))
  174. assert paths == (Pattern('/root/bar/*', Pattern_type.INCLUDE),)
  175. def test_device_map_patterns_gives_device_id_per_path():
  176. flexmock(module.os.path).should_receive('exists').and_return(True)
  177. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  178. flexmock(module.os).should_receive('stat').with_args('/bar').and_return(flexmock(st_dev=66))
  179. device_map = module.device_map_patterns((Pattern('/foo'), Pattern('/bar')))
  180. assert device_map == (
  181. Pattern('/foo', device=55),
  182. Pattern('/bar', device=66),
  183. )
  184. def test_device_map_patterns_only_considers_root_patterns():
  185. flexmock(module.os.path).should_receive('exists').and_return(True)
  186. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  187. flexmock(module.os).should_receive('stat').with_args('/bar*').never()
  188. device_map = module.device_map_patterns(
  189. (Pattern('/foo'), Pattern('/bar*', Pattern_type.INCLUDE))
  190. )
  191. assert device_map == (
  192. Pattern('/foo', device=55),
  193. Pattern('/bar*', Pattern_type.INCLUDE),
  194. )
  195. def test_device_map_patterns_with_missing_path_does_not_error():
  196. flexmock(module.os.path).should_receive('exists').and_return(True).and_return(False)
  197. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  198. flexmock(module.os).should_receive('stat').with_args('/bar').never()
  199. device_map = module.device_map_patterns((Pattern('/foo'), Pattern('/bar')))
  200. assert device_map == (
  201. Pattern('/foo', device=55),
  202. Pattern('/bar'),
  203. )
  204. def test_device_map_patterns_uses_working_directory_to_construct_path():
  205. flexmock(module.os.path).should_receive('exists').and_return(True)
  206. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  207. flexmock(module.os).should_receive('stat').with_args('/working/dir/bar').and_return(
  208. flexmock(st_dev=66)
  209. )
  210. device_map = module.device_map_patterns(
  211. (Pattern('/foo'), Pattern('bar')), working_directory='/working/dir'
  212. )
  213. assert device_map == (
  214. Pattern('/foo', device=55),
  215. Pattern('bar', device=66),
  216. )
  217. def test_device_map_patterns_with_existing_device_id_does_not_overwrite_it():
  218. flexmock(module.os.path).should_receive('exists').and_return(True)
  219. flexmock(module.os).should_receive('stat').with_args('/foo').and_return(flexmock(st_dev=55))
  220. flexmock(module.os).should_receive('stat').with_args('/bar').and_return(flexmock(st_dev=100))
  221. device_map = module.device_map_patterns((Pattern('/foo'), Pattern('/bar', device=66)))
  222. assert device_map == (
  223. Pattern('/foo', device=55),
  224. Pattern('/bar', device=66),
  225. )
  226. @pytest.mark.parametrize(
  227. 'patterns,expected_patterns',
  228. (
  229. ((Pattern('/', device=1), Pattern('/root', device=1)), (Pattern('/', device=1),)),
  230. ((Pattern('/', device=1), Pattern('/root/', device=1)), (Pattern('/', device=1),)),
  231. (
  232. (Pattern('/', device=1), Pattern('/root', device=2)),
  233. (Pattern('/', device=1), Pattern('/root', device=2)),
  234. ),
  235. ((Pattern('/root', device=1), Pattern('/', device=1)), (Pattern('/', device=1),)),
  236. (
  237. (Pattern('/root', device=1), Pattern('/root/foo', device=1)),
  238. (Pattern('/root', device=1),),
  239. ),
  240. (
  241. (Pattern('/root/', device=1), Pattern('/root/foo', device=1)),
  242. (Pattern('/root/', device=1),),
  243. ),
  244. (
  245. (Pattern('/root', device=1), Pattern('/root/foo/', device=1)),
  246. (Pattern('/root', device=1),),
  247. ),
  248. (
  249. (Pattern('/root', device=1), Pattern('/root/foo', device=2)),
  250. (Pattern('/root', device=1), Pattern('/root/foo', device=2)),
  251. ),
  252. (
  253. (Pattern('/root/foo', device=1), Pattern('/root', device=1)),
  254. (Pattern('/root', device=1),),
  255. ),
  256. (
  257. (Pattern('/root', device=None), Pattern('/root/foo', device=None)),
  258. (Pattern('/root'), Pattern('/root/foo')),
  259. ),
  260. (
  261. (
  262. Pattern('/root', device=1),
  263. Pattern('/etc', device=1),
  264. Pattern('/root/foo/bar', device=1),
  265. ),
  266. (Pattern('/root', device=1), Pattern('/etc', device=1)),
  267. ),
  268. (
  269. (
  270. Pattern('/root', device=1),
  271. Pattern('/root/foo', device=1),
  272. Pattern('/root/foo/bar', device=1),
  273. ),
  274. (Pattern('/root', device=1),),
  275. ),
  276. ((Pattern('/dup', device=1), Pattern('/dup', device=1)), (Pattern('/dup', device=1),)),
  277. (
  278. (Pattern('/foo', device=1), Pattern('/bar', device=1)),
  279. (Pattern('/foo', device=1), Pattern('/bar', device=1)),
  280. ),
  281. (
  282. (Pattern('/foo', device=1), Pattern('/bar', device=2)),
  283. (Pattern('/foo', device=1), Pattern('/bar', device=2)),
  284. ),
  285. ((Pattern('/root/foo', device=1),), (Pattern('/root/foo', device=1),)),
  286. (
  287. (Pattern('/', device=1), Pattern('/root', Pattern_type.INCLUDE, device=1)),
  288. (Pattern('/', device=1), Pattern('/root', Pattern_type.INCLUDE, device=1)),
  289. ),
  290. (
  291. (Pattern('/root', Pattern_type.INCLUDE, device=1), Pattern('/', device=1)),
  292. (Pattern('/root', Pattern_type.INCLUDE, device=1), Pattern('/', device=1)),
  293. ),
  294. ),
  295. )
  296. def test_deduplicate_patterns_omits_child_paths_on_the_same_filesystem(patterns, expected_patterns):
  297. assert module.deduplicate_patterns(patterns) == expected_patterns
  298. def test_process_patterns_includes_patterns():
  299. flexmock(module).should_receive('deduplicate_patterns').and_return(
  300. (Pattern('foo'), Pattern('bar'))
  301. )
  302. flexmock(module).should_receive('device_map_patterns').and_return({})
  303. flexmock(module).should_receive('expand_patterns').with_args(
  304. (Pattern('foo'), Pattern('bar')),
  305. working_directory='/working',
  306. skip_paths=set(),
  307. ).and_return(()).once()
  308. assert module.process_patterns(
  309. (Pattern('foo'), Pattern('bar')),
  310. working_directory='/working',
  311. ) == [Pattern('foo'), Pattern('bar')]
  312. def test_process_patterns_skips_expand_for_requested_paths():
  313. skip_paths = {flexmock()}
  314. flexmock(module).should_receive('deduplicate_patterns').and_return(
  315. (Pattern('foo'), Pattern('bar'))
  316. )
  317. flexmock(module).should_receive('device_map_patterns').and_return({})
  318. flexmock(module).should_receive('expand_patterns').with_args(
  319. (Pattern('foo'), Pattern('bar')),
  320. working_directory='/working',
  321. skip_paths=skip_paths,
  322. ).and_return(()).once()
  323. assert module.process_patterns(
  324. (Pattern('foo'), Pattern('bar')),
  325. working_directory='/working',
  326. skip_expand_paths=skip_paths,
  327. ) == [Pattern('foo'), Pattern('bar')]
  328. def test_run_create_executes_and_calls_hooks_for_configured_repository():
  329. flexmock(module.logger).answer = lambda message: None
  330. flexmock(module.borgmatic.config.validate).should_receive('repositories_match').never()
  331. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  332. flexmock()
  333. )
  334. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once()
  335. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  336. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  337. 'call_hooks_even_if_unconfigured'
  338. ).and_return({})
  339. flexmock(module).should_receive('collect_patterns').and_return(())
  340. flexmock(module).should_receive('process_patterns').and_return([])
  341. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  342. create_arguments = flexmock(
  343. repository=None,
  344. progress=flexmock(),
  345. statistics=flexmock(),
  346. json=False,
  347. list_details=flexmock(),
  348. )
  349. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  350. list(
  351. module.run_create(
  352. config_filename='test.yaml',
  353. repository={'path': 'repo'},
  354. config={},
  355. config_paths=['/tmp/test.yaml'],
  356. local_borg_version=None,
  357. create_arguments=create_arguments,
  358. global_arguments=global_arguments,
  359. dry_run_label='',
  360. local_path=None,
  361. remote_path=None,
  362. )
  363. )
  364. def test_run_create_runs_with_selected_repository():
  365. flexmock(module.logger).answer = lambda message: None
  366. flexmock(module.borgmatic.config.validate).should_receive(
  367. 'repositories_match'
  368. ).once().and_return(True)
  369. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  370. flexmock()
  371. )
  372. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once()
  373. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  374. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  375. 'call_hooks_even_if_unconfigured'
  376. ).and_return({})
  377. flexmock(module).should_receive('collect_patterns').and_return(())
  378. flexmock(module).should_receive('process_patterns').and_return([])
  379. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  380. create_arguments = flexmock(
  381. repository=flexmock(),
  382. progress=flexmock(),
  383. statistics=flexmock(),
  384. json=False,
  385. list_details=flexmock(),
  386. )
  387. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  388. list(
  389. module.run_create(
  390. config_filename='test.yaml',
  391. repository={'path': 'repo'},
  392. config={},
  393. config_paths=['/tmp/test.yaml'],
  394. local_borg_version=None,
  395. create_arguments=create_arguments,
  396. global_arguments=global_arguments,
  397. dry_run_label='',
  398. local_path=None,
  399. remote_path=None,
  400. )
  401. )
  402. def test_run_create_bails_if_repository_does_not_match():
  403. flexmock(module.logger).answer = lambda message: None
  404. flexmock(module.borgmatic.config.validate).should_receive(
  405. 'repositories_match'
  406. ).once().and_return(False)
  407. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').never()
  408. flexmock(module.borgmatic.borg.create).should_receive('create_archive').never()
  409. create_arguments = flexmock(
  410. repository=flexmock(),
  411. progress=flexmock(),
  412. statistics=flexmock(),
  413. json=False,
  414. list_details=flexmock(),
  415. )
  416. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  417. list(
  418. module.run_create(
  419. config_filename='test.yaml',
  420. repository='repo',
  421. config={},
  422. config_paths=['/tmp/test.yaml'],
  423. local_borg_version=None,
  424. create_arguments=create_arguments,
  425. global_arguments=global_arguments,
  426. dry_run_label='',
  427. local_path=None,
  428. remote_path=None,
  429. )
  430. )
  431. def test_run_create_with_both_list_and_json_errors():
  432. flexmock(module.logger).answer = lambda message: None
  433. flexmock(module.borgmatic.config.validate).should_receive(
  434. 'repositories_match'
  435. ).once().and_return(True)
  436. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').never()
  437. flexmock(module.borgmatic.borg.create).should_receive('create_archive').never()
  438. create_arguments = flexmock(
  439. repository=flexmock(),
  440. progress=flexmock(),
  441. statistics=flexmock(),
  442. json=True,
  443. list_details=flexmock(),
  444. )
  445. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  446. with pytest.raises(ValueError):
  447. list(
  448. module.run_create(
  449. config_filename='test.yaml',
  450. repository={'path': 'repo'},
  451. config={'list_details': True},
  452. config_paths=['/tmp/test.yaml'],
  453. local_borg_version=None,
  454. create_arguments=create_arguments,
  455. global_arguments=global_arguments,
  456. dry_run_label='',
  457. local_path=None,
  458. remote_path=None,
  459. )
  460. )
  461. def test_run_create_with_both_list_and_progress_errors():
  462. flexmock(module.logger).answer = lambda message: None
  463. flexmock(module.borgmatic.config.validate).should_receive(
  464. 'repositories_match'
  465. ).once().and_return(True)
  466. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').never()
  467. flexmock(module.borgmatic.borg.create).should_receive('create_archive').never()
  468. create_arguments = flexmock(
  469. repository=flexmock(),
  470. progress=flexmock(),
  471. statistics=flexmock(),
  472. json=False,
  473. list_details=flexmock(),
  474. )
  475. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  476. with pytest.raises(ValueError):
  477. list(
  478. module.run_create(
  479. config_filename='test.yaml',
  480. repository={'path': 'repo'},
  481. config={'list_details': True, 'progress': True},
  482. config_paths=['/tmp/test.yaml'],
  483. local_borg_version=None,
  484. create_arguments=create_arguments,
  485. global_arguments=global_arguments,
  486. dry_run_label='',
  487. local_path=None,
  488. remote_path=None,
  489. )
  490. )
  491. def test_run_create_produces_json():
  492. flexmock(module.logger).answer = lambda message: None
  493. flexmock(module.borgmatic.config.validate).should_receive(
  494. 'repositories_match'
  495. ).once().and_return(True)
  496. flexmock(module.borgmatic.config.paths).should_receive('Runtime_directory').and_return(
  497. flexmock()
  498. )
  499. flexmock(module.borgmatic.borg.create).should_receive('create_archive').once().and_return(
  500. flexmock()
  501. )
  502. parsed_json = flexmock()
  503. flexmock(module.borgmatic.actions.json).should_receive('parse_json').and_return(parsed_json)
  504. flexmock(module.borgmatic.hooks.dispatch).should_receive('call_hooks').and_return({})
  505. flexmock(module.borgmatic.hooks.dispatch).should_receive(
  506. 'call_hooks_even_if_unconfigured'
  507. ).and_return({})
  508. flexmock(module).should_receive('collect_patterns').and_return(())
  509. flexmock(module).should_receive('process_patterns').and_return([])
  510. flexmock(module.os.path).should_receive('join').and_return('/run/borgmatic/bootstrap')
  511. create_arguments = flexmock(
  512. repository=flexmock(),
  513. progress=flexmock(),
  514. statistics=flexmock(),
  515. json=True,
  516. list_details=flexmock(),
  517. )
  518. global_arguments = flexmock(monitoring_verbosity=1, dry_run=False)
  519. assert list(
  520. module.run_create(
  521. config_filename='test.yaml',
  522. repository={'path': 'repo'},
  523. config={},
  524. config_paths=['/tmp/test.yaml'],
  525. local_borg_version=None,
  526. create_arguments=create_arguments,
  527. global_arguments=global_arguments,
  528. dry_run_label='',
  529. local_path=None,
  530. remote_path=None,
  531. )
  532. ) == [parsed_json]