| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610 | 
							- from binascii import hexlify
 
- from configparser import RawConfigParser
 
- import os
 
- from io import StringIO
 
- import stat
 
- import subprocess
 
- import sys
 
- import shutil
 
- import tempfile
 
- import time
 
- import unittest
 
- from hashlib import sha256
 
- from attic import xattr
 
- from attic.archive import Archive, ChunkBuffer, CHUNK_MAX
 
- from attic.archiver import Archiver
 
- from attic.cache import Cache
 
- from attic.crypto import bytes_to_long, num_aes_blocks
 
- from attic.helpers import Manifest
 
- from attic.remote import RemoteRepository, PathNotAllowed
 
- from attic.repository import Repository
 
- from attic.testsuite import BaseTestCase
 
- from attic.testsuite.mock import patch
 
- try:
 
-     import llfuse
 
-     has_llfuse = True
 
- except ImportError:
 
-     has_llfuse = False
 
- has_lchflags = hasattr(os, 'lchflags')
 
- src_dir = os.path.join(os.getcwd(), os.path.dirname(__file__), '..')
 
- class changedir:
 
-     def __init__(self, dir):
 
-         self.dir = dir
 
-     def __enter__(self):
 
-         self.old = os.getcwd()
 
-         os.chdir(self.dir)
 
-     def __exit__(self, *args, **kw):
 
-         os.chdir(self.old)
 
- class environment_variable:
 
-     def __init__(self, **values):
 
-         self.values = values
 
-         self.old_values = {}
 
-     def __enter__(self):
 
-         for k, v in self.values.items():
 
-             self.old_values[k] = os.environ.get(k)
 
-             os.environ[k] = v
 
-     def __exit__(self, *args, **kw):
 
-         for k, v in self.old_values.items():
 
-             if v is not None:
 
-                 os.environ[k] = v
 
- class ArchiverTestCaseBase(BaseTestCase):
 
-     prefix = ''
 
-     def setUp(self):
 
-         os.environ['BORG_CHECK_I_KNOW_WHAT_I_AM_DOING'] = '1'
 
-         self.archiver = Archiver()
 
-         self.tmpdir = tempfile.mkdtemp()
 
-         self.repository_path = os.path.join(self.tmpdir, 'repository')
 
-         self.repository_location = self.prefix + self.repository_path
 
-         self.input_path = os.path.join(self.tmpdir, 'input')
 
-         self.output_path = os.path.join(self.tmpdir, 'output')
 
-         self.keys_path = os.path.join(self.tmpdir, 'keys')
 
-         self.cache_path = os.path.join(self.tmpdir, 'cache')
 
-         self.exclude_file_path = os.path.join(self.tmpdir, 'excludes')
 
-         os.environ['BORG_KEYS_DIR'] = self.keys_path
 
-         os.environ['BORG_CACHE_DIR'] = self.cache_path
 
-         os.mkdir(self.input_path)
 
-         os.mkdir(self.output_path)
 
-         os.mkdir(self.keys_path)
 
-         os.mkdir(self.cache_path)
 
-         with open(self.exclude_file_path, 'wb') as fd:
 
-             fd.write(b'input/file2\n# A commment line, then a blank line\n\n')
 
-         self._old_wd = os.getcwd()
 
-         os.chdir(self.tmpdir)
 
-     def tearDown(self):
 
-         shutil.rmtree(self.tmpdir)
 
-         os.chdir(self._old_wd)
 
-     def cmd(self, *args, **kw):
 
-         exit_code = kw.get('exit_code', 0)
 
-         fork = kw.get('fork', False)
 
-         if fork:
 
-             try:
 
-                 output = subprocess.check_output((sys.executable, '-m', 'attic.archiver') + args)
 
-                 ret = 0
 
-             except subprocess.CalledProcessError as e:
 
-                 output = e.output
 
-                 ret = e.returncode
 
-             output = os.fsdecode(output)
 
-             if ret != exit_code:
 
-                 print(output)
 
-             self.assert_equal(exit_code, ret)
 
-             return output
 
-         args = list(args)
 
