|  | @@ -81,6 +81,107 @@ hooks:
 | 
											
												
													
														|  |      with open(config_path, 'w') as config_file:
 |  |      with open(config_path, 'w') as config_file:
 | 
											
												
													
														|  |          config_file.write(config)
 |  |          config_file.write(config)
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | 
 |  | +def write_custom_restore_configuration(
 | 
											
												
													
														|  | 
 |  | +    source_directory,
 | 
											
												
													
														|  | 
 |  | +    config_path,
 | 
											
												
													
														|  | 
 |  | +    repository_path,
 | 
											
												
													
														|  | 
 |  | +    borgmatic_source_directory,
 | 
											
												
													
														|  | 
 |  | +    postgresql_dump_format='custom',
 | 
											
												
													
														|  | 
 |  | +    mongodb_dump_format='archive',
 | 
											
												
													
														|  | 
 |  | +):
 | 
											
												
													
														|  | 
 |  | +    '''
 | 
											
												
													
														|  | 
 |  | +    Write out borgmatic configuration into a file at the config path. Set the options so as to work
 | 
											
												
													
														|  | 
 |  | +    for testing with custom restore options. This includes a custom restore_hostname, restore_port,
 | 
											
												
													
														|  | 
 |  | +    restore_username, restore_password and restore_path.
 | 
											
												
													
														|  | 
 |  | +    '''
 | 
											
												
													
														|  | 
 |  | +    config = f'''
 | 
											
												
													
														|  | 
 |  | +location:
 | 
											
												
													
														|  | 
 |  | +    source_directories:
 | 
											
												
													
														|  | 
 |  | +        - {source_directory}
 | 
											
												
													
														|  | 
 |  | +    repositories:
 | 
											
												
													
														|  | 
 |  | +        - {repository_path}
 | 
											
												
													
														|  | 
 |  | +    borgmatic_source_directory: {borgmatic_source_directory}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +storage:
 | 
											
												
													
														|  | 
 |  | +    encryption_passphrase: "test"
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +hooks:
 | 
											
												
													
														|  | 
 |  | +    postgresql_databases:
 | 
											
												
													
														|  | 
 |  | +        - name: test
 | 
											
												
													
														|  | 
 |  | +          hostname: postgresql
 | 
											
												
													
														|  | 
 |  | +          username: postgres
 | 
											
												
													
														|  | 
 |  | +          password: test
 | 
											
												
													
														|  | 
 |  | +          format: {postgresql_dump_format}
 | 
											
												
													
														|  | 
 |  | +          restore_hostname: postgresql2
 | 
											
												
													
														|  | 
 |  | +          restore_port: 5432
 | 
											
												
													
														|  | 
 |  | +          restore_username: postgres2
 | 
											
												
													
														|  | 
 |  | +          restore_password: test2
 | 
											
												
													
														|  | 
 |  | +    mysql_databases:
 | 
											
												
													
														|  | 
 |  | +        - name: test
 | 
											
												
													
														|  | 
 |  | +          hostname: mysql
 | 
											
												
													
														|  | 
 |  | +          username: root
 | 
											
												
													
														|  | 
 |  | +          password: test
 | 
											
												
													
														|  | 
 |  | +          restore_hostname: mysql2
 | 
											
												
													
														|  | 
 |  | +          restore_port: 3306
 | 
											
												
													
														|  | 
 |  | +          restore_username: root
 | 
											
												
													
														|  | 
 |  | +          restore_password: test2
 | 
											
												
													
														|  | 
 |  | +    mongodb_databases:
 | 
											
												
													
														|  | 
 |  | +        - name: test
 | 
											
												
													
														|  | 
 |  | +          hostname: mongodb
 | 
											
												
													
														|  | 
 |  | +          username: root
 | 
											
												
													
														|  | 
 |  | +          password: test
 | 
											
												
													
														|  | 
 |  | +          authentication_database: admin
 | 
											
												
													
														|  | 
 |  | +          format: {mongodb_dump_format}
 | 
											
												
													
														|  | 
 |  | +          restore_hostname: mongodb2
 | 
											
												
													
														|  | 
 |  | +          restore_port: 27017
 | 
											
												
													
														|  | 
 |  | +          restore_username: root2
 | 
											
												
													
														|  | 
 |  | +          restore_password: test2
 | 
											
												
													
														|  | 
 |  | +    sqlite_databases:
 | 
											
												
													
														|  | 
 |  | +        - name: sqlite_test
 | 
											
												
													
														|  | 
 |  | +          path: /tmp/sqlite_test.db
 | 
											
												
													
														|  | 
 |  | +          restore_path: /tmp/sqlite_test2.db
 | 
											
												
													
														|  | 
 |  | +'''
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    with open(config_path, 'w') as config_file:
 | 
											
												
													
														|  | 
 |  | +        config_file.write(config)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +def write_custom_restore_configuration_for_cli_arguments(
 | 
											
												
													
														|  | 
 |  | +    source_directory,
 | 
											
												
													
														|  | 
 |  | +    config_path,
 | 
											
												
													
														|  | 
 |  | +    repository_path,
 | 
											
												
													
														|  | 
 |  | +    borgmatic_source_directory,
 | 
											
												
													
														|  | 
 |  | +    postgresql_dump_format='custom',
 | 
											
												
													
														|  | 
 |  | +):
 | 
											
												
													
														|  | 
 |  | +    '''
 | 
											
												
													
														|  | 
 |  | +    Write out borgmatic configuration into a file at the config path. Set the options so as to work
 | 
											
												
													
														|  | 
 |  | +    for testing with custom restore options, but this time using CLI arguments. This includes a
 | 
											
												
													
														|  | 
 |  | +    custom restore_hostname, restore_port, restore_username and restore_password as we only test
 | 
											
												
													
														|  | 
 |  | +    these options for PostgreSQL.
 | 
											
												
													
														|  | 
 |  | +    '''
 | 
											
												
													
														|  | 
 |  | +    config = f'''
 | 
											
												
													
														|  | 
 |  | +location:
 | 
											
												
													
														|  | 
 |  | +    source_directories:
 | 
											
												
													
														|  | 
 |  | +        - {source_directory}
 | 
											
												
													
														|  | 
 |  | +    repositories:
 | 
											
												
													
														|  | 
 |  | +        - {repository_path}
 | 
											
												
													
														|  | 
 |  | +    borgmatic_source_directory: {borgmatic_source_directory}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +storage:
 | 
											
												
													
														|  | 
 |  | +    encryption_passphrase: "test"
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +hooks:
 | 
											
												
													
														|  | 
 |  | +    postgresql_databases:
 | 
											
												
													
														|  | 
 |  | +        - name: test
 | 
											
												
													
														|  | 
 |  | +          hostname: postgresql
 | 
											
												
													
														|  | 
 |  | +          username: postgres
 | 
											
												
													
														|  | 
 |  | +          password: test
 | 
											
												
													
														|  | 
 |  | +          format: {postgresql_dump_format}
 | 
											
												
													
														|  | 
 |  | +'''
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    with open(config_path, 'w') as config_file:
 | 
											
												
													
														|  | 
 |  | +        config_file.write(config)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  def test_database_dump_and_restore():
 |  |  def test_database_dump_and_restore():
 | 
											
												
													
														|  |      # Create a Borg repository.
 |  |      # Create a Borg repository.
 | 
											
										
											
												
													
														|  | @@ -125,6 +226,92 @@ def test_database_dump_and_restore():
 | 
											
												
													
														|  |          shutil.rmtree(temporary_directory)
 |  |          shutil.rmtree(temporary_directory)
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | 
 |  | +def test_database_dump_and_restore_with_restore_cli_arguments():
 | 
											
												
													
														|  | 
 |  | +    # Create a Borg repository.
 | 
											
												
													
														|  | 
 |  | +    temporary_directory = tempfile.mkdtemp()
 | 
											
												
													
														|  | 
 |  | +    repository_path = os.path.join(temporary_directory, 'test.borg')
 | 
											
												
													
														|  | 
 |  | +    borgmatic_source_directory = os.path.join(temporary_directory, '.borgmatic')
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    # Write out a special file to ensure that it gets properly excluded and Borg doesn't hang on it.
 | 
											
												
													
														|  | 
 |  | +    os.mkfifo(os.path.join(temporary_directory, 'special_file'))
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    original_working_directory = os.getcwd()
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    try:
 | 
											
												
													
														|  | 
 |  | +        config_path = os.path.join(temporary_directory, 'test.yaml')
 | 
											
												
													
														|  | 
 |  | +        write_custom_restore_configuration_for_cli_arguments(
 | 
											
												
													
														|  | 
 |  | +            temporary_directory, config_path, repository_path, borgmatic_source_directory
 | 
											
												
													
														|  | 
 |  | +        )
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        subprocess.check_call(
 | 
											
												
													
														|  | 
 |  | +            ['borgmatic', '-v', '2', '--config', config_path, 'init', '--encryption', 'repokey']
 | 
											
												
													
														|  | 
 |  | +        )
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        # Run borgmatic to generate a backup archive including a database dump.
 | 
											
												
													
														|  | 
 |  | +        subprocess.check_call(['borgmatic', 'create', '--config', config_path, '-v', '2'])
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        # Get the created archive name.
 | 
											
												
													
														|  | 
 |  | +        output = subprocess.check_output(
 | 
											
												
													
														|  | 
 |  | +            ['borgmatic', '--config', config_path, 'list', '--json']
 | 
											
												
													
														|  | 
 |  | +        ).decode(sys.stdout.encoding)
 | 
											
												
													
														|  | 
 |  | +        parsed_output = json.loads(output)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        assert len(parsed_output) == 1
 | 
											
												
													
														|  | 
 |  | +        assert len(parsed_output[0]['archives']) == 1
 | 
											
												
													
														|  | 
 |  | +        archive_name = parsed_output[0]['archives'][0]['archive']
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        # Restore the database from the archive.
 | 
											
												
													
														|  | 
 |  | +        subprocess.check_call(
 | 
											
												
													
														|  | 
 |  | +            ['borgmatic', '-v', '2', '--config', config_path, 'restore', '--archive', archive_name, '--hostname', 'postgresql2', '--port', '5432', '--username', 'postgres2', '--password', 'test2']
 | 
											
												
													
														|  | 
 |  | +        )
 | 
											
												
													
														|  | 
 |  | +    finally:
 | 
											
												
													
														|  | 
 |  | +        os.chdir(original_working_directory)
 | 
											
												
													
														|  | 
 |  | +        shutil.rmtree(temporary_directory)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +def test_database_dump_and_restore_to_different_hostname_port_username_password():
 | 
											
												
													
														|  | 
 |  | +    # Create a Borg repository.
 | 
											
												
													
														|  | 
 |  | +    temporary_directory = tempfile.mkdtemp()
 | 
											
												
													
														|  | 
 |  | +    repository_path = os.path.join(temporary_directory, 'test.borg')
 | 
											
												
													
														|  | 
 |  | +    borgmatic_source_directory = os.path.join(temporary_directory, '.borgmatic')
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    # Write out a special file to ensure that it gets properly excluded and Borg doesn't hang on it.
 | 
											
												
													
														|  | 
 |  | +    os.mkfifo(os.path.join(temporary_directory, 'special_file'))
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    original_working_directory = os.getcwd()
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    try:
 | 
											
												
													
														|  | 
 |  | +        config_path = os.path.join(temporary_directory, 'test.yaml')
 | 
											
												
													
														|  | 
 |  | +        write_custom_restore_configuration(
 | 
											
												
													
														|  | 
 |  | +            temporary_directory, config_path, repository_path, borgmatic_source_directory
 | 
											
												
													
														|  | 
 |  | +        )
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        subprocess.check_call(
 | 
											
												
													
														|  | 
 |  | +            ['borgmatic', '-v', '2', '--config', config_path, 'init', '--encryption', 'repokey']
 | 
											
												
													
														|  | 
 |  | +        )
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        # Run borgmatic to generate a backup archive including a database dump.
 | 
											
												
													
														|  | 
 |  | +        subprocess.check_call(['borgmatic', 'create', '--config', config_path, '-v', '2'])
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        # Get the created archive name.
 | 
											
												
													
														|  | 
 |  | +        output = subprocess.check_output(
 | 
											
												
													
														|  | 
 |  | +            ['borgmatic', '--config', config_path, 'list', '--json']
 | 
											
												
													
														|  | 
 |  | +        ).decode(sys.stdout.encoding)
 | 
											
												
													
														|  | 
 |  | +        parsed_output = json.loads(output)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        assert len(parsed_output) == 1
 | 
											
												
													
														|  | 
 |  | +        assert len(parsed_output[0]['archives']) == 1
 | 
											
												
													
														|  | 
 |  | +        archive_name = parsed_output[0]['archives'][0]['archive']
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        # Restore the database from the archive.
 | 
											
												
													
														|  | 
 |  | +        subprocess.check_call(
 | 
											
												
													
														|  | 
 |  | +            ['borgmatic', '-v', '2', '--config', config_path, 'restore', '--archive', archive_name]
 | 
											
												
													
														|  | 
 |  | +        )
 | 
											
												
													
														|  | 
 |  | +    finally:
 | 
											
												
													
														|  | 
 |  | +        os.chdir(original_working_directory)
 | 
											
												
													
														|  | 
 |  | +        shutil.rmtree(temporary_directory)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |  def test_database_dump_and_restore_with_directory_format():
 |  |  def test_database_dump_and_restore_with_directory_format():
 | 
											
												
													
														|  |      # Create a Borg repository.
 |  |      # Create a Borg repository.
 | 
											
												
													
														|  |      temporary_directory = tempfile.mkdtemp()
 |  |      temporary_directory = tempfile.mkdtemp()
 |