test_create.py 20 KB

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