-         stdin, stdout, stderr = sys.stdin, sys.stdout, sys.stderr
 
-         try:
 
-             sys.stdin = StringIO()
 
-             output = StringIO()
 
-             sys.stdout = sys.stderr = output
 
-             ret = self.archiver.run(args)
 
-             sys.stdin, sys.stdout, sys.stderr = stdin, stdout, stderr
 
-             if ret != exit_code:
 
-                 print(output.getvalue())
 
-             self.assert_equal(exit_code, ret)
 
-             return output.getvalue()
 
-         finally:
 
-             sys.stdin, sys.stdout, sys.stderr = stdin, stdout, stderr
 
-     def create_src_archive(self, name):
 
-         self.cmd('create', self.repository_location + '::' + name, src_dir)
 
- class ArchiverTestCase(ArchiverTestCaseBase):
 
-     def create_regular_file(self, name, size=0, contents=None):
 
-         filename = os.path.join(self.input_path, name)
 
-         if not os.path.exists(os.path.dirname(filename)):
 
-             os.makedirs(os.path.dirname(filename))
 
-         with open(filename, 'wb') as fd:
 
-             if contents is None:
 
-                 contents = b'X' * size
 
-             fd.write(contents)
 
-     def create_test_files(self):
 
-         """Create a minimal test case including all supported file types
 
-         """
 
-         # File
 
-         self.create_regular_file('empty', size=0)
 
-         # next code line raises OverflowError on 32bit cpu (raspberry pi 2):
 
-         # 2600-01-01 > 2**64 ns
 
-         #os.utime('input/empty', (19880895600, 19880895600))
 
-         # thus, we better test with something not that far in future:
 
-         # 2038-01-19 (1970 + 2^31 - 1 seconds) is the 32bit "deadline":
 
-         os.utime('input/empty', (2**31 - 1, 2**31 - 1))
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.create_regular_file('flagfile', size=1024)
 
-         # Directory
 
-         self.create_regular_file('dir2/file2', size=1024 * 80)
 
-         # File owner
 
-         os.chown('input/file1', 100, 200)
 
-         # File mode
 
-         os.chmod('input/file1', 0o7755)
 
-         os.chmod('input/dir2', 0o555)
 
-         # Block device
 
-         os.mknod('input/bdev', 0o600 | stat.S_IFBLK,  os.makedev(10, 20))
 
-         # Char device
 
-         os.mknod('input/cdev', 0o600 | stat.S_IFCHR,  os.makedev(30, 40))
 
-         # Hard link
 
-         os.link(os.path.join(self.input_path, 'file1'),
 
-                 os.path.join(self.input_path, 'hardlink'))
 
-         # Symlink
 
-         os.symlink('somewhere', os.path.join(self.input_path, 'link1'))
 
-         if xattr.is_enabled(self.input_path):
 
-             xattr.setxattr(os.path.join(self.input_path, 'file1'), 'user.foo', b'bar')
 
-             # XXX this always fails for me
 
-             # ubuntu 14.04, on a TMP dir filesystem with user_xattr, using fakeroot
 
-             # same for newer ubuntu and centos.
 
-             # if this is supported just on specific platform, platform should be checked first,
 
-             # so that the test setup for all tests using it does not fail here always for others.
 
-             #xattr.setxattr(os.path.join(self.input_path, 'link1'), 'user.foo_symlink', b'bar_symlink', follow_symlinks=False)
 
-         # FIFO node
 
-         os.mkfifo(os.path.join(self.input_path, 'fifo1'))
 
-         if has_lchflags:
 
-             os.lchflags(os.path.join(self.input_path, 'flagfile'), stat.UF_NODUMP)
 
-     def test_basic_functionality(self):
 
-         self.create_test_files()
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         self.cmd('create', self.repository_location + '::test.2', 'input')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test')
 
-         self.assert_equal(len(self.cmd('list', self.repository_location).splitlines()), 2)
 
-         self.assert_equal(len(self.cmd('list', self.repository_location + '::test').splitlines()), 11)
 
-         self.assert_dirs_equal('input', 'output/input')
 
-         info_output = self.cmd('info', self.repository_location + '::test')
 
