test_create.py 22 KB

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