test_create.py 22 KB

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