1
0
Fork 0
mirror of https://github.com/borgbackup/borg.git synced 2025-01-04 06:21:46 +00:00

repository3/manifest: tests reenabled, fixes

This commit is contained in:
Thomas Waldmann 2024-08-12 23:37:57 +02:00
parent 8b9c052acc
commit b637542dcf
No known key found for this signature in database
GPG key ID: 243ACFA951F78E01
3 changed files with 31 additions and 31 deletions

View file

@ -52,7 +52,7 @@
from .platform import acl_get, acl_set, set_flags, get_flags, swidth, hostname from .platform import acl_get, acl_set, set_flags, get_flags, swidth, hostname
from .remote import cache_if_remote from .remote import cache_if_remote
from .remote3 import RemoteRepository3 from .remote3 import RemoteRepository3
from .repository3 import Repository3, LIST_SCAN_LIMIT from .repository3 import Repository3, LIST_SCAN_LIMIT, NoManifestError
from .repoobj import RepoObj from .repoobj import RepoObj
has_link = hasattr(os, "link") has_link = hasattr(os, "link")
@ -1860,8 +1860,9 @@ def check(
self.repo_objs = RepoObj(self.key) self.repo_objs = RepoObj(self.key)
if verify_data: if verify_data:
self.verify_data() self.verify_data()
if not isinstance(repository, (Repository3, RemoteRepository3)) and Manifest.MANIFEST_ID not in self.chunks: try:
logger.error("Repository manifest not found!") repository.get_manifest()
except NoManifestError:
self.error_found = True self.error_found = True
self.manifest = self.rebuild_manifest() self.manifest = self.rebuild_manifest()
else: else:
@ -1905,12 +1906,16 @@ def make_key(self, repository):
# try the manifest first! # try the manifest first!
attempt += 1 attempt += 1
cdata = repository.get_manifest()
try: try:
return key_factory(repository, cdata) cdata = repository.get_manifest()
except UnsupportedPayloadError: except NoManifestError:
# we get here, if the cdata we got has a corrupted key type byte pass
pass # ignore it, just continue trying else:
try:
return key_factory(repository, cdata)
except UnsupportedPayloadError:
# we get here, if the cdata we got has a corrupted key type byte
pass # ignore it, just continue trying
for chunkid, _ in self.chunks.iteritems(): for chunkid, _ in self.chunks.iteritems():
attempt += 1 attempt += 1

View file

@ -32,6 +32,7 @@
from .helpers import get_socket_filename from .helpers import get_socket_filename
from .locking import LockTimeout, NotLocked, NotMyLock, LockFailed from .locking import LockTimeout, NotLocked, NotMyLock, LockFailed
from .logger import create_logger, borg_serve_log_queue from .logger import create_logger, borg_serve_log_queue
from .manifest import NoManifestError
from .helpers import msgpack from .helpers import msgpack
from .repository import Repository from .repository import Repository
from .repository3 import Repository3 from .repository3 import Repository3
@ -835,6 +836,8 @@ def handle_error(unpacked):
raise NotLocked(args[0]) raise NotLocked(args[0])
elif error == "NotMyLock": elif error == "NotMyLock":
raise NotMyLock(args[0]) raise NotMyLock(args[0])
elif error == "NoManifestError":
raise NoManifestError
else: else:
raise self.RPCError(unpacked) raise self.RPCError(unpacked)

View file

@ -8,6 +8,7 @@
from ...constants import * # NOQA from ...constants import * # NOQA
from ...helpers import bin_to_hex, msgpack from ...helpers import bin_to_hex, msgpack
from ...manifest import Manifest from ...manifest import Manifest
from ...remote3 import RemoteRepository3
from ...repository3 import Repository3 from ...repository3 import Repository3
from ..repository3 import fchunk from ..repository3 import fchunk
from . import cmd, src_file, create_src_archive, open_archive, generate_archiver_tests, RK_ENCRYPTION from . import cmd, src_file, create_src_archive, open_archive, generate_archiver_tests, RK_ENCRYPTION
@ -192,11 +193,12 @@ def test_missing_manifest(archivers, request):
archiver = request.getfixturevalue(archivers) archiver = request.getfixturevalue(archivers)
check_cmd_setup(archiver) check_cmd_setup(archiver)
archive, repository = open_archive(archiver.repository_path, "archive1") archive, repository = open_archive(archiver.repository_path, "archive1")
if isinstance(repository, Repository3):
pytest.skip("Test not adapted to Repository3")
with repository: with repository:
repository.delete(Manifest.MANIFEST_ID) if isinstance(repository, (Repository3, RemoteRepository3)):
repository.commit(compact=False) repository.store_delete("config/manifest")
else:
repository.delete(Manifest.MANIFEST_ID)
repository.commit(compact=False)
cmd(archiver, "check", exit_code=1) cmd(archiver, "check", exit_code=1)
output = cmd(archiver, "check", "-v", "--repair", exit_code=0) output = cmd(archiver, "check", "-v", "--repair", exit_code=0)
assert "archive1" in output assert "archive1" in output
@ -208,12 +210,10 @@ def test_corrupted_manifest(archivers, request):
archiver = request.getfixturevalue(archivers) archiver = request.getfixturevalue(archivers)
check_cmd_setup(archiver) check_cmd_setup(archiver)
archive, repository = open_archive(archiver.repository_path, "archive1") archive, repository = open_archive(archiver.repository_path, "archive1")
if isinstance(repository, Repository3):
pytest.skip("Test not adapted to Repository3")
with repository: with repository:
manifest = repository.get(Manifest.MANIFEST_ID) manifest = repository.get_manifest()
corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:] corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:]
repository.put(Manifest.MANIFEST_ID, corrupted_manifest) repository.put_manifest(corrupted_manifest)
repository.commit(compact=False) repository.commit(compact=False)
cmd(archiver, "check", exit_code=1) cmd(archiver, "check", exit_code=1)
output = cmd(archiver, "check", "-v", "--repair", exit_code=0) output = cmd(archiver, "check", "-v", "--repair", exit_code=0)
@ -226,8 +226,6 @@ def test_spoofed_manifest(archivers, request):
archiver = request.getfixturevalue(archivers) archiver = request.getfixturevalue(archivers)
check_cmd_setup(archiver) check_cmd_setup(archiver)
archive, repository = open_archive(archiver.repository_path, "archive1") archive, repository = open_archive(archiver.repository_path, "archive1")
if isinstance(repository, Repository3):
pytest.skip("Test not adapted to Repository3")
with repository: with repository:
manifest = Manifest.load(repository, Manifest.NO_OPERATION_CHECK) manifest = Manifest.load(repository, Manifest.NO_OPERATION_CHECK)
cdata = manifest.repo_objs.format( cdata = manifest.repo_objs.format(
@ -247,7 +245,7 @@ def test_spoofed_manifest(archivers, request):
) )
# maybe a repo-side attacker could manage to move the fake manifest file chunk over to the manifest ID. # maybe a repo-side attacker could manage to move the fake manifest file chunk over to the manifest ID.
# we simulate this here by directly writing the fake manifest data to the manifest ID. # we simulate this here by directly writing the fake manifest data to the manifest ID.
repository.put(Manifest.MANIFEST_ID, cdata) repository.put_manifest(cdata)
repository.commit(compact=False) repository.commit(compact=False)
# borg should notice that the manifest has the wrong ro_type. # borg should notice that the manifest has the wrong ro_type.
cmd(archiver, "check", exit_code=1) cmd(archiver, "check", exit_code=1)
@ -262,12 +260,10 @@ def test_manifest_rebuild_corrupted_chunk(archivers, request):
archiver = request.getfixturevalue(archivers) archiver = request.getfixturevalue(archivers)
check_cmd_setup(archiver) check_cmd_setup(archiver)
archive, repository = open_archive(archiver.repository_path, "archive1") archive, repository = open_archive(archiver.repository_path, "archive1")
if isinstance(repository, Repository3):
pytest.skip("Test not adapted to Repository3")
with repository: with repository:
manifest = repository.get(Manifest.MANIFEST_ID) manifest = repository.get_manifest()
corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:] corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:]
repository.put(Manifest.MANIFEST_ID, corrupted_manifest) repository.put_manifest(corrupted_manifest)
chunk = repository.get(archive.id) chunk = repository.get(archive.id)
corrupted_chunk = chunk + b"corrupted!" corrupted_chunk = chunk + b"corrupted!"
repository.put(archive.id, corrupted_chunk) repository.put(archive.id, corrupted_chunk)
@ -282,13 +278,11 @@ def test_manifest_rebuild_duplicate_archive(archivers, request):
archiver = request.getfixturevalue(archivers) archiver = request.getfixturevalue(archivers)
check_cmd_setup(archiver) check_cmd_setup(archiver)
archive, repository = open_archive(archiver.repository_path, "archive1") archive, repository = open_archive(archiver.repository_path, "archive1")
if isinstance(repository, Repository3):
pytest.skip("Test not adapted to Repository3")
repo_objs = archive.repo_objs repo_objs = archive.repo_objs
with repository: with repository:
manifest = repository.get(Manifest.MANIFEST_ID) manifest = repository.get_manifest()
corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:] corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:]
repository.put(Manifest.MANIFEST_ID, corrupted_manifest) repository.put_manifest(corrupted_manifest)
archive_dict = { archive_dict = {
"command_line": "", "command_line": "",
"item_ptrs": [], "item_ptrs": [],
@ -314,14 +308,12 @@ def test_spoofed_archive(archivers, request):
archiver = request.getfixturevalue(archivers) archiver = request.getfixturevalue(archivers)
check_cmd_setup(archiver) check_cmd_setup(archiver)
archive, repository = open_archive(archiver.repository_path, "archive1") archive, repository = open_archive(archiver.repository_path, "archive1")
if isinstance(repository, Repository3):
pytest.skip("Test not adapted to Repository3")
repo_objs = archive.repo_objs repo_objs = archive.repo_objs
with repository: with repository:
# attacker would corrupt or delete the manifest to trigger a rebuild of it: # attacker would corrupt or delete the manifest to trigger a rebuild of it:
manifest = repository.get(Manifest.MANIFEST_ID) manifest = repository.get_manifest()
corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:] corrupted_manifest = manifest[:123] + b"corrupted!" + manifest[123:]
repository.put(Manifest.MANIFEST_ID, corrupted_manifest) repository.put_manifest(corrupted_manifest)
archive_dict = { archive_dict = {
"command_line": "", "command_line": "",
"item_ptrs": [], "item_ptrs": [],