-         self.assert_in('Number of files: 4', info_output)
 
-         shutil.rmtree(self.cache_path)
 
-         with environment_variable(BORG_UNKNOWN_UNENCRYPTED_REPO_ACCESS_IS_OK='1'):
 
-             info_output2 = self.cmd('info', self.repository_location + '::test')
 
-         # info_output2 starts with some "initializing cache" text but should
 
-         # end the same way as info_output
 
-         assert info_output2.endswith(info_output)
 
-     def _extract_repository_id(self, path):
 
-         return Repository(self.repository_path).id
 
-     def _set_repository_id(self, path, id):
 
-         config = RawConfigParser()
 
-         config.read(os.path.join(path, 'config'))
 
-         config.set('repository', 'id', hexlify(id).decode('ascii'))
 
-         with open(os.path.join(path, 'config'), 'w') as fd:
 
-             config.write(fd)
 
-         return Repository(self.repository_path).id
 
-     def test_sparse_file(self):
 
-         # no sparse file support on Mac OS X
 
-         sparse_support = sys.platform != 'darwin'
 
-         filename = os.path.join(self.input_path, 'sparse')
 
-         content = b'foobar'
 
-         hole_size = 5 * CHUNK_MAX  # 5 full chunker buffers
 
-         with open(filename, 'wb') as fd:
 
-             # create a file that has a hole at the beginning and end (if the
 
-             # OS and filesystem supports sparse files)
 
-             fd.seek(hole_size, 1)
 
-             fd.write(content)
 
-             fd.seek(hole_size, 1)
 
-             pos = fd.tell()
 
-             fd.truncate(pos)
 
-         total_len = hole_size + len(content) + hole_size
 
-         st = os.stat(filename)
 
-         self.assert_equal(st.st_size, total_len)
 
-         if sparse_support and hasattr(st, 'st_blocks'):
 
-             self.assert_true(st.st_blocks * 512 < total_len / 10)  # is input sparse?
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         with changedir('output'):
 
-             self.cmd('extract', '--sparse', self.repository_location + '::test')
 
-         self.assert_dirs_equal('input', 'output/input')
 
-         filename = os.path.join(self.output_path, 'input', 'sparse')
 
-         with open(filename, 'rb') as fd:
 
-             # check if file contents are as expected
 
-             self.assert_equal(fd.read(hole_size), b'\0' * hole_size)
 
-             self.assert_equal(fd.read(len(content)), content)
 
-             self.assert_equal(fd.read(hole_size), b'\0' * hole_size)
 
-         st = os.stat(filename)
 
-         self.assert_equal(st.st_size, total_len)
 
-         if sparse_support and hasattr(st, 'st_blocks'):
 
-             self.assert_true(st.st_blocks * 512 < total_len / 10)  # is output sparse?
 
-     def test_repository_swap_detection(self):
 
-         self.create_test_files()
 
-         os.environ['BORG_PASSPHRASE'] = 'passphrase'
 
-         self.cmd('init', '--encryption=passphrase', self.repository_location)
 
-         repository_id = self._extract_repository_id(self.repository_path)
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         shutil.rmtree(self.repository_path)
 
-         self.cmd('init', '--encryption=none', self.repository_location)
 
-         self._set_repository_id(self.repository_path, repository_id)
 
-         self.assert_equal(repository_id, self._extract_repository_id(self.repository_path))
 
-         self.assert_raises(Cache.EncryptionMethodMismatch, lambda :self.cmd('create', self.repository_location + '::test.2', 'input'))
 
-     def test_repository_swap_detection2(self):
 
-         self.create_test_files()
 
-         self.cmd('init', '--encryption=none', self.repository_location + '_unencrypted')
 
-         os.environ['BORG_PASSPHRASE'] = 'passphrase'
 
-         self.cmd('init', '--encryption=passphrase', self.repository_location + '_encrypted')
 
-         self.cmd('create', self.repository_location + '_encrypted::test', 'input')
 
-         shutil.rmtree(self.repository_path + '_encrypted')
 
-         os.rename(self.repository_path + '_unencrypted', self.repository_path + '_encrypted')
 
