2019-12-11 01:41:01 +01:00
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import tempfile
|
|
|
|
|
2020-05-15 19:12:49 +02:00
|
|
|
import pytest
|
|
|
|
|
2019-12-11 01:41:01 +01:00
|
|
|
|
2020-05-18 20:31:29 +02:00
|
|
|
def write_configuration(
|
2022-10-03 21:58:13 +02:00
|
|
|
source_directory,
|
|
|
|
config_path,
|
|
|
|
repository_path,
|
|
|
|
borgmatic_source_directory,
|
|
|
|
postgresql_dump_format='custom',
|
2023-02-21 00:18:51 +01:00
|
|
|
mongodb_dump_format='archive',
|
2020-05-18 20:31:29 +02:00
|
|
|
):
|
2019-12-11 01:41:01 +01:00
|
|
|
'''
|
|
|
|
Write out borgmatic configuration into a file at the config path. Set the options so as to work
|
|
|
|
for testing. This includes injecting the given repository path, borgmatic source directory for
|
2020-05-18 20:31:29 +02:00
|
|
|
storing database dumps, dump format (for PostgreSQL), and encryption passphrase.
|
2019-12-11 01:41:01 +01:00
|
|
|
'''
|
2022-10-03 21:58:13 +02:00
|
|
|
config = f'''
|
2019-12-11 01:41:01 +01:00
|
|
|
location:
|
|
|
|
source_directories:
|
2022-10-03 21:58:13 +02:00
|
|
|
- {source_directory}
|
2019-12-11 01:41:01 +01:00
|
|
|
repositories:
|
2022-10-03 21:58:13 +02:00
|
|
|
- {repository_path}
|
|
|
|
borgmatic_source_directory: {borgmatic_source_directory}
|
2019-12-11 01:41:01 +01:00
|
|
|
|
|
|
|
storage:
|
|
|
|
encryption_passphrase: "test"
|
|
|
|
|
|
|
|
hooks:
|
|
|
|
postgresql_databases:
|
|
|
|
- name: test
|
2019-12-11 01:43:43 +01:00
|
|
|
hostname: postgresql
|
2023-04-21 19:34:50 +02:00
|
|
|
username: postgres
|
2019-12-11 01:41:01 +01:00
|
|
|
password: test
|
2022-10-03 21:58:13 +02:00
|
|
|
format: {postgresql_dump_format}
|
2020-04-25 00:32:53 +02:00
|
|
|
- name: all
|
|
|
|
hostname: postgresql
|
2023-04-21 19:34:50 +02:00
|
|
|
username: postgres
|
2020-04-25 00:32:53 +02:00
|
|
|
password: test
|
2023-01-26 08:31:07 +01:00
|
|
|
- name: all
|
|
|
|
format: custom
|
|
|
|
hostname: postgresql
|
2023-04-21 19:34:50 +02:00
|
|
|
username: postgres
|
2023-01-26 08:31:07 +01:00
|
|
|
password: test
|
2019-12-12 01:43:01 +01:00
|
|
|
mysql_databases:
|
|
|
|
- name: test
|
|
|
|
hostname: mysql
|
|
|
|
username: root
|
|
|
|
password: test
|
2020-04-22 21:17:22 +02:00
|
|
|
- name: all
|
|
|
|
hostname: mysql
|
|
|
|
username: root
|
|
|
|
password: test
|
2023-01-26 08:31:07 +01:00
|
|
|
- name: all
|
|
|
|
format: sql
|
|
|
|
hostname: mysql
|
|
|
|
username: root
|
|
|
|
password: test
|
2021-12-26 01:00:58 +01:00
|
|
|
mongodb_databases:
|
|
|
|
- name: test
|
|
|
|
hostname: mongodb
|
|
|
|
username: root
|
|
|
|
password: test
|
2022-01-05 01:26:38 +01:00
|
|
|
authentication_database: admin
|
2023-02-21 00:18:51 +01:00
|
|
|
format: {mongodb_dump_format}
|
2021-12-26 01:00:58 +01:00
|
|
|
- name: all
|
|
|
|
hostname: mongodb
|
|
|
|
username: root
|
|
|
|
password: test
|
2023-03-02 19:25:16 +01:00
|
|
|
sqlite_databases:
|
|
|
|
- name: sqlite_test
|
2023-03-03 20:59:01 +01:00
|
|
|
path: /tmp/sqlite_test.db
|
2022-10-03 21:58:13 +02:00
|
|
|
'''
|
2019-12-11 01:41:01 +01:00
|
|
|
|
2021-12-26 01:00:58 +01:00
|
|
|
with open(config_path, 'w') as config_file:
|
|
|
|
config_file.write(config)
|
2019-12-11 01:41:01 +01:00
|
|
|
|
2023-06-18 02:59:11 +02:00
|
|
|
|
2023-06-18 02:17:35 +02:00
|
|
|
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
|
2023-06-18 21:40:01 +02:00
|
|
|
restore_port: 5433
|
2023-06-18 02:17:35 +02:00
|
|
|
restore_username: postgres2
|
|
|
|
restore_password: test2
|
|
|
|
mysql_databases:
|
|
|
|
- name: test
|
|
|
|
hostname: mysql
|
|
|
|
username: root
|
|
|
|
password: test
|
|
|
|
restore_hostname: mysql2
|
2023-06-18 21:40:01 +02:00
|
|
|
restore_port: 3307
|
2023-06-18 02:17:35 +02:00
|
|
|
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
|
2023-06-18 21:40:01 +02:00
|
|
|
restore_port: 27018
|
2023-06-18 02:17:35 +02:00
|
|
|
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)
|
|
|
|
|
|
|
|
|
2023-06-18 21:40:01 +02:00
|
|
|
def write_simple_custom_restore_configuration(
|
2023-06-18 02:17:35 +02:00
|
|
|
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)
|
|
|
|
|
2019-12-11 01:41:01 +01:00
|
|
|
|
|
|
|
def test_database_dump_and_restore():
|
|
|
|
# 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')
|
|
|
|
|
2022-10-03 21:58:13 +02:00
|
|
|
# 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'))
|
|
|
|
|
2019-12-11 01:41:01 +01:00
|
|
|
original_working_directory = os.getcwd()
|
|
|
|
|
|
|
|
try:
|
|
|
|
config_path = os.path.join(temporary_directory, 'test.yaml')
|
2022-10-03 21:58:13 +02:00
|
|
|
write_configuration(
|
|
|
|
temporary_directory, config_path, repository_path, borgmatic_source_directory
|
|
|
|
)
|
2019-12-11 01:41:01 +01:00
|
|
|
|
|
|
|
subprocess.check_call(
|
2023-06-25 00:52:20 +02:00
|
|
|
['borgmatic', '-v', '2', '--config', config_path, 'rcreate', '--encryption', 'repokey']
|
2019-12-11 01:41:01 +01:00
|
|
|
)
|
|
|
|
|
2020-05-15 19:12:49 +02:00
|
|
|
# Run borgmatic to generate a backup archive including a database dump.
|
2021-12-29 22:18:50 +01:00
|
|
|
subprocess.check_call(['borgmatic', 'create', '--config', config_path, '-v', '2'])
|
2019-12-11 01:41:01 +01:00
|
|
|
|
|
|
|
# Get the created archive name.
|
|
|
|
output = subprocess.check_output(
|
2021-12-29 22:18:50 +01:00
|
|
|
['borgmatic', '--config', config_path, 'list', '--json']
|
2019-12-11 01:41:01 +01:00
|
|
|
).decode(sys.stdout.encoding)
|
2023-06-18 02:17:35 +02:00
|
|
|
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_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')
|
|
|
|
|
|
|
|
original_working_directory = os.getcwd()
|
|
|
|
|
|
|
|
try:
|
|
|
|
config_path = os.path.join(temporary_directory, 'test.yaml')
|
2023-06-18 21:40:01 +02:00
|
|
|
write_simple_custom_restore_configuration(
|
2023-06-18 02:17:35 +02:00
|
|
|
temporary_directory, config_path, repository_path, borgmatic_source_directory
|
|
|
|
)
|
|
|
|
|
|
|
|
subprocess.check_call(
|
2023-06-25 00:52:20 +02:00
|
|
|
['borgmatic', '-v', '2', '--config', config_path, 'rcreate', '--encryption', 'repokey']
|
2023-06-18 02:17:35 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
# 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(
|
2023-06-18 02:59:11 +02:00
|
|
|
[
|
|
|
|
'borgmatic',
|
|
|
|
'-v',
|
|
|
|
'2',
|
|
|
|
'--config',
|
|
|
|
config_path,
|
|
|
|
'restore',
|
|
|
|
'--archive',
|
|
|
|
archive_name,
|
|
|
|
'--hostname',
|
|
|
|
'postgresql2',
|
|
|
|
'--port',
|
2023-06-18 21:40:01 +02:00
|
|
|
'5433',
|
2023-06-18 02:59:11 +02:00
|
|
|
'--username',
|
|
|
|
'postgres2',
|
|
|
|
'--password',
|
|
|
|
'test2',
|
|
|
|
]
|
2023-06-18 02:17:35 +02:00
|
|
|
)
|
|
|
|
finally:
|
|
|
|
os.chdir(original_working_directory)
|
|
|
|
shutil.rmtree(temporary_directory)
|
|
|
|
|
|
|
|
|
2023-06-18 21:40:01 +02:00
|
|
|
def test_database_dump_and_restore_with_restore_configuration_options():
|
2023-06-18 02:17:35 +02:00
|
|
|
# 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')
|
|
|
|
|
|
|
|
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(
|
2023-06-25 00:52:20 +02:00
|
|
|
['borgmatic', '-v', '2', '--config', config_path, 'rcreate', '--encryption', 'repokey']
|
2023-06-18 02:17:35 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
# 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)
|
2019-12-11 01:41:01 +01:00
|
|
|
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(
|
2023-04-21 18:31:37 +02:00
|
|
|
['borgmatic', '-v', '2', '--config', config_path, 'restore', '--archive', archive_name]
|
2019-12-11 01:41:01 +01:00
|
|
|
)
|
|
|
|
finally:
|
|
|
|
os.chdir(original_working_directory)
|
|
|
|
shutil.rmtree(temporary_directory)
|
2020-05-15 19:12:49 +02:00
|
|
|
|
|
|
|
|
2020-05-18 20:31:29 +02:00
|
|
|
def test_database_dump_and_restore_with_directory_format():
|
|
|
|
# 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')
|
|
|
|
|
|
|
|
original_working_directory = os.getcwd()
|
|
|
|
|
|
|
|
try:
|
|
|
|
config_path = os.path.join(temporary_directory, 'test.yaml')
|
|
|
|
write_configuration(
|
2022-10-03 21:58:13 +02:00
|
|
|
temporary_directory,
|
2020-05-18 20:31:29 +02:00
|
|
|
config_path,
|
|
|
|
repository_path,
|
|
|
|
borgmatic_source_directory,
|
|
|
|
postgresql_dump_format='directory',
|
2023-02-21 00:18:51 +01:00
|
|
|
mongodb_dump_format='directory',
|
2020-05-18 20:31:29 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
subprocess.check_call(
|
2023-06-25 00:52:20 +02:00
|
|
|
['borgmatic', '-v', '2', '--config', config_path, 'rcreate', '--encryption', 'repokey']
|
2020-05-18 20:31:29 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
# Run borgmatic to generate a backup archive including a database dump.
|
2021-12-29 22:18:50 +01:00
|
|
|
subprocess.check_call(['borgmatic', 'create', '--config', config_path, '-v', '2'])
|
2020-05-18 20:31:29 +02:00
|
|
|
|
|
|
|
# Restore the database from the archive.
|
|
|
|
subprocess.check_call(
|
2021-12-29 22:18:50 +01:00
|
|
|
['borgmatic', '--config', config_path, 'restore', '--archive', 'latest']
|
2020-05-18 20:31:29 +02:00
|
|
|
)
|
|
|
|
finally:
|
|
|
|
os.chdir(original_working_directory)
|
|
|
|
shutil.rmtree(temporary_directory)
|
|
|
|
|
|
|
|
|
2020-05-15 19:12:49 +02:00
|
|
|
def test_database_dump_with_error_causes_borgmatic_to_exit():
|
|
|
|
# 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')
|
|
|
|
|
|
|
|
original_working_directory = os.getcwd()
|
|
|
|
|
|
|
|
try:
|
|
|
|
config_path = os.path.join(temporary_directory, 'test.yaml')
|
2022-10-03 21:58:13 +02:00
|
|
|
write_configuration(
|
|
|
|
temporary_directory, config_path, repository_path, borgmatic_source_directory
|
|
|
|
)
|
2020-05-15 19:12:49 +02:00
|
|
|
|
|
|
|
subprocess.check_call(
|
2023-06-25 00:52:20 +02:00
|
|
|
['borgmatic', '-v', '2', '--config', config_path, 'rcreate', '--encryption', 'repokey']
|
2020-05-15 19:12:49 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
# Run borgmatic with a config override such that the database dump fails.
|
|
|
|
with pytest.raises(subprocess.CalledProcessError):
|
|
|
|
subprocess.check_call(
|
|
|
|
[
|
|
|
|
'borgmatic',
|
|
|
|
'create',
|
|
|
|
'--config',
|
|
|
|
config_path,
|
|
|
|
'-v',
|
|
|
|
'2',
|
|
|
|
'--override',
|
2023-03-24 07:11:14 +01:00
|
|
|
"hooks.postgresql_databases=[{'name': 'nope'}]", # noqa: FS003
|
2020-05-15 19:12:49 +02:00
|
|
|
]
|
|
|
|
)
|
|
|
|
finally:
|
|
|
|
os.chdir(original_working_directory)
|
|
|
|
shutil.rmtree(temporary_directory)
|