| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493 | import loggingimport pytestfrom flexmock import flexmockfrom borgmatic.hooks.data_source import mariadb as moduledef test_parse_extra_options_passes_through_empty_options():    assert module.parse_extra_options('') == ((), None)def test_parse_extra_options_with_defaults_extra_file_removes_and_and_parses_out_filename():    assert module.parse_extra_options('--defaults-extra-file=extra.cnf --skip-ssl') == (        ('--skip-ssl',),        'extra.cnf',    )def test_parse_extra_options_without_defaults_extra_file_passes_through_options():    assert module.parse_extra_options('--skip-ssl --and=stuff') == (        ('--skip-ssl', '--and=stuff'),        None,    )def test_make_defaults_file_pipe_without_username_or_password_bails():    flexmock(module.os).should_receive('pipe').never()    assert module.make_defaults_file_options(username=None, password=None) == ()def test_make_defaults_file_option_with_username_and_password_writes_them_to_file_descriptor():    read_descriptor = 99    write_descriptor = flexmock()    flexmock(module.os).should_receive('pipe').and_return(read_descriptor, write_descriptor)    flexmock(module.os).should_receive('write').with_args(        write_descriptor, b'[client]\nuser=root\npassword="trustsome1"'    ).once()    flexmock(module.os).should_receive('close')    flexmock(module.os).should_receive('set_inheritable')    assert module.make_defaults_file_options(username='root', password='trustsome1') == (        '--defaults-extra-file=/dev/fd/99',    )def test_make_defaults_file_escapes_password_containing_backslash():    read_descriptor = 99    write_descriptor = flexmock()    flexmock(module.os).should_receive('pipe').and_return(read_descriptor, write_descriptor)    flexmock(module.os).should_receive('write').with_args(        write_descriptor, b'[client]\nuser=root\n' + br'password="trust\\nsome1"'    ).once()    flexmock(module.os).should_receive('close')    flexmock(module.os).should_receive('set_inheritable')    assert module.make_defaults_file_options(username='root', password=r'trust\nsome1') == (        '--defaults-extra-file=/dev/fd/99',    )def test_make_defaults_file_pipe_with_only_username_writes_it_to_file_descriptor():    read_descriptor = 99    write_descriptor = flexmock()    flexmock(module.os).should_receive('pipe').and_return(read_descriptor, write_descriptor)    flexmock(module.os).should_receive('write').with_args(        write_descriptor, b'[client]\nuser=root'    ).once()    flexmock(module.os).should_receive('close')    flexmock(module.os).should_receive('set_inheritable')    assert module.make_defaults_file_options(username='root', password=None) == (        '--defaults-extra-file=/dev/fd/99',    )def test_make_defaults_file_pipe_with_only_password_writes_it_to_file_descriptor():    read_descriptor = 99    write_descriptor = flexmock()    flexmock(module.os).should_receive('pipe').and_return(read_descriptor, write_descriptor)    flexmock(module.os).should_receive('write').with_args(        write_descriptor, b'[client]\npassword="trustsome1"'    ).once()    flexmock(module.os).should_receive('close')    flexmock(module.os).should_receive('set_inheritable')    assert module.make_defaults_file_options(username=None, password='trustsome1') == (        '--defaults-extra-file=/dev/fd/99',    )def test_make_defaults_file_option_with_defaults_extra_filename_includes_it_in_file_descriptor():    read_descriptor = 99    write_descriptor = flexmock()    flexmock(module.os).should_receive('pipe').and_return(read_descriptor, write_descriptor)    flexmock(module.os).should_receive('write').with_args(        write_descriptor, b'!include extra.cnf\n[client]\nuser=root\npassword="trustsome1"'    ).once()    flexmock(module.os).should_receive('close')    flexmock(module.os).should_receive('set_inheritable')    assert module.make_defaults_file_options(        username='root', password='trustsome1', defaults_extra_filename='extra.cnf'    ) == ('--defaults-extra-file=/dev/fd/99',)def test_make_defaults_file_option_with_only_defaults_extra_filename_uses_it_instead_of_file_descriptor():    flexmock(module.os).should_receive('pipe').never()    assert module.make_defaults_file_options(        username=None, password=None, defaults_extra_filename='extra.cnf'    ) == ('--defaults-extra-file=extra.cnf',)def test_database_names_to_dump_passes_through_name():    environment = flexmock()    names = module.database_names_to_dump(        {'name': 'foo'}, {}, 'root', 'trustsome1', environment, dry_run=False    )    assert names == ('foo',)def test_database_names_to_dump_bails_for_dry_run():    environment = flexmock()    flexmock(module).should_receive('execute_command_and_capture_output').never()    names = module.database_names_to_dump(        {'name': 'all'}, {}, 'root', 'trustsome1', environment, dry_run=True    )    assert names == ()def test_database_names_to_dump_queries_mariadb_for_database_names():    environment = flexmock()    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module).should_receive('execute_command_and_capture_output').with_args(        (            'mariadb',            '--defaults-extra-file=/dev/fd/99',            '--skip-column-names',            '--batch',            '--execute',            'show schemas',        ),        environment=environment,    ).and_return('foo\nbar\nmysql\n').once()    names = module.database_names_to_dump(        {'name': 'all'}, {}, 'root', 'trustsome1', environment, dry_run=False    )    assert names == ('foo', 'bar')def test_database_names_to_dump_with_environment_password_transport_skips_defaults_file_and_passes_user_flag():    environment = flexmock()    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').never()    flexmock(module).should_receive('execute_command_and_capture_output').with_args(        (            'mariadb',            '--user',            'root',            '--skip-column-names',            '--batch',            '--execute',            'show schemas',        ),        environment=environment,    ).and_return('foo\nbar\nmysql\n').once()    names = module.database_names_to_dump(        {'name': 'all', 'password_transport': 'environment'},        {},        'root',        'trustsome1',        environment,        dry_run=False,    )    assert names == ('foo', 'bar')def test_database_names_to_dump_runs_mariadb_with_tls():    environment = flexmock()    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module).should_receive('execute_command_and_capture_output').with_args(        (            'mariadb',            '--defaults-extra-file=/dev/fd/99',            '--ssl',            '--skip-column-names',            '--batch',            '--execute',            'show schemas',        ),        environment=environment,    ).and_return('foo\nbar\nmysql\n').once()    names = module.database_names_to_dump(        {'name': 'all', 'tls': True}, {}, 'root', 'trustsome1', environment, dry_run=False    )    assert names == ('foo', 'bar')def test_database_names_to_dump_runs_mariadb_without_tls():    environment = flexmock()    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module).should_receive('execute_command_and_capture_output').with_args(        (            'mariadb',            '--defaults-extra-file=/dev/fd/99',            '--skip-ssl',            '--skip-column-names',            '--batch',            '--execute',            'show schemas',        ),        environment=environment,    ).and_return('foo\nbar\nmysql\n').once()    names = module.database_names_to_dump(        {'name': 'all', 'tls': False}, {}, 'root', 'trustsome1', environment, dry_run=False    )    assert names == ('foo', 'bar')def test_use_streaming_true_for_any_databases():    assert module.use_streaming(        databases=[flexmock(), flexmock()],        config=flexmock(),    )def test_use_streaming_false_for_no_databases():    assert not module.use_streaming(databases=[], config=flexmock())def test_dump_data_sources_dumps_each_database():    databases = [{'name': 'foo'}, {'name': 'bar'}]    processes = [flexmock(), flexmock()]    flexmock(module).should_receive('make_dump_path').and_return('')    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).and_return(None)    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('database_names_to_dump').with_args(        database=databases[0],        config={},        username=None,        password=None,        environment={'USER': 'root'},        dry_run=False,    ).and_return(('foo',))    flexmock(module).should_receive('database_names_to_dump').with_args(        database=databases[1],        config={},        username=None,        password=None,        environment={'USER': 'root'},        dry_run=False,    ).and_return(('bar',))    for name, process in zip(('foo', 'bar'), processes):        flexmock(module).should_receive('execute_dump_command').with_args(            database={'name': name},            config={},            username=None,            password=None,            dump_path=object,            database_names=(name,),            environment={'USER': 'root'},            dry_run=object,            dry_run_label=object,        ).and_return(process).once()    assert (        module.dump_data_sources(            databases,            {},            config_paths=('test.yaml',),            borgmatic_runtime_directory='/run/borgmatic',            patterns=[],            dry_run=False,        )        == processes    )def test_dump_data_sources_dumps_with_password():    database = {'name': 'foo', 'username': 'root', 'password': 'trustsome1'}    process = flexmock()    flexmock(module).should_receive('make_dump_path').and_return('')    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('database_names_to_dump').with_args(        database=database,        config={},        username='root',        password='trustsome1',        environment={'USER': 'root'},        dry_run=False,    ).and_return(('foo',))    flexmock(module).should_receive('execute_dump_command').with_args(        database=database,        config={},        username='root',        password='trustsome1',        dump_path=object,        database_names=('foo',),        environment={'USER': 'root'},        dry_run=object,        dry_run_label=object,    ).and_return(process).once()    assert module.dump_data_sources(        [database],        {},        config_paths=('test.yaml',),        borgmatic_runtime_directory='/run/borgmatic',        patterns=[],        dry_run=False,    ) == [process]def test_dump_data_sources_dumps_with_environment_password_transport_passes_password_environment_variable():    database = {        'name': 'foo',        'username': 'root',        'password': 'trustsome1',        'password_transport': 'environment',    }    process = flexmock()    flexmock(module).should_receive('make_dump_path').and_return('')    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('database_names_to_dump').with_args(        database=database,        config={},        username='root',        password='trustsome1',        environment={'USER': 'root', 'MYSQL_PWD': 'trustsome1'},        dry_run=False,    ).and_return(('foo',))    flexmock(module).should_receive('execute_dump_command').with_args(        database=database,        config={},        username='root',        password='trustsome1',        dump_path=object,        database_names=('foo',),        environment={'USER': 'root', 'MYSQL_PWD': 'trustsome1'},        dry_run=object,        dry_run_label=object,    ).and_return(process).once()    assert module.dump_data_sources(        [database],        {},        config_paths=('test.yaml',),        borgmatic_runtime_directory='/run/borgmatic',        patterns=[],        dry_run=False,    ) == [process]def test_dump_data_sources_dumps_all_databases_at_once():    databases = [{'name': 'all'}]    process = flexmock()    flexmock(module).should_receive('make_dump_path').and_return('')    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('database_names_to_dump').and_return(('foo', 'bar'))    flexmock(module).should_receive('execute_dump_command').with_args(        database={'name': 'all'},        config={},        username=None,        password=None,        dump_path=object,        database_names=('foo', 'bar'),        environment={'USER': 'root'},        dry_run=object,        dry_run_label=object,    ).and_return(process).once()    assert module.dump_data_sources(        databases,        {},        config_paths=('test.yaml',),        borgmatic_runtime_directory='/run/borgmatic',        patterns=[],        dry_run=False,    ) == [process]def test_dump_data_sources_dumps_all_databases_separately_when_format_configured():    databases = [{'name': 'all', 'format': 'sql'}]    processes = [flexmock(), flexmock()]    flexmock(module).should_receive('make_dump_path').and_return('')    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).and_return(None)    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('database_names_to_dump').and_return(('foo', 'bar'))    for name, process in zip(('foo', 'bar'), processes):        flexmock(module).should_receive('execute_dump_command').with_args(            database={'name': name, 'format': 'sql'},            config={},            username=None,            password=None,            dump_path=object,            database_names=(name,),            environment={'USER': 'root'},            dry_run=object,            dry_run_label=object,        ).and_return(process).once()    assert (        module.dump_data_sources(            databases,            {},            config_paths=('test.yaml',),            borgmatic_runtime_directory='/run/borgmatic',            patterns=[],            dry_run=False,        )        == processes    )def test_dump_data_sources_errors_for_missing_all_databases():    databases = [{'name': 'all'}]    flexmock(module).should_receive('make_dump_path').and_return('')    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(        'databases/localhost/all'    )    flexmock(module).should_receive('database_names_to_dump').and_return(())    with pytest.raises(ValueError):        assert module.dump_data_sources(            databases,            {},            config_paths=('test.yaml',),            borgmatic_runtime_directory='/run/borgmatic',            patterns=[],            dry_run=False,        )def test_dump_data_sources_does_not_error_for_missing_all_databases_with_dry_run():    databases = [{'name': 'all'}]    flexmock(module).should_receive('make_dump_path').and_return('')    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return(        'databases/localhost/all'    )    flexmock(module).should_receive('database_names_to_dump').and_return(())    assert (        module.dump_data_sources(            databases,            {},            config_paths=('test.yaml',),            borgmatic_runtime_directory='/run/borgmatic',            patterns=[],            dry_run=True,        )        == []    )def test_database_names_to_dump_runs_mariadb_with_list_options():    database = {'name': 'all', 'list_options': '--defaults-extra-file=mariadb.cnf --skip-ssl'}    flexmock(module).should_receive('parse_extra_options').and_return(        ('--skip-ssl',), 'mariadb.cnf'    )    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', 'mariadb.cnf'    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module).should_receive('execute_command_and_capture_output').with_args(        (            'mariadb',            '--defaults-extra-file=/dev/fd/99',            '--skip-ssl',            '--skip-column-names',            '--batch',            '--execute',            'show schemas',        ),        environment=None,    ).and_return(('foo\nbar')).once()    assert module.database_names_to_dump(database, {}, 'root', 'trustsome1', None, '') == (        'foo',        'bar',    )def test_database_names_to_dump_runs_non_default_mariadb_with_list_options():    database = {        'name': 'all',        'list_options': '--defaults-extra-file=mariadb.cnf --skip-ssl',        'mariadb_command': 'custom_mariadb',    }    flexmock(module).should_receive('parse_extra_options').and_return(        ('--skip-ssl',), 'mariadb.cnf'    )    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', 'mariadb.cnf'    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module).should_receive('execute_command_and_capture_output').with_args(        environment=None,        full_command=(            'custom_mariadb',  # Custom MariaDB command            '--defaults-extra-file=/dev/fd/99',            '--skip-ssl',            '--skip-column-names',            '--batch',            '--execute',            'show schemas',        ),    ).and_return(('foo\nbar')).once()    assert module.database_names_to_dump(database, {}, 'root', 'trustsome1', None, '') == (        'foo',        'bar',    )def test_execute_dump_command_runs_mariadb_dump():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--defaults-extra-file=/dev/fd/99',            '--add-drop-database',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo'},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_with_environment_password_transport_skips_defaults_file_and_passes_user_flag():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').never()    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--add-drop-database',            '--user',            'root',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo', 'password_transport': 'environment'},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_runs_mariadb_dump_without_add_drop_database():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--defaults-extra-file=/dev/fd/99',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo', 'add_drop_database': False},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_runs_mariadb_dump_with_hostname_and_port():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--defaults-extra-file=/dev/fd/99',            '--add-drop-database',            '--host',            'database.example.org',            '--port',            '5433',            '--protocol',            'tcp',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo', 'hostname': 'database.example.org', 'port': 5433},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_runs_mariadb_dump_with_tls():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--defaults-extra-file=/dev/fd/99',            '--add-drop-database',            '--ssl',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo', 'tls': True},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_runs_mariadb_dump_without_tls():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--defaults-extra-file=/dev/fd/99',            '--add-drop-database',            '--skip-ssl',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo', 'tls': False},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_runs_mariadb_dump_with_username_and_password():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--defaults-extra-file=/dev/fd/99',            '--add-drop-database',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment={},        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo', 'username': 'root', 'password': 'trustsome1'},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment={},            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_runs_mariadb_dump_with_options():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return(('--stuff=such',), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'mariadb-dump',            '--defaults-extra-file=/dev/fd/99',            '--stuff=such',            '--add-drop-database',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={'name': 'foo', 'options': '--stuff=such'},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_runs_non_default_mariadb_dump_with_options():    process = flexmock()    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return(('--stuff=such',), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').with_args(        (            'custom_mariadb_dump',  # Custom MariaDB dump command            '--defaults-extra-file=/dev/fd/99',            '--stuff=such',            '--add-drop-database',            '--databases',            'foo',            '--result-file',            'dump',        ),        environment=None,        run_to_completion=False,    ).and_return(process).once()    assert (        module.execute_dump_command(            database={                'name': 'foo',                'mariadb_dump_command': 'custom_mariadb_dump',                'options': '--stuff=such',            },  # Custom MariaDB dump command specified            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=False,            dry_run_label='',        )        == process    )def test_execute_dump_command_with_duplicate_dump_skips_mariadb_dump():    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(True)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump').never()    flexmock(module).should_receive('execute_command').never()    assert (        module.execute_dump_command(            database={'name': 'foo'},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=True,            dry_run_label='SO DRY',        )        is None    )def test_execute_dump_command_with_dry_run_skips_mariadb_dump():    flexmock(module.dump).should_receive('make_data_source_dump_filename').and_return('dump')    flexmock(module.os.path).should_receive('exists').and_return(False)    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.dump).should_receive('create_named_pipe_for_dump')    flexmock(module).should_receive('execute_command').never()    assert (        module.execute_dump_command(            database={'name': 'foo'},            config={},            username='root',            password='trustsome1',            dump_path=flexmock(),            database_names=('foo',),            environment=None,            dry_run=True,            dry_run_label='SO DRY',        )        is None    )def test_restore_data_source_dump_runs_mariadb_to_restore():    hook_config = [{'name': 'foo'}, {'name': 'bar'}]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        None, None, None    ).and_return(())    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        ('mariadb', '--batch'),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source={'name': 'foo'},        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_runs_mariadb_with_options():    hook_config = [{'name': 'foo', 'restore_options': '--harder'}]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return(('--harder',), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        None, None, None    ).and_return(())    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        ('mariadb', '--harder', '--batch'),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_runs_non_default_mariadb_with_options():    hook_config = [        {'name': 'foo', 'restore_options': '--harder', 'mariadb_command': 'custom_mariadb'}    ]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return(('--harder',), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        None, None, None    ).and_return(())    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        ('custom_mariadb', '--harder', '--batch'),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_runs_mariadb_with_hostname_and_port():    hook_config = [{'name': 'foo', 'hostname': 'database.example.org', 'port': 5433}]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        None, None, None    ).and_return(())    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        (            'mariadb',            '--batch',            '--host',            'database.example.org',            '--port',            '5433',            '--protocol',            'tcp',        ),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_runs_mariadb_with_tls():    hook_config = [{'name': 'foo', 'tls': True}]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        None, None, None    ).and_return(())    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        (            'mariadb',            '--batch',            '--ssl',        ),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_runs_mariadb_without_tls():    hook_config = [{'name': 'foo', 'tls': False}]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        None, None, None    ).and_return(())    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        (            'mariadb',            '--batch',            '--skip-ssl',        ),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_runs_mariadb_with_username_and_password():    hook_config = [{'name': 'foo', 'username': 'root', 'password': 'trustsome1'}]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'root', 'trustsome1', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        ('mariadb', '--defaults-extra-file=/dev/fd/99', '--batch'),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_with_environment_password_transport_skips_defaults_file_and_passes_user_flag():    hook_config = [        {            'name': 'foo',            'username': 'root',            'password': 'trustsome1',            'password_transport': 'environment',        }    ]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').never()    flexmock(module.os).should_receive('environ').and_return(        {'USER': 'root', 'MYSQL_PWD': 'trustsome1'}    )    flexmock(module).should_receive('execute_command_with_processes').with_args(        ('mariadb', '--batch', '--user', 'root'),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root', 'MYSQL_PWD': 'trustsome1'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_with_connection_params_uses_connection_params_for_restore():    hook_config = [        {            'name': 'foo',            'username': 'root',            'password': 'trustsome1',            'restore_hostname': 'restorehost',            'restore_port': 'restoreport',            'restore_username': 'restoreusername',            'restore_password': 'restorepassword',        }    ]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'cliusername', 'clipassword', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        (            'mariadb',            '--defaults-extra-file=/dev/fd/99',            '--batch',            '--host',            'clihost',            '--port',            'cliport',            '--protocol',            'tcp',        ),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': 'clihost',            'port': 'cliport',            'username': 'cliusername',            'password': 'clipassword',        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_without_connection_params_uses_restore_params_in_config_for_restore():    hook_config = [        {            'name': 'foo',            'username': 'root',            'password': 'trustsome1',            'hostname': 'dbhost',            'port': 'dbport',            'tls': True,            'restore_username': 'restoreuser',            'restore_password': 'restorepass',            'restore_hostname': 'restorehost',            'restore_port': 'restoreport',            'restore_tls': False,        }    ]    extract_process = flexmock(stdout=flexmock())    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        'restoreuser', 'restorepass', None    ).and_return(('--defaults-extra-file=/dev/fd/99',))    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').with_args(        (            'mariadb',            '--defaults-extra-file=/dev/fd/99',            '--batch',            '--host',            'restorehost',            '--port',            'restoreport',            '--protocol',            'tcp',            '--skip-ssl',        ),        processes=[extract_process],        output_log_level=logging.DEBUG,        input_file=extract_process.stdout,        environment={'USER': 'root'},    ).once()    module.restore_data_source_dump(        hook_config,        {},        data_source=hook_config[0],        dry_run=False,        extract_process=extract_process,        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )def test_restore_data_source_dump_with_dry_run_skips_restore():    hook_config = [{'name': 'foo'}]    flexmock(module.borgmatic.hooks.credential.parse).should_receive(        'resolve_credential'    ).replace_with(lambda value, config: value)    flexmock(module).should_receive('parse_extra_options').and_return((), None)    flexmock(module).should_receive('make_defaults_file_options').with_args(        None, None, None    ).and_return(())    flexmock(module.os).should_receive('environ').and_return({'USER': 'root'})    flexmock(module).should_receive('execute_command_with_processes').never()    module.restore_data_source_dump(        hook_config,        {},        data_source={'name': 'foo'},        dry_run=True,        extract_process=flexmock(),        connection_params={            'hostname': None,            'port': None,            'username': None,            'password': None,        },        borgmatic_runtime_directory='/run/borgmatic',    )
 |