-         self.assert_raises(Cache.RepositoryAccessAborted, lambda :self.cmd('create', self.repository_location + '_encrypted::test.2', 'input'))
 
-     def test_strip_components(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.create_regular_file('dir/file')
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test', '--strip-components', '3')
 
-             self.assert_true(not os.path.exists('file'))
 
-             with self.assert_creates_file('file'):
 
-                 self.cmd('extract', self.repository_location + '::test', '--strip-components', '2')
 
-             with self.assert_creates_file('dir/file'):
 
-                 self.cmd('extract', self.repository_location + '::test', '--strip-components', '1')
 
-             with self.assert_creates_file('input/dir/file'):
 
-                 self.cmd('extract', self.repository_location + '::test', '--strip-components', '0')
 
-     def test_extract_include_exclude(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.create_regular_file('file2', size=1024 * 80)
 
-         self.create_regular_file('file3', size=1024 * 80)
 
-         self.create_regular_file('file4', size=1024 * 80)
 
-         self.cmd('create', '--exclude=input/file4', self.repository_location + '::test', 'input')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test', 'input/file1', )
 
-         self.assert_equal(sorted(os.listdir('output/input')), ['file1'])
 
-         with changedir('output'):
 
-             self.cmd('extract', '--exclude=input/file2', self.repository_location + '::test')
 
-         self.assert_equal(sorted(os.listdir('output/input')), ['file1', 'file3'])
 
-         with changedir('output'):
 
-             self.cmd('extract', '--exclude-from=' + self.exclude_file_path, self.repository_location + '::test')
 
-         self.assert_equal(sorted(os.listdir('output/input')), ['file1', 'file3'])
 
-     def test_exclude_caches(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.create_regular_file('cache1/CACHEDIR.TAG', contents=b'Signature: 8a477f597d28d172789f06886806bc55 extra stuff')
 
-         self.create_regular_file('cache2/CACHEDIR.TAG', contents=b'invalid signature')
 
-         self.cmd('create', '--exclude-caches', self.repository_location + '::test', 'input')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test')
 
-         self.assert_equal(sorted(os.listdir('output/input')), ['cache2', 'file1'])
 
-         self.assert_equal(sorted(os.listdir('output/input/cache2')), ['CACHEDIR.TAG'])
 
-     def test_path_normalization(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.create_regular_file('dir1/dir2/file', size=1024 * 80)
 
-         with changedir('input/dir1/dir2'):
 
-             self.cmd('create', self.repository_location + '::test', '../../../input/dir1/../dir1/dir2/..')
 
-         output = self.cmd('list', self.repository_location + '::test')
 
-         self.assert_not_in('..', output)
 
-         self.assert_in(' input/dir1/dir2/file', output)
 
-     def test_exclude_normalization(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.create_regular_file('file2', size=1024 * 80)
 
-         with changedir('input'):
 
-             self.cmd('create', '--exclude=file1', self.repository_location + '::test1', '.')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test1')
 
-         self.assert_equal(sorted(os.listdir('output')), ['file2'])
 
-         with changedir('input'):
 
-             self.cmd('create', '--exclude=./file1', self.repository_location + '::test2', '.')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test2')
 
-         self.assert_equal(sorted(os.listdir('output')), ['file2'])
 
-         self.cmd('create', '--exclude=input/./file1', self.repository_location + '::test3', 'input')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test3')
 
-         self.assert_equal(sorted(os.listdir('output/input')), ['file2'])
 
-     def test_repeated_files(self):
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test', 'input', 'input')
 
-     def test_overwrite(self):
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.create_regular_file('dir2/file2', size=1024 * 80)
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         # Overwriting regular files and directories should be supported
 
-         os.mkdir('output/input')
 
-         os.mkdir('output/input/file1')
 
-         os.mkdir('output/input/dir2')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test')
 
-         self.assert_dirs_equal('input', 'output/input')
 
-         # But non-empty dirs should fail
 
-         os.unlink('output/input/file1')
 
-         os.mkdir('output/input/file1')
 
-         os.mkdir('output/input/file1/dir')
 
-         with changedir('output'):
 
-             self.cmd('extract', self.repository_location + '::test', exit_code=1)
 
-     def test_rename(self):
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.create_regular_file('dir2/file2', size=1024 * 80)
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         self.cmd('create', self.repository_location + '::test.2', 'input')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test.2')
 
-         self.cmd('rename', self.repository_location + '::test', 'test.3')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test.2')
 
-         self.cmd('rename', self.repository_location + '::test.2', 'test.4')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test.3')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test.4')
 
-         # Make sure both archives have been renamed
 
-         repository = Repository(self.repository_path)
 
-         manifest, key = Manifest.load(repository)
 
-         self.assert_equal(len(manifest.archives), 2)
 
-         self.assert_in('test.3', manifest.archives)
 
-         self.assert_in('test.4', manifest.archives)
 
-     def test_delete(self):
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.create_regular_file('dir2/file2', size=1024 * 80)
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         self.cmd('create', self.repository_location + '::test.2', 'input')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test.2')
 
-         self.cmd('delete', self.repository_location + '::test')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test.2')
 
-         self.cmd('delete', self.repository_location + '::test.2')
 
-         # Make sure all data except the manifest has been deleted
 
-         repository = Repository(self.repository_path)
 
-         self.assert_equal(len(repository), 1)
 
-     def test_corrupted_repository(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.create_src_archive('test')
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::test')
 
-         self.cmd('check', self.repository_location)
 
-         name = sorted(os.listdir(os.path.join(self.tmpdir, 'repository', 'data', '0')), reverse=True)[0]
 
-         with open(os.path.join(self.tmpdir, 'repository', 'data', '0', name), 'r+') as fd:
 
-             fd.seek(100)
 
-             fd.write('XXXX')
 
-         self.cmd('check', self.repository_location, exit_code=1)
 
-     def test_readonly_repository(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.create_src_archive('test')
 
-         os.system('chmod -R ugo-w ' + self.repository_path)
 
-         try:
 
-             self.cmd('extract', '--dry-run', self.repository_location + '::test')
 
-         finally:
 
-             # Restore permissions so shutil.rmtree is able to delete it
 
-             os.system('chmod -R u+w ' + self.repository_path)
 
-     def test_cmdline_compatibility(self):
 
-         self.create_regular_file('file1', size=1024 * 80)
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         output = self.cmd('verify', '-v', self.repository_location + '::test')
 
-         self.assert_in('"borg verify" has been deprecated', output)
 
-         output = self.cmd('prune', self.repository_location, '--hourly=1')
 
-         self.assert_in('"--hourly" has been deprecated. Use "--keep-hourly" instead', output)
 
-     def test_prune_repository(self):
 
-         self.cmd('init', self.repository_location)
 
-         self.cmd('create', self.repository_location + '::test1', src_dir)
 
-         self.cmd('create', self.repository_location + '::test2', src_dir)
 
-         output = self.cmd('prune', '-v', '--dry-run', self.repository_location, '--keep-daily=2')
 
-         self.assert_in('Keeping archive: test2', output)
 
-         self.assert_in('Would prune:     test1', output)
 
-         output = self.cmd('list', self.repository_location)
 
-         self.assert_in('test1', output)
 
-         self.assert_in('test2', output)
 
-         self.cmd('prune', self.repository_location, '--keep-daily=2')
 
-         output = self.cmd('list', self.repository_location)
 
-         self.assert_not_in('test1', output)
 
-         self.assert_in('test2', output)
 
-     def test_usage(self):
 
-         self.assert_raises(SystemExit, lambda: self.cmd())
 
-         self.assert_raises(SystemExit, lambda: self.cmd('-h'))
 
-     @unittest.skipUnless(has_llfuse, 'llfuse not installed')
 
-     def test_fuse_mount_repository(self):
 
-         mountpoint = os.path.join(self.tmpdir, 'mountpoint')
 
-         os.mkdir(mountpoint)
 
-         self.cmd('init', self.repository_location)
 
-         self.create_test_files()
 
-         self.cmd('create', self.repository_location + '::archive', 'input')
 
-         self.cmd('create', self.repository_location + '::archive2', 'input')
 
-         try:
 
-             self.cmd('mount', self.repository_location, mountpoint, fork=True)
 
-             self.wait_for_mount(mountpoint)
 
-             self.assert_dirs_equal(self.input_path, os.path.join(mountpoint, 'archive', 'input'))
 
-             self.assert_dirs_equal(self.input_path, os.path.join(mountpoint, 'archive2', 'input'))
 
-         finally:
 
-             if sys.platform.startswith('linux'):
 
-                 os.system('fusermount -u ' + mountpoint)
 
-             else:
 
-                 os.system('umount ' + mountpoint)
 
-             os.rmdir(mountpoint)
 
-             # Give the daemon some time to exit
 
-             time.sleep(.2)
 
-     @unittest.skipUnless(has_llfuse, 'llfuse not installed')
 
-     def test_fuse_mount_archive(self):
 
-         mountpoint = os.path.join(self.tmpdir, 'mountpoint')
 
-         os.mkdir(mountpoint)
 
-         self.cmd('init', self.repository_location)
 
-         self.create_test_files()
 
-         self.cmd('create', self.repository_location + '::archive', 'input')
 
-         try:
 
-             self.cmd('mount', self.repository_location + '::archive', mountpoint, fork=True)
 
-             self.wait_for_mount(mountpoint)
 
-             self.assert_dirs_equal(self.input_path, os.path.join(mountpoint, 'input'))
 
-         finally:
 
-             if sys.platform.startswith('linux'):
 
-                 os.system('fusermount -u ' + mountpoint)
 
-             else:
 
-                 os.system('umount ' + mountpoint)
 
-             os.rmdir(mountpoint)
 
-             # Give the daemon some time to exit
 
-             time.sleep(.2)
 
-     def verify_aes_counter_uniqueness(self, method):
 
-         seen = set()  # Chunks already seen
 
-         used = set()  # counter values already used
 
-         def verify_uniqueness():
 
-             repository = Repository(self.repository_path)
 
-             for key, _ in repository.open_index(repository.get_transaction_id()).iteritems():
 
-                 data = repository.get(key)
 
-                 hash = sha256(data).digest()
 
-                 if hash not in seen:
 
-                     seen.add(hash)
 
-                     num_blocks = num_aes_blocks(len(data) - 41)
 
-                     nonce = bytes_to_long(data[33:41])
 
-                     for counter in range(nonce, nonce + num_blocks):
 
-                         self.assert_not_in(counter, used)
 
-                         used.add(counter)
 
-         self.create_test_files()
 
-         os.environ['BORG_PASSPHRASE'] = 'passphrase'
 
-         self.cmd('init', '--encryption=' + method, self.repository_location)
 
-         verify_uniqueness()
 
-         self.cmd('create', self.repository_location + '::test', 'input')
 
-         verify_uniqueness()
 
-         self.cmd('create', self.repository_location + '::test.2', 'input')
 
-         verify_uniqueness()
 
-         self.cmd('delete', self.repository_location + '::test.2')
 
-         verify_uniqueness()
 
-         self.assert_equal(used, set(range(len(used))))
 
-     def test_aes_counter_uniqueness_keyfile(self):
 
-         self.verify_aes_counter_uniqueness('keyfile')
 
-     def test_aes_counter_uniqueness_passphrase(self):
 
-         self.verify_aes_counter_uniqueness('passphrase')
 
- class ArchiverCheckTestCase(ArchiverTestCaseBase):
 
-     def setUp(self):
 
-         super(ArchiverCheckTestCase, self).setUp()
 
-         with patch.object(ChunkBuffer, 'BUFFER_SIZE', 10):
 
-             self.cmd('init', self.repository_location)
 
-             self.create_src_archive('archive1')
 
-             self.create_src_archive('archive2')
 
-     def open_archive(self, name):
 
-         repository = Repository(self.repository_path)
 
-         manifest, key = Manifest.load(repository)
 
-         archive = Archive(repository, key, manifest, name)
 
-         return archive, repository
 
-     def test_check_usage(self):
 
-         output = self.cmd('check', self.repository_location, exit_code=0)
 
-         self.assert_in('Starting repository check', output)
 
-         self.assert_in('Starting archive consistency check', output)
 
-         output = self.cmd('check', '--repository-only', self.repository_location, exit_code=0)
 
-         self.assert_in('Starting repository check', output)
 
-         self.assert_not_in('Starting archive consistency check', output)
 
-         output = self.cmd('check', '--archives-only', self.repository_location, exit_code=0)
 
-         self.assert_not_in('Starting repository check', output)
 
-         self.assert_in('Starting archive consistency check', output)
 
-     def test_missing_file_chunk(self):
 
-         archive, repository = self.open_archive('archive1')
 
-         for item in archive.iter_items():
 
-             if item[b'path'].endswith('testsuite/archiver.py'):
 
-                 repository.delete(item[b'chunks'][-1][0])
 
-                 break
 
-         repository.commit()
 
-         self.cmd('check', self.repository_location, exit_code=1)
 
-         self.cmd('check', '--repair', self.repository_location, exit_code=0)
 
-         self.cmd('check', self.repository_location, exit_code=0)
 
-     def test_missing_archive_item_chunk(self):
 
-         archive, repository = self.open_archive('archive1')
 
-         repository.delete(archive.metadata[b'items'][-5])
 
-         repository.commit()
 
-         self.cmd('check', self.repository_location, exit_code=1)
 
-         self.cmd('check', '--repair', self.repository_location, exit_code=0)
 
-         self.cmd('check', self.repository_location, exit_code=0)
 
-     def test_missing_archive_metadata(self):
 
-         archive, repository = self.open_archive('archive1')
 
-         repository.delete(archive.id)
 
-         repository.commit()
 
-         self.cmd('check', self.repository_location, exit_code=1)
 
-         self.cmd('check', '--repair', self.repository_location, exit_code=0)
 
-         self.cmd('check', self.repository_location, exit_code=0)
 
-     def test_missing_manifest(self):
 
-         archive, repository = self.open_archive('archive1')
 
-         repository.delete(Manifest.MANIFEST_ID)
 
-         repository.commit()
 
-         self.cmd('check', self.repository_location, exit_code=1)
 
-         output = self.cmd('check', '--repair', self.repository_location, exit_code=0)
 
-         self.assert_in('archive1', output)
 
-         self.assert_in('archive2', output)
 
-         self.cmd('check', self.repository_location, exit_code=0)
 
-     def test_extra_chunks(self):
 
-         self.cmd('check', self.repository_location, exit_code=0)
 
-         repository = Repository(self.repository_location)
 
-         repository.put(b'01234567890123456789012345678901', b'xxxx')
 
-         repository.commit()
 
-         repository.close()
 
-         self.cmd('check', self.repository_location, exit_code=1)
 
-         self.cmd('check', self.repository_location, exit_code=1)
 
-         self.cmd('check', '--repair', self.repository_location, exit_code=0)
 
-         self.cmd('check', self.repository_location, exit_code=0)
 
-         self.cmd('extract', '--dry-run', self.repository_location + '::archive1', exit_code=0)
 
- class RemoteArchiverTestCase(ArchiverTestCase):
 
-     prefix = '__testsuite__:'
 
-     def test_remote_repo_restrict_to_path(self):
 
-         self.cmd('init', self.repository_location)
 
-         path_prefix = os.path.dirname(self.repository_path)
 
-         with patch.object(RemoteRepository, 'extra_test_args', ['--restrict-to-path', '/foo']):
 
-             self.assert_raises(PathNotAllowed, lambda: self.cmd('init', self.repository_location + '_1'))
 
-         with patch.object(RemoteRepository, 'extra_test_args', ['--restrict-to-path', path_prefix]):
 
-             self.cmd('init', self.repository_location + '_2')
 
-         with patch.object(RemoteRepository, 'extra_test_args', ['--restrict-to-path', '/foo', '--restrict-to-path', path_prefix]):
 
-             self.cmd('init', self.repository_location + '_3')
 
 
  |