123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374 |
- import io
- import sys
- import pytest
- import ruamel.yaml
- from flexmock import flexmock
- from borgmatic.config import load as module
- def test_load_configuration_parses_contents():
- builtins = flexmock(sys.modules['builtins'])
- builtins.should_receive('open').with_args('config.yaml').and_return('key: value')
- assert module.load_configuration('config.yaml') == {'key': 'value'}
- def test_load_configuration_inlines_include_relative_to_current_directory():
- builtins = flexmock(sys.modules['builtins'])
- flexmock(module.os).should_receive('getcwd').and_return('/tmp')
- flexmock(module.os.path).should_receive('isabs').and_return(False)
- flexmock(module.os.path).should_receive('exists').and_return(True)
- include_file = io.StringIO('value')
- include_file.name = 'include.yaml'
- builtins.should_receive('open').with_args('/tmp/include.yaml').and_return(include_file)
- config_file = io.StringIO('key: !include include.yaml')
- config_file.name = 'config.yaml'
- builtins.should_receive('open').with_args('config.yaml').and_return(config_file)
- assert module.load_configuration('config.yaml') == {'key': 'value'}
- def test_load_configuration_inlines_include_relative_to_config_parent_directory():
- builtins = flexmock(sys.modules['builtins'])
- flexmock(module.os).should_receive('getcwd').and_return('/tmp')
- flexmock(module.os.path).should_receive('isabs').with_args('/etc').and_return(True)
- flexmock(module.os.path).should_receive('isabs').with_args('/etc/config.yaml').and_return(True)
- flexmock(module.os.path).should_receive('isabs').with_args('include.yaml').and_return(False)
- flexmock(module.os.path).should_receive('exists').with_args('/tmp/include.yaml').and_return(
- False
- )
- flexmock(module.os.path).should_receive('exists').with_args('/etc/include.yaml').and_return(
- True
- )
- include_file = io.StringIO('value')
- include_file.name = 'include.yaml'
- builtins.should_receive('open').with_args('/etc/include.yaml').and_return(include_file)
- config_file = io.StringIO('key: !include include.yaml')
- config_file.name = '/etc/config.yaml'
- builtins.should_receive('open').with_args('/etc/config.yaml').and_return(config_file)
- assert module.load_configuration('/etc/config.yaml') == {'key': 'value'}
- def test_load_configuration_raises_if_relative_include_does_not_exist():
- builtins = flexmock(sys.modules['builtins'])
- flexmock(module.os).should_receive('getcwd').and_return('/tmp')
- flexmock(module.os.path).should_receive('isabs').with_args('/etc').and_return(True)
- flexmock(module.os.path).should_receive('isabs').with_args('/etc/config.yaml').and_return(True)
- flexmock(module.os.path).should_receive('isabs').with_args('include.yaml').and_return(False)
- flexmock(module.os.path).should_receive('exists').and_return(False)
- config_file = io.StringIO('key: !include include.yaml')
- config_file.name = '/etc/config.yaml'
- builtins.should_receive('open').with_args('/etc/config.yaml').and_return(config_file)
- with pytest.raises(FileNotFoundError):
- module.load_configuration('/etc/config.yaml')
- def test_load_configuration_inlines_absolute_include():
- builtins = flexmock(sys.modules['builtins'])
- flexmock(module.os).should_receive('getcwd').and_return('/tmp')
- flexmock(module.os.path).should_receive('isabs').and_return(True)
- flexmock(module.os.path).should_receive('exists').never()
- include_file = io.StringIO('value')
- include_file.name = '/root/include.yaml'
- builtins.should_receive('open').with_args('/root/include.yaml').and_return(include_file)
- config_file = io.StringIO('key: !include /root/include.yaml')
- config_file.name = 'config.yaml'
- builtins.should_receive('open').with_args('config.yaml').and_return(config_file)
- assert module.load_configuration('config.yaml') == {'key': 'value'}
- def test_load_configuration_raises_if_absolute_include_does_not_exist():
- builtins = flexmock(sys.modules['builtins'])
- flexmock(module.os).should_receive('getcwd').and_return('/tmp')
- flexmock(module.os.path).should_receive('isabs').and_return(True)
- builtins.should_receive('open').with_args('/root/include.yaml').and_raise(FileNotFoundError)
- config_file = io.StringIO('key: !include /root/include.yaml')
- config_file.name = 'config.yaml'
- builtins.should_receive('open').with_args('config.yaml').and_return(config_file)
- with pytest.raises(FileNotFoundError):
- assert module.load_configuration('config.yaml')
- def test_load_configuration_merges_include():
- builtins = flexmock(sys.modules['builtins'])
- flexmock(module.os).should_receive('getcwd').and_return('/tmp')
- flexmock(module.os.path).should_receive('isabs').and_return(False)
- flexmock(module.os.path).should_receive('exists').and_return(True)
- include_file = io.StringIO(
- '''
- foo: bar
- baz: quux
- '''
- )
- include_file.name = 'include.yaml'
- builtins.should_receive('open').with_args('/tmp/include.yaml').and_return(include_file)
- config_file = io.StringIO(
- '''
- foo: override
- <<: !include include.yaml
- '''
- )
- config_file.name = 'config.yaml'
- builtins.should_receive('open').with_args('config.yaml').and_return(config_file)
- assert module.load_configuration('config.yaml') == {'foo': 'override', 'baz': 'quux'}
- def test_load_configuration_does_not_merge_include_list():
- builtins = flexmock(sys.modules['builtins'])
- flexmock(module.os).should_receive('getcwd').and_return('/tmp')
- flexmock(module.os.path).should_receive('isabs').and_return(False)
- flexmock(module.os.path).should_receive('exists').and_return(True)
- include_file = io.StringIO(
- '''
- - one
- - two
- '''
- )
- include_file.name = 'include.yaml'
- builtins.should_receive('open').with_args('/tmp/include.yaml').and_return(include_file)
- config_file = io.StringIO(
- '''
- foo: bar
- repositories:
- <<: !include include.yaml
- '''
- )
- config_file.name = 'config.yaml'
- builtins.should_receive('open').with_args('config.yaml').and_return(config_file)
- with pytest.raises(ruamel.yaml.error.YAMLError):
- assert module.load_configuration('config.yaml')
- def test_deep_merge_nodes_replaces_colliding_scalar_values():
- node_values = [
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='retention'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='keep_hourly'
- ),
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:int', value='24'),
- ),
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='keep_daily'
- ),
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:int', value='7'),
- ),
- ],
- ),
- ),
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='retention'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='keep_daily'
- ),
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:int', value='5'),
- ),
- ],
- ),
- ),
- ]
- result = module.deep_merge_nodes(node_values)
- assert len(result) == 1
- (section_key, section_value) = result[0]
- assert section_key.value == 'retention'
- options = section_value.value
- assert len(options) == 2
- assert options[0][0].value == 'keep_hourly'
- assert options[0][1].value == '24'
- assert options[1][0].value == 'keep_daily'
- assert options[1][1].value == '5'
- def test_deep_merge_nodes_keeps_non_colliding_scalar_values():
- node_values = [
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='retention'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='keep_hourly'
- ),
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:int', value='24'),
- ),
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='keep_daily'
- ),
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:int', value='7'),
- ),
- ],
- ),
- ),
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='retention'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='keep_minutely'
- ),
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:int', value='10'),
- ),
- ],
- ),
- ),
- ]
- result = module.deep_merge_nodes(node_values)
- assert len(result) == 1
- (section_key, section_value) = result[0]
- assert section_key.value == 'retention'
- options = section_value.value
- assert len(options) == 3
- assert options[0][0].value == 'keep_hourly'
- assert options[0][1].value == '24'
- assert options[1][0].value == 'keep_daily'
- assert options[1][1].value == '7'
- assert options[2][0].value == 'keep_minutely'
- assert options[2][1].value == '10'
- def test_deep_merge_nodes_keeps_deeply_nested_values():
- node_values = [
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='storage'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='lock_wait'
- ),
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:int', value='5'),
- ),
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='extra_borg_options'
- ),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='init'
- ),
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='--init-option'
- ),
- ),
- ],
- ),
- ),
- ],
- ),
- ),
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='storage'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='extra_borg_options'
- ),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='prune'
- ),
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='--prune-option'
- ),
- ),
- ],
- ),
- ),
- ],
- ),
- ),
- ]
- result = module.deep_merge_nodes(node_values)
- assert len(result) == 1
- (section_key, section_value) = result[0]
- assert section_key.value == 'storage'
- options = section_value.value
- assert len(options) == 2
- assert options[0][0].value == 'lock_wait'
- assert options[0][1].value == '5'
- assert options[1][0].value == 'extra_borg_options'
- nested_options = options[1][1].value
- assert len(nested_options) == 2
- assert nested_options[0][0].value == 'init'
- assert nested_options[0][1].value == '--init-option'
- assert nested_options[1][0].value == 'prune'
- assert nested_options[1][1].value == '--prune-option'
- def test_deep_merge_nodes_appends_colliding_sequence_values():
- node_values = [
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='hooks'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='before_backup'
- ),
- ruamel.yaml.nodes.SequenceNode(
- tag='tag:yaml.org,2002:int', value=['echo 1', 'echo 2']
- ),
- ),
- ],
- ),
- ),
- (
- ruamel.yaml.nodes.ScalarNode(tag='tag:yaml.org,2002:str', value='hooks'),
- ruamel.yaml.nodes.MappingNode(
- tag='tag:yaml.org,2002:map',
- value=[
- (
- ruamel.yaml.nodes.ScalarNode(
- tag='tag:yaml.org,2002:str', value='before_backup'
- ),
- ruamel.yaml.nodes.SequenceNode(
- tag='tag:yaml.org,2002:int', value=['echo 3', 'echo 4']
- ),
- ),
- ],
- ),
- ),
- ]
- result = module.deep_merge_nodes(node_values)
- assert len(result) == 1
- (section_key, section_value) = result[0]
- assert section_key.value == 'hooks'
- options = section_value.value
- assert len(options) == 1
- assert options[0][0].value == 'before_backup'
- assert options[0][1].value == ['echo 1', 'echo 2', 'echo 3', 'echo 4']
|