Sfoglia il codice sorgente

Fix windows build after merge with master.

Antti Aalto 8 anni fa
parent
commit
9fe09c831f

+ 10 - 7
setup.py

@@ -23,12 +23,15 @@ if my_python < min_python:
 # Are we building on ReadTheDocs?
 on_rtd = os.environ.get('READTHEDOCS')
 
-install_requires = [
-    # msgpack pure python data corruption was fixed in 0.4.6.
-    # Also, we might use some rather recent API features.
-    'msgpack-python>=0.4.6',
-    'pyzmq',
-]
+if sys.platform != 'win32':
+    install_requires = [
+        # msgpack pure python data corruption was fixed in 0.4.6.
+        # Also, we might use some rather recent API features.
+        'msgpack-python>=0.4.6',
+        'pyzmq',
+    ]
+else:
+    install_requires = ['msgpack-python>=0.4.6']
 
 # note for package maintainers: if you package borgbackup for distribution,
 # please add llfuse as a *requirement* on all platforms that have a working
@@ -790,7 +793,7 @@ if not on_rtd:
 
 def parse(root, describe_command=None):
     file = open('src/borg/_version.py', 'w')
-    output = subprocess.check_output("git describe --tags --long").decode().strip()
+    output = subprocess.check_output("git describe --long").decode().strip()
     file.write('version = "' + output + '"\n')
     return output
 

+ 13 - 11
src/borg/archive.py

@@ -667,8 +667,9 @@ Utilization of max. archive size: {csize_max:.0%}
         backup_io.op = 'attrs'
         uid = gid = None
         if not self.numeric_owner:
-            uid = user2uid(item.user)
-            gid = group2gid(item.group)
+            if sys.platform != 'win32':
+                uid = user2uid(item.user)
+                gid = group2gid(item.group)
         uid = item.uid if uid is None else uid
         gid = item.gid if gid is None else gid
         # This code is a bit of a mess due to os specific differences
@@ -700,7 +701,7 @@ Utilization of max. archive size: {csize_max:.0%}
         try:
             if sys.platform == 'win32':
                 os.utime(path, ns=(atime, mtime))
-            if fd:
+            elif fd:
                 os.utime(fd, None, ns=(atime, mtime))
             else:
                 os.utime(path, None, ns=(atime, mtime), follow_symlinks=False)
@@ -934,12 +935,6 @@ class MetadataCollector:
             gid=st.st_gid,
             mtime=safe_ns(st.st_mtime_ns),
         )
-        if sys.platform == 'win32':
-            user_name, user_sid = get_owner(path)
-            attrs.update({
-                'uid': 0, 'user_sid': user_sid, 'user': user_name,
-                'gid': st.st_gid, 'group': gid2group(st.st_gid),
-            })
         # borg can work with archives only having mtime (older attic archives do not have
         # atime/ctime). it can be useful to omit atime/ctime, if they change without the
         # file content changing - e.g. to get better metadata deduplication.
@@ -950,8 +945,9 @@ class MetadataCollector:
         if self.numeric_owner:
             attrs['user'] = attrs['group'] = None
         else:
-            attrs['user'] = uid2user(st.st_uid)
-            attrs['group'] = gid2group(st.st_gid)
+            if sys.platform != 'win32':
+                attrs['user'] = uid2user(st.st_uid)
+                attrs['group'] = gid2group(st.st_gid)
         return attrs
 
     def stat_ext_attrs(self, st, path):
@@ -970,6 +966,12 @@ class MetadataCollector:
 
     def stat_attrs(self, st, path):
         attrs = self.stat_simple_attrs(st)
+        if sys.platform == 'win32':
+            user_name, user_sid = get_owner(path)
+            attrs.update({
+                'uid': 0, 'user_sid': user_sid, 'user': user_name,
+                'gid': st.st_gid, 'group': '',
+            })
         attrs.update(self.stat_ext_attrs(st, path))
         return attrs
 

+ 4 - 2
src/borg/archiver.py

@@ -1188,7 +1188,10 @@ class Archiver:
         elif args.short:
             format = "{path}{NL}"
         else:
-            format = "{mode} {user:6} {group:6} {size:8} {mtime} {path}{extra}{NL}"
+            if sys.platform != 'win32':
+                format = "{mode} {user:6} {group:6} {size:8} {mtime} {path}{extra}{NL}"
+            else:
+                format = "{user:8} {size:8} {mtime} {path}{extra}{NL}"
 
         def _list_inner(cache):
             archive = Archive(repository, key, manifest, args.location.archive, cache=cache,
@@ -1207,7 +1210,6 @@ class Archiver:
 
         return self.exit_code
 
-
     def _list_repository(self, args, repository, manifest, key, write):
         if args.format is not None:
             format = args.format

+ 0 - 1620
src/borg/helpers.py

@@ -1,1620 +0,0 @@
-import argparse
-import getpass
-import hashlib
-import logging
-import os
-import os.path
-import platform
-import re
-import signal
-import socket
-import sys
-import stat
-import textwrap
-import time
-import unicodedata
-import uuid
-from binascii import hexlify
-from collections import namedtuple, deque
-from contextlib import contextmanager
-from datetime import datetime, timezone, timedelta
-from fnmatch import translate
-from functools import wraps, partial
-from itertools import islice
-from operator import attrgetter
-from string import Formatter
-
-if sys.platform != 'win32':
-    import grp
-    import pwd
-else:
-    import posixpath
-
-
-import msgpack
-import msgpack.fallback
-
-from .logger import create_logger
-logger = create_logger()
-
-from . import __version__ as borg_version
-from . import chunker
-from . import crypto
-from . import hashindex
-from . import shellpattern
-from .constants import *  # NOQA
-from .compress import COMPR_BUFFER, get_compressor
-
-# meta dict, data bytes
-_Chunk = namedtuple('_Chunk', 'meta data')
-
-
-def Chunk(data, **meta):
-    return _Chunk(meta, data)
-
-
-class Error(Exception):
-    """Error base class"""
-
-    # if we raise such an Error and it is only catched by the uppermost
-    # exception handler (that exits short after with the given exit_code),
-    # it is always a (fatal and abrupt) EXIT_ERROR, never just a warning.
-    exit_code = EXIT_ERROR
-    # show a traceback?
-    traceback = False
-
-    def get_message(self):
-        return type(self).__doc__.format(*self.args)
-
-
-class ErrorWithTraceback(Error):
-    """like Error, but show a traceback also"""
-    traceback = True
-
-
-class IntegrityError(ErrorWithTraceback):
-    """Data integrity error"""
-
-
-class ExtensionModuleError(Error):
-    """The Borg binary extension modules do not seem to be properly installed"""
-
-
-class NoManifestError(Error):
-    """Repository has no manifest."""
-
-
-class PlaceholderError(Error):
-    """Formatting Error: "{}".format({}): {}({})"""
-
-
-def check_extension_modules():
-    from . import platform
-    if hashindex.API_VERSION != 2:
-        raise ExtensionModuleError
-    if chunker.API_VERSION != 2:
-        raise ExtensionModuleError
-    if crypto.API_VERSION != 3:
-        raise ExtensionModuleError
-    if platform.API_VERSION != 3:
-        raise ExtensionModuleError
-
-
-class Manifest:
-
-    MANIFEST_ID = b'\0' * 32
-
-    def __init__(self, key, repository, item_keys=None):
-        self.archives = {}
-        self.config = {}
-        self.key = key
-        self.repository = repository
-        self.item_keys = frozenset(item_keys) if item_keys is not None else ITEM_KEYS
-
-    @property
-    def id_str(self):
-        return bin_to_hex(self.id)
-
-    @classmethod
-    def load(cls, repository, key=None):
-        from .key import key_factory
-        from .repository import Repository
-        try:
-            cdata = repository.get(cls.MANIFEST_ID)
-        except Repository.ObjectNotFound:
-            raise NoManifestError
-        if not key:
-            key = key_factory(repository, cdata)
-        manifest = cls(key, repository)
-        _, data = key.decrypt(None, cdata)
-        manifest.id = key.id_hash(data)
-        m = msgpack.unpackb(data)
-        if not m.get(b'version') == 1:
-            raise ValueError('Invalid manifest version')
-        manifest.archives = dict((k.decode('utf-8'), v) for k, v in m[b'archives'].items())
-        manifest.timestamp = m.get(b'timestamp')
-        if manifest.timestamp:
-            manifest.timestamp = manifest.timestamp.decode('ascii')
-        manifest.config = m[b'config']
-        # valid item keys are whatever is known in the repo or every key we know
-        manifest.item_keys = ITEM_KEYS | frozenset(key.decode() for key in m.get(b'item_keys', []))
-        return manifest, key
-
-    def write(self):
-        self.timestamp = datetime.utcnow().isoformat()
-        data = msgpack.packb(StableDict({
-            'version': 1,
-            'archives': self.archives,
-            'timestamp': self.timestamp,
-            'config': self.config,
-            'item_keys': tuple(self.item_keys),
-        }))
-        self.id = self.key.id_hash(data)
-        self.repository.put(self.MANIFEST_ID, self.key.encrypt(Chunk(data)))
-
-    def list_archive_infos(self, sort_by=None, reverse=False):
-        # inexpensive Archive.list_archives replacement if we just need .name, .id, .ts
-        ArchiveInfo = namedtuple('ArchiveInfo', 'name id ts')
-        archives = []
-        for name, values in self.archives.items():
-            ts = parse_timestamp(values[b'time'].decode('utf-8'))
-            id = values[b'id']
-            archives.append(ArchiveInfo(name=name, id=id, ts=ts))
-        if sort_by is not None:
-            archives = sorted(archives, key=attrgetter(sort_by), reverse=reverse)
-        return archives
-
-
-def prune_within(archives, within):
-    multiplier = {'H': 1, 'd': 24, 'w': 24 * 7, 'm': 24 * 31, 'y': 24 * 365}
-    try:
-        hours = int(within[:-1]) * multiplier[within[-1]]
-    except (KeyError, ValueError):
-        # I don't like how this displays the original exception too:
-        raise argparse.ArgumentTypeError('Unable to parse --within option: "%s"' % within)
-    if hours <= 0:
-        raise argparse.ArgumentTypeError('Number specified using --within option must be positive')
-    target = datetime.now(timezone.utc) - timedelta(seconds=hours * 3600)
-    return [a for a in archives if a.ts > target]
-
-
-def prune_split(archives, pattern, n, skip=[]):
-    last = None
-    keep = []
-    if n == 0:
-        return keep
-    for a in sorted(archives, key=attrgetter('ts'), reverse=True):
-        period = to_localtime(a.ts).strftime(pattern)
-        if period != last:
-            last = period
-            if a not in skip:
-                keep.append(a)
-                if len(keep) == n:
-                    break
-    return keep
-
-
-def get_home_dir():
-    """Get user's home directory while preferring a possibly set HOME
-    environment variable
-    """
-    # os.path.expanduser() behaves differently for '~' and '~someuser' as
-    # parameters: when called with an explicit username, the possibly set
-    # environment variable HOME is no longer respected. So we have to check if
-    # it is set and only expand the user's home directory if HOME is unset.
-    if os.environ.get('HOME', ''):
-        return os.environ.get('HOME')
-    else:
-        return os.path.expanduser('~%s' % os.environ.get('USER', ''))
-
-
-def get_keys_dir():
-    """Determine where to repository keys and cache"""
-
-    xdg_config = os.environ.get('XDG_CONFIG_HOME', os.path.join(get_home_dir(), '.config'))
-    keys_dir = os.environ.get('BORG_KEYS_DIR', os.path.join(xdg_config, 'borg', 'keys'))
-    if not os.path.exists(keys_dir):
-        os.makedirs(keys_dir)
-        os.chmod(keys_dir, stat.S_IRWXU)
-    return keys_dir
-
-
-def get_cache_dir():
-    """Determine where to repository keys and cache"""
-    xdg_cache = os.environ.get('XDG_CACHE_HOME', os.path.join(get_home_dir(), '.cache'))
-    cache_dir = os.environ.get('BORG_CACHE_DIR', os.path.join(xdg_cache, 'borg'))
-    if not os.path.exists(cache_dir):
-        os.makedirs(cache_dir)
-        os.chmod(cache_dir, stat.S_IRWXU)
-        with open(os.path.join(cache_dir, CACHE_TAG_NAME), 'wb') as fd:
-            fd.write(CACHE_TAG_CONTENTS)
-            fd.write(textwrap.dedent("""
-                # This file is a cache directory tag created by Borg.
-                # For information about cache directory tags, see:
-                #       http://www.brynosaurus.com/cachedir/
-                """).encode('ascii'))
-    return cache_dir
-
-
-def to_localtime(ts):
-    """Convert datetime object from UTC to local time zone"""
-    return datetime(*time.localtime((ts - datetime(1970, 1, 1, tzinfo=timezone.utc)).total_seconds())[:6])
-
-
-def parse_timestamp(timestamp):
-    """Parse a ISO 8601 timestamp string"""
-    if '.' in timestamp:  # microseconds might not be present
-        return datetime.strptime(timestamp, '%Y-%m-%dT%H:%M:%S.%f').replace(tzinfo=timezone.utc)
-    else:
-        return datetime.strptime(timestamp, '%Y-%m-%dT%H:%M:%S').replace(tzinfo=timezone.utc)
-
-
-def load_excludes(fh):
-    """Load and parse exclude patterns from file object. Lines empty or starting with '#' after stripping whitespace on
-    both line ends are ignored.
-    """
-    return [parse_pattern(pattern) for pattern in clean_lines(fh)]
-
-
-def update_excludes(args):
-    """Merge exclude patterns from files with those on command line."""
-    if hasattr(args, 'exclude_files') and args.exclude_files:
-        if not hasattr(args, 'excludes') or args.excludes is None:
-            args.excludes = []
-        for file in args.exclude_files:
-            args.excludes += load_excludes(file)
-            file.close()
-
-
-class PatternMatcher:
-    def __init__(self, fallback=None):
-        self._items = []
-
-        # Value to return from match function when none of the patterns match.
-        self.fallback = fallback
-
-    def empty(self):
-        return not len(self._items)
-
-    def add(self, patterns, value):
-        """Add list of patterns to internal list. The given value is returned from the match function when one of the
-        given patterns matches.
-        """
-        self._items.extend((i, value) for i in patterns)
-
-    def match(self, path):
-        for (pattern, value) in self._items:
-            if pattern.match(path):
-                return value
-
-        return self.fallback
-
-
-def normalized(func):
-    """ Decorator for the Pattern match methods, returning a wrapper that
-    normalizes OSX paths to match the normalized pattern on OSX, and
-    returning the original method on other platforms"""
-    @wraps(func)
-    def normalize_wrapper(self, path):
-        return func(self, unicodedata.normalize("NFD", path))
-
-    if sys.platform in ('darwin',):
-        # HFS+ converts paths to a canonical form, so users shouldn't be
-        # required to enter an exact match
-        return normalize_wrapper
-    else:
-        # Windows and Unix filesystems allow different forms, so users
-        # always have to enter an exact match
-        return func
-
-
-class PatternBase:
-    """Shared logic for inclusion/exclusion patterns.
-    """
-    PREFIX = NotImplemented
-
-    def __init__(self, pattern):
-        self.pattern_orig = pattern
-        self.match_count = 0
-
-        if sys.platform in ('darwin',):
-            pattern = unicodedata.normalize("NFD", pattern)
-
-        self._prepare(pattern)
-
-    @normalized
-    def match(self, path):
-        matches = self._match(path)
-
-        if matches:
-            self.match_count += 1
-
-        return matches
-
-    def __repr__(self):
-        return '%s(%s)' % (type(self), self.pattern)
-
-    def __str__(self):
-        return self.pattern_orig
-
-    def _prepare(self, pattern):
-        raise NotImplementedError
-
-    def _match(self, path):
-        raise NotImplementedError
-
-
-# For PathPrefixPattern, FnmatchPattern and ShellPattern, we require that the pattern either match the whole path
-# or an initial segment of the path up to but not including a path separator. To unify the two cases, we add a path
-# separator to the end of the path before matching.
-
-
-class PathPrefixPattern(PatternBase):
-    """Literal files or directories listed on the command line
-    for some operations (e.g. extract, but not create).
-    If a directory is specified, all paths that start with that
-    path match as well.  A trailing slash makes no difference.
-    """
-    PREFIX = "pp"
-
-    def _prepare(self, pattern):
-        if sys.platform != 'win32':
-            self.pattern = os.path.normpath(pattern).rstrip(os.path.sep) + os.path.sep
-        else:
-            self.pattern = posixpath.normpath(pattern).rstrip(posixpath.sep) + posixpath.sep
-
-    def _match(self, path):
-        if sys.platform != 'win32':
-            return (path + os.path.sep).startswith(self.pattern)
-        else:
-            return (path + posixpath.sep).startswith(self.pattern)
-
-
-class FnmatchPattern(PatternBase):
-    """Shell glob patterns to exclude.  A trailing slash means to
-    exclude the contents of a directory, but not the directory itself.
-    """
-    PREFIX = "fm"
-
-    def _prepare(self, pattern):
-        if sys.platform != 'win32':
-            if pattern.endswith(os.path.sep):
-                pattern = os.path.normpath(pattern).rstrip(os.path.sep) + os.path.sep + '*' + os.path.sep
-            else:
-                pattern = os.path.normpath(pattern) + os.path.sep + '*'
-        else:
-            if pattern.endswith(os.path.sep) or pattern.endswith(posixpath.sep):
-                pattern = posixpath.normpath(pattern).rstrip(posixpath.sep) + posixpath.sep + '*' + posixpath.sep
-            else:
-                pattern = posixpath.normpath(pattern) + posixpath.sep + '*'
-
-        self.pattern = pattern
-
-        # fnmatch and re.match both cache compiled regular expressions.
-        # Nevertheless, this is about 10 times faster.
-        self.regex = re.compile(translate(self.pattern))
-
-    def _match(self, path):
-        if sys.platform != 'win32':
-            return (self.regex.match(path + os.path.sep) is not None)
-        else:
-            return (self.regex.match(path.replace('\\', '/') + posixpath.sep) is not None)
-
-
-class ShellPattern(PatternBase):
-    """Shell glob patterns to exclude.  A trailing slash means to
-    exclude the contents of a directory, but not the directory itself.
-    """
-    PREFIX = "sh"
-
-    def _prepare(self, pattern):
-        sep = os.path.sep
-
-        if pattern.endswith(sep):
-            pattern = os.path.normpath(pattern).rstrip(sep) + sep + "**" + sep + "*" + sep
-        else:
-            pattern = os.path.normpath(pattern) + sep + "**" + sep + "*"
-
-        self.pattern = pattern
-        self.regex = re.compile(shellpattern.translate(self.pattern))
-
-    def _match(self, path):
-        return (self.regex.match(path + os.path.sep) is not None)
-
-
-class RegexPattern(PatternBase):
-    """Regular expression to exclude.
-    """
-    PREFIX = "re"
-
-    def _prepare(self, pattern):
-        self.pattern = pattern
-        self.regex = re.compile(pattern)
-
-    def _match(self, path):
-        # Normalize path separators
-        if os.path.sep != '/':
-            path = path.replace(os.path.sep, '/')
-
-        return (self.regex.search(path) is not None)
-
-
-_PATTERN_STYLES = set([
-    FnmatchPattern,
-    PathPrefixPattern,
-    RegexPattern,
-    ShellPattern,
-])
-
-_PATTERN_STYLE_BY_PREFIX = dict((i.PREFIX, i) for i in _PATTERN_STYLES)
-
-
-def parse_pattern(pattern, fallback=FnmatchPattern):
-    """Read pattern from string and return an instance of the appropriate implementation class.
-    """
-    if len(pattern) > 2 and pattern[2] == ":" and pattern[:2].isalnum():
-        (style, pattern) = (pattern[:2], pattern[3:])
-
-        cls = _PATTERN_STYLE_BY_PREFIX.get(style, None)
-
-        if cls is None:
-            raise ValueError("Unknown pattern style: {}".format(style))
-    else:
-        cls = fallback
-
-    return cls(pattern)
-
-
-def timestamp(s):
-    """Convert a --timestamp=s argument to a datetime object"""
-    try:
-        # is it pointing to a file / directory?
-        ts = os.stat(s).st_mtime
-        return datetime.utcfromtimestamp(ts)
-    except OSError:
-        # didn't work, try parsing as timestamp. UTC, no TZ, no microsecs support.
-        for format in ('%Y-%m-%dT%H:%M:%SZ', '%Y-%m-%dT%H:%M:%S+00:00',
-                       '%Y-%m-%dT%H:%M:%S', '%Y-%m-%d %H:%M:%S',
-                       '%Y-%m-%dT%H:%M', '%Y-%m-%d %H:%M',
-                       '%Y-%m-%d', '%Y-%j',
-                       ):
-            try:
-                return datetime.strptime(s, format)
-            except ValueError:
-                continue
-        raise ValueError
-
-
-def ChunkerParams(s):
-    if s.strip().lower() == "default":
-        return CHUNKER_PARAMS
-    chunk_min, chunk_max, chunk_mask, window_size = s.split(',')
-    if int(chunk_max) > 23:
-        # do not go beyond 2**23 (8MB) chunk size now,
-        # COMPR_BUFFER can only cope with up to this size
-        raise ValueError('max. chunk size exponent must not be more than 23 (2^23 = 8MiB max. chunk size)')
-    return int(chunk_min), int(chunk_max), int(chunk_mask), int(window_size)
-
-
-def CompressionSpec(s):
-    values = s.split(',')
-    count = len(values)
-    if count < 1:
-        raise ValueError
-    # --compression algo[,level]
-    name = values[0]
-    if name in ('none', 'lz4', ):
-        return dict(name=name)
-    if name in ('zlib', 'lzma', ):
-        if count < 2:
-            level = 6  # default compression level in py stdlib
-        elif count == 2:
-            level = int(values[1])
-            if not 0 <= level <= 9:
-                raise ValueError
-        else:
-            raise ValueError
-        return dict(name=name, level=level)
-    if name == 'auto':
-        if 2 <= count <= 3:
-            compression = ','.join(values[1:])
-        else:
-            raise ValueError
-        return dict(name=name, spec=CompressionSpec(compression))
-    raise ValueError
-
-
-def PrefixSpec(s):
-    return replace_placeholders(s)
-
-
-def dir_is_cachedir(path):
-    """Determines whether the specified path is a cache directory (and
-    therefore should potentially be excluded from the backup) according to
-    the CACHEDIR.TAG protocol
-    (http://www.brynosaurus.com/cachedir/spec.html).
-    """
-
-    tag_path = os.path.join(path, CACHE_TAG_NAME)
-    try:
-        if os.path.exists(tag_path):
-            with open(tag_path, 'rb') as tag_file:
-                tag_data = tag_file.read(len(CACHE_TAG_CONTENTS))
-                if tag_data == CACHE_TAG_CONTENTS:
-                    return True
-    except OSError:
-        pass
-    return False
-
-
-def dir_is_tagged(path, exclude_caches, exclude_if_present):
-    """Determines whether the specified path is excluded by being a cache
-    directory or containing user-specified tag files. Returns a list of the
-    paths of the tag files (either CACHEDIR.TAG or the matching
-    user-specified files).
-    """
-    tag_paths = []
-    if exclude_caches and dir_is_cachedir(path):
-        tag_paths.append(os.path.join(path, CACHE_TAG_NAME))
-    if exclude_if_present is not None:
-        for tag in exclude_if_present:
-            tag_path = os.path.join(path, tag)
-            if os.path.isfile(tag_path):
-                tag_paths.append(tag_path)
-    return tag_paths
-
-
-def partial_format(format, mapping):
-    """
-    Apply format.format_map(mapping) while preserving unknown keys
-
-    Does not support attribute access, indexing and ![rsa] conversions
-    """
-    for key, value in mapping.items():
-        key = re.escape(key)
-        format = re.sub(r'(?<!\{)((\{%s\})|(\{%s:[^\}]*\}))' % (key, key),
-                        lambda match: match.group(1).format_map(mapping),
-                        format)
-    return format
-
-
-def format_line(format, data):
-    try:
-        return format.format(**data)
-    except Exception as e:
-        raise PlaceholderError(format, data, e.__class__.__name__, str(e))
-
-
-def replace_placeholders(text):
-    """Replace placeholders in text with their values."""
-    current_time = datetime.now()
-    data = {
-        'pid': os.getpid(),
-        'fqdn': socket.getfqdn(),
-        'hostname': socket.gethostname(),
-        'now': current_time.now(),
-        'utcnow': current_time.utcnow(),
-        'user': getpass.getuser(),
-        'uuid4': str(uuid.uuid4()),
-    }
-    return format_line(text, data)
-
-
-def safe_timestamp(item_timestamp_ns):
-    try:
-        return datetime.fromtimestamp(bigint_to_int(item_timestamp_ns) / 1e9)
-    except OverflowError:
-        # likely a broken file time and datetime did not want to go beyond year 9999
-        return datetime(9999, 12, 31, 23, 59, 59)
-
-
-def format_time(t):
-    """use ISO-8601 date and time format
-    """
-    return t.strftime('%a, %Y-%m-%d %H:%M:%S')
-
-
-def format_timedelta(td):
-    """Format timedelta in a human friendly format
-    """
-    # Since td.total_seconds() requires python 2.7
-    ts = (td.microseconds + (td.seconds + td.days * 24 * 3600) * 10 ** 6) / float(10 ** 6)
-    s = ts % 60
-    m = int(ts / 60) % 60
-    h = int(ts / 3600) % 24
-    txt = '%.2f seconds' % s
-    if m:
-        txt = '%d minutes %s' % (m, txt)
-    if h:
-        txt = '%d hours %s' % (h, txt)
-    if td.days:
-        txt = '%d days %s' % (td.days, txt)
-    return txt
-
-
-def format_file_size(v, precision=2, sign=False):
-    """Format file size into a human friendly format
-    """
-    return sizeof_fmt_decimal(v, suffix='B', sep=' ', precision=precision, sign=sign)
-
-
-def sizeof_fmt(num, suffix='B', units=None, power=None, sep='', precision=2, sign=False):
-    prefix = '+' if sign and num > 0 else ''
-
-    for unit in units[:-1]:
-        if abs(round(num, precision)) < power:
-            if isinstance(num, int):
-                return "{}{}{}{}{}".format(prefix, num, sep, unit, suffix)
-            else:
-                return "{}{:3.{}f}{}{}{}".format(prefix, num, precision, sep, unit, suffix)
-        num /= float(power)
-    return "{}{:.{}f}{}{}{}".format(prefix, num, precision, sep, units[-1], suffix)
-
-
-def sizeof_fmt_iec(num, suffix='B', sep='', precision=2, sign=False):
-    return sizeof_fmt(num, suffix=suffix, sep=sep, precision=precision, sign=sign,
-                      units=['', 'Ki', 'Mi', 'Gi', 'Ti', 'Pi', 'Ei', 'Zi', 'Yi'], power=1024)
-
-
-def sizeof_fmt_decimal(num, suffix='B', sep='', precision=2, sign=False):
-    return sizeof_fmt(num, suffix=suffix, sep=sep, precision=precision, sign=sign,
-                      units=['', 'k', 'M', 'G', 'T', 'P', 'E', 'Z', 'Y'], power=1000)
-
-
-def format_archive(archive):
-    return '%-36s %s [%s]' % (
-        archive.name,
-        format_time(to_localtime(archive.ts)),
-        bin_to_hex(archive.id),
-    )
-
-
-def memoize(function):
-    cache = {}
-
-    def decorated_function(*args):
-        try:
-            return cache[args]
-        except KeyError:
-            val = function(*args)
-            cache[args] = val
-            return val
-    return decorated_function
-
-
-@memoize
-def uid2user(uid, default=None):
-    try:
-        if sys.platform != 'win32':
-            return pwd.getpwuid(uid).pw_name
-        else:
-            return os.getlogin()
-    except KeyError:
-        return default
-
-
-@memoize
-def user2uid(user, default=None):
-    try:
-        if sys.platform != 'win32':
-            return user and pwd.getpwnam(user).pw_uid
-        else:
-            return user and 0
-    except KeyError:
-        return default
-
-
-@memoize
-def gid2group(gid, default=None):
-    try:
-        if sys.platform != 'win32':
-            return grp.getgrgid(gid).gr_name
-        else:
-            return ''
-    except KeyError:
-        return default
-
-
-@memoize
-def group2gid(group, default=None):
-    if sys.platform != 'win32':
-        if group == '':
-            return 0  # From windows
-        try:
-            return group and grp.getgrnam(group).gr_gid
-        except KeyError:
-            return default
-    else:
-        return 0
-
-
-def getuid():
-    if sys.platform != 'win32':
-        return os.getuid()
-    else:
-        return 0
-
-
-def posix_acl_use_stored_uid_gid(acl):
-    """Replace the user/group field with the stored uid/gid
-    """
-    entries = []
-    for entry in safe_decode(acl).split('\n'):
-        if entry:
-            fields = entry.split(':')
-            if len(fields) == 4:
-                entries.append(':'.join([fields[0], fields[3], fields[2]]))
-            else:
-                entries.append(entry)
-    return safe_encode('\n'.join(entries))
-
-
-def safe_decode(s, coding='utf-8', errors='surrogateescape'):
-    """decode bytes to str, with round-tripping "invalid" bytes"""
-    if s is None:
-        return None
-    return s.decode(coding, errors)
-
-
-def safe_encode(s, coding='utf-8', errors='surrogateescape'):
-    """encode str to bytes, with round-tripping "invalid" bytes"""
-    if s is None:
-        return None
-    return s.encode(coding, errors)
-
-
-def bin_to_hex(binary):
-    return hexlify(binary).decode('ascii')
-
-
-class Location:
-    """Object representing a repository / archive location
-    """
-    proto = user = host = port = path = archive = None
-    # borg mount's FUSE filesystem creates one level of directories from
-    # the archive names. Thus, we must not accept "/" in archive names.
-    ssh_re = re.compile(r'(?P<proto>ssh)://(?:(?P<user>[^@]+)@)?'
-                        r'(?P<host>[^:/#]+)(?::(?P<port>\d+))?'
-                        r'(?P<path>[^:]+)(?:::(?P<archive>[^/]+))?$')
-    file_re = None
-    if sys.platform != 'win32':
-        file_re = re.compile(r'(?P<proto>file)://'
-                            r'(?P<path>[^:]+)(?:::(?P<archive>[^/]+))?$')
-    else:
-        file_re = re.compile(r'((?P<proto>file)://)?'
-                            r'(?P<drive>[a-zA-Z])?:[\\/](?P<path>[^:]+)(?:::(?P<archive>[^/]+))?$')
-    scp_re = re.compile(r'((?:(?P<user>[^@]+)@)?(?P<host>[^:/]+):)?'
-                        r'(?P<path>[^:]+)(?:::(?P<archive>[^/]+))?$')
-    # get the repo from BORG_RE env and the optional archive from param.
-    # if the syntax requires giving REPOSITORY (see "borg mount"),
-    # use "::" to let it use the env var.
-    # if REPOSITORY argument is optional, it'll automatically use the env.
-    env_re = re.compile(r'(?:::(?P<archive>[^/]+)?)?$')
-
-    def __init__(self, text=''):
-        self.orig = text
-        if not self.parse(self.orig):
-            raise ValueError
-
-    def parse(self, text):
-        text = replace_placeholders(text)
-        valid = self._parse(text)
-        if valid:
-            return True
-        m = self.env_re.match(text)
-        if not m:
-            return False
-        repo = os.environ.get('BORG_REPO')
-        if repo is None:
-            return False
-        valid = self._parse(repo)
-        if not valid:
-            return False
-        self.archive = m.group('archive')
-        return True
-
-    def _parse(self, text):
-        if sys.platform == 'win32':
-            m = self.file_re.match(text)
-            if m:
-                self.proto = 'file'
-                self.path = posixpath.normpath(m.group('drive') + ":\\" + m.group('path'))
-                self.archive = m.group('archive')
-                return True
-
-        m = self.ssh_re.match(text)
-        if m:
-            self.proto = m.group('proto')
-            self.user = m.group('user')
-            self.host = m.group('host')
-            self.port = m.group('port') and int(m.group('port')) or None
-            if sys.platform != 'win32':
-                self.path = os.path.normpath(m.group('path'))
-            else:
-                self.path = posixpath.normpath(m.group('path'))
-            self.archive = m.group('archive')
-            return True
-        if sys.platform != 'win32':
-            m = self.file_re.match(text)
-            if m:
-                self.proto = m.group('proto')
-                self.path = os.path.normpath(m.group('path'))
-                self.archive = m.group('archive')
-                return True
-        m = self.scp_re.match(text)
-        if m:
-            self.user = m.group('user')
-            self.host = m.group('host')
-            if sys.platform != 'win32':
-                self.path = os.path.normpath(m.group('path'))
-            else:
-                self.path = posixpath.normpath(m.group('path'))
-            self.archive = m.group('archive')
-            self.proto = self.host and 'ssh' or 'file'
-            return True
-        return False
-
-    def __str__(self):
-        items = [
-            'proto=%r' % self.proto,
-            'user=%r' % self.user,
-            'host=%r' % self.host,
-            'port=%r' % self.port,
-            'path=%r' % self.path,
-            'archive=%r' % self.archive,
-        ]
-        return ', '.join(items)
-
-    def to_key_filename(self):
-        name = re.sub('[^\w]', '_', self.path).strip('_')
-        if self.proto != 'file':
-            name = self.host + '__' + name
-        return os.path.join(get_keys_dir(), name)
-
-    def __repr__(self):
-        return "Location(%s)" % self
-
-    def canonical_path(self):
-        if self.proto == 'file':
-            return self.path
-        else:
-            if self.path and self.path.startswith('~'):
-                path = '/' + self.path
-            elif self.path and not self.path.startswith('/'):
-                path = '/~/' + self.path
-            else:
-                path = self.path
-            return 'ssh://{}{}{}{}'.format('{}@'.format(self.user) if self.user else '',
-                                           self.host,
-                                           ':{}'.format(self.port) if self.port else '',
-                                           path)
-
-
-def location_validator(archive=None):
-    def validator(text):
-        try:
-            loc = Location(text)
-        except ValueError:
-            raise argparse.ArgumentTypeError('Invalid location format: "%s"' % text) from None
-        if archive is True and not loc.archive:
-            raise argparse.ArgumentTypeError('"%s": No archive specified' % text)
-        elif archive is False and loc.archive:
-            raise argparse.ArgumentTypeError('"%s" No archive can be specified' % text)
-        return loc
-    return validator
-
-
-def archivename_validator():
-    def validator(text):
-        if '/' in text or '::' in text or not text:
-            raise argparse.ArgumentTypeError('Invalid repository name: "%s"' % text)
-        return text
-    return validator
-
-
-def decode_dict(d, keys, encoding='utf-8', errors='surrogateescape'):
-    for key in keys:
-        if isinstance(d.get(key), bytes):
-            d[key] = d[key].decode(encoding, errors)
-    return d
-
-
-def remove_surrogates(s, errors='replace'):
-    """Replace surrogates generated by fsdecode with '?'
-    """
-    return s.encode('utf-8', errors).decode('utf-8')
-
-_safe_re = None
-if sys.platform != 'win32':
-    _safe_re = re.compile(r'^((\.\.)?/+)+')
-else:
-    _safe_re = re.compile(r'^((\.\.)?[/\\]+)+')
-
-
-def make_path_safe(path):
-    """Make path safe by making it relative and local
-    """
-    if sys.platform != 'win32':
-        return _safe_re.sub('', path) or '.'
-    else:
-        tail = path
-        if len(path) > 2 and (path[0:2] == '//' or path[0:2] == '\\\\' or path[1] == ':'):
-            drive, tail = os.path.splitdrive(path)
-        tail = tail.replace('\\', '/')
-        return posixpath.normpath(_safe_re.sub('', tail) or '.')
-
-
-def daemonize():
-    """Detach process from controlling terminal and run in background
-    """
-    pid = os.fork()
-    if pid:
-        os._exit(0)
-    os.setsid()
-    pid = os.fork()
-    if pid:
-        os._exit(0)
-    os.chdir('/')
-    os.close(0)
-    os.close(1)
-    os.close(2)
-    fd = os.open('/dev/null', os.O_RDWR)
-    os.dup2(fd, 0)
-    os.dup2(fd, 1)
-    os.dup2(fd, 2)
-
-
-class StableDict(dict):
-    """A dict subclass with stable items() ordering"""
-    def items(self):
-        return sorted(super().items())
-
-
-def bigint_to_int(mtime):
-    """Convert bytearray to int
-    """
-    if isinstance(mtime, bytes):
-        return int.from_bytes(mtime, 'little', signed=True)
-    return mtime
-
-
-def int_to_bigint(value):
-    """Convert integers larger than 64 bits to bytearray
-
-    Smaller integers are left alone
-    """
-    if value.bit_length() > 63:
-        return value.to_bytes((value.bit_length() + 9) // 8, 'little', signed=True)
-    return value
-
-
-def is_slow_msgpack():
-    return msgpack.Packer is msgpack.fallback.Packer
-
-
-FALSISH = ('No', 'NO', 'no', 'N', 'n', '0', )
-TRUISH = ('Yes', 'YES', 'yes', 'Y', 'y', '1', )
-DEFAULTISH = ('Default', 'DEFAULT', 'default', 'D', 'd', '', )
-
-
-def yes(msg=None, false_msg=None, true_msg=None, default_msg=None,
-        retry_msg=None, invalid_msg=None, env_msg=None,
-        falsish=FALSISH, truish=TRUISH, defaultish=DEFAULTISH,
-        default=False, retry=True, env_var_override=None, ofile=None, input=input):
-    """Output <msg> (usually a question) and let user input an answer.
-    Qualifies the answer according to falsish, truish and defaultish as True, False or <default>.
-    If it didn't qualify and retry_msg is None (no retries wanted),
-    return the default [which defaults to False]. Otherwise let user retry
-    answering until answer is qualified.
-
-    If env_var_override is given and this var is present in the environment, do not ask
-    the user, but just use the env var contents as answer as if it was typed in.
-    Otherwise read input from stdin and proceed as normal.
-    If EOF is received instead an input or an invalid input without retry possibility,
-    return default.
-
-    :param msg: introducing message to output on ofile, no \n is added [None]
-    :param retry_msg: retry message to output on ofile, no \n is added [None]
-    :param false_msg: message to output before returning False [None]
-    :param true_msg: message to output before returning True [None]
-    :param default_msg: message to output before returning a <default> [None]
-    :param invalid_msg: message to output after a invalid answer was given [None]
-    :param env_msg: message to output when using input from env_var_override [None],
-           needs to have 2 placeholders for answer and env var name, e.g.: "{} (from {})"
-    :param falsish: sequence of answers qualifying as False
-    :param truish: sequence of answers qualifying as True
-    :param defaultish: sequence of answers qualifying as <default>
-    :param default: default return value (defaultish answer was given or no-answer condition) [False]
-    :param retry: if True and input is incorrect, retry. Otherwise return default. [True]
-    :param env_var_override: environment variable name [None]
-    :param ofile: output stream [sys.stderr]
-    :param input: input function [input from builtins]
-    :return: boolean answer value, True or False
-    """
-    # note: we do not assign sys.stderr as default above, so it is
-    # really evaluated NOW,  not at function definition time.
-    if ofile is None:
-        ofile = sys.stderr
-    if default not in (True, False):
-        raise ValueError("invalid default value, must be True or False")
-    if msg:
-        print(msg, file=ofile, end='', flush=True)
-    while True:
-        answer = None
-        if env_var_override:
-            answer = os.environ.get(env_var_override)
-            if answer is not None and env_msg:
-                print(env_msg.format(answer, env_var_override), file=ofile)
-        if answer is None:
-            try:
-                answer = input()
-            except EOFError:
-                # avoid defaultish[0], defaultish could be empty
-                answer = truish[0] if default else falsish[0]
-        if answer in defaultish:
-            if default_msg:
-                print(default_msg, file=ofile)
-            return default
-        if answer in truish:
-            if true_msg:
-                print(true_msg, file=ofile)
-            return True
-        if answer in falsish:
-            if false_msg:
-                print(false_msg, file=ofile)
-            return False
-        # if we get here, the answer was invalid
-        if invalid_msg:
-            print(invalid_msg, file=ofile)
-        if not retry:
-            return default
-        if retry_msg:
-            print(retry_msg, file=ofile, end='', flush=True)
-        # in case we used an environment variable and it gave an invalid answer, do not use it again:
-        env_var_override = None
-
-
-class ProgressIndicatorPercent:
-    def __init__(self, total, step=5, start=0, same_line=False, msg="%3.0f%%"):
-        """
-        Percentage-based progress indicator
-
-        :param total: total amount of items
-        :param step: step size in percent
-        :param start: at which percent value to start
-        :param same_line: if True, emit output always on same line
-        :param msg: output message, must contain one %f placeholder for the percentage
-        """
-        self.counter = 0  # 0 .. (total-1)
-        self.total = total
-        self.trigger_at = start  # output next percentage value when reaching (at least) this
-        self.step = step
-        self.msg = msg
-        self.same_line = same_line
-        self.handler = None
-        self.logger = logging.getLogger('borg.output.progress')
-
-        # If there are no handlers, set one up explicitly because the
-        # terminator and propagation needs to be set.  If there are,
-        # they must have been set up by BORG_LOGGING_CONF: skip setup.
-        if not self.logger.handlers:
-            self.handler = logging.StreamHandler(stream=sys.stderr)
-            self.handler.setLevel(logging.INFO)
-            self.handler.terminator = '\r' if self.same_line else '\n'
-
-            self.logger.addHandler(self.handler)
-            if self.logger.level == logging.NOTSET:
-                self.logger.setLevel(logging.WARN)
-            self.logger.propagate = False
-
-    def __del__(self):
-        if self.handler is not None:
-            self.logger.removeHandler(self.handler)
-            self.handler.close()
-
-    def progress(self, current=None):
-        if current is not None:
-            self.counter = current
-        pct = self.counter * 100 / self.total
-        self.counter += 1
-        if pct >= self.trigger_at:
-            self.trigger_at += self.step
-            return pct
-
-    def show(self, current=None):
-        pct = self.progress(current)
-        if pct is not None:
-            return self.output(pct)
-
-    def output(self, percent):
-        self.logger.info(self.msg % percent)
-
-    def finish(self):
-        if self.same_line:
-            self.logger.info(" " * len(self.msg % 100.0))
-
-
-class ProgressIndicatorEndless:
-    def __init__(self, step=10, file=None):
-        """
-        Progress indicator (long row of dots)
-
-        :param step: every Nth call, call the func
-        :param file: output file, default: sys.stderr
-        """
-        self.counter = 0  # call counter
-        self.triggered = 0  # increases 1 per trigger event
-        self.step = step  # trigger every <step> calls
-        if file is None:
-            file = sys.stderr
-        self.file = file
-
-    def progress(self):
-        self.counter += 1
-        trigger = self.counter % self.step == 0
-        if trigger:
-            self.triggered += 1
-        return trigger
-
-    def show(self):
-        trigger = self.progress()
-        if trigger:
-            return self.output(self.triggered)
-
-    def output(self, triggered):
-        print('.', end='', file=self.file, flush=True)
-
-    def finish(self):
-        print(file=self.file)
-
-
-def sysinfo():
-    info = []
-    info.append('Platform: %s' % (' '.join(platform.uname()), ))
-    if sys.platform.startswith('linux'):
-        info.append('Linux: %s %s %s' % platform.linux_distribution())
-    info.append('Borg: %s  Python: %s %s' % (borg_version, platform.python_implementation(), platform.python_version()))
-    info.append('PID: %d  CWD: %s' % (os.getpid(), os.getcwd()))
-    info.append('sys.argv: %r' % sys.argv)
-    info.append('SSH_ORIGINAL_COMMAND: %r' % os.environ.get('SSH_ORIGINAL_COMMAND'))
-    info.append('')
-    return '\n'.join(info)
-
-
-def log_multi(*msgs, level=logging.INFO, logger=logger):
-    """
-    log multiple lines of text, each line by a separate logging call for cosmetic reasons
-
-    each positional argument may be a single or multiple lines (separated by newlines) of text.
-    """
-    lines = []
-    for msg in msgs:
-        lines.extend(msg.splitlines())
-    for line in lines:
-        logger.log(level, line)
-
-
-class BaseFormatter:
-    FIXED_KEYS = {
-        # Formatting aids
-        'LF': '\n',
-        'SPACE': ' ',
-        'TAB': '\t',
-        'CR': '\r',
-        'NUL': '\0',
-        'NEWLINE': os.linesep,
-        'NL': os.linesep,
-    }
-
-    def get_item_data(self, item):
-        raise NotImplementedError
-
-    def format_item(self, item):
-        return self.format.format_map(self.get_item_data(item))
-
-    @staticmethod
-    def keys_help():
-        return " - NEWLINE: OS dependent line separator\n" \
-               " - NL: alias of NEWLINE\n" \
-               " - NUL: NUL character for creating print0 / xargs -0 like output, see barchive/bpath\n" \
-               " - SPACE\n" \
-               " - TAB\n" \
-               " - CR\n" \
-               " - LF"
-
-
-class ArchiveFormatter(BaseFormatter):
-
-    def __init__(self, format):
-        self.format = partial_format(format, self.FIXED_KEYS)
-
-    def get_item_data(self, archive):
-        return {
-            'barchive': archive.name,
-            'archive': remove_surrogates(archive.name),
-            'id': bin_to_hex(archive.id),
-            'time': format_time(to_localtime(archive.ts)),
-        }
-
-    @staticmethod
-    def keys_help():
-        return " - archive: archive name interpreted as text (might be missing non-text characters, see barchive)\n" \
-               " - barchive: verbatim archive name, can contain any character except NUL\n" \
-               " - time: time of creation of the archive\n" \
-               " - id: internal ID of the archive"
-
-
-class ItemFormatter(BaseFormatter):
-    KEY_DESCRIPTIONS = {
-        'bpath': 'verbatim POSIX path, can contain any character except NUL',
-        'path': 'path interpreted as text (might be missing non-text characters, see bpath)',
-        'source': 'link target for links (identical to linktarget)',
-        'extra': 'prepends {source} with " -> " for soft links and " link to " for hard links',
-        'csize': 'compressed size',
-        'num_chunks': 'number of chunks in this file',
-        'unique_chunks': 'number of unique chunks in this file',
-    }
-    KEY_GROUPS = (
-        ('type', 'mode', 'uid', 'gid', 'user', 'group', 'path', 'bpath', 'source', 'linktarget', 'flags'),
-        ('size', 'csize', 'num_chunks', 'unique_chunks'),
-        ('mtime', 'ctime', 'atime', 'isomtime', 'isoctime', 'isoatime'),
-        tuple(sorted(hashlib.algorithms_guaranteed)),
-        ('archiveid', 'archivename', 'extra'),
-    )
-
-    @classmethod
-    def available_keys(cls):
-        class FakeArchive:
-            fpr = name = ""
-
-        from .item import Item
-        fake_item = Item(mode=0, path='', user='', group='', mtime=0, uid=0, gid=0)
-        formatter = cls(FakeArchive, "")
-        keys = []
-        keys.extend(formatter.call_keys.keys())
-        keys.extend(formatter.get_item_data(fake_item).keys())
-        return keys
-
-    @classmethod
-    def keys_help(cls):
-        help = []
-        keys = cls.available_keys()
-        for key in cls.FIXED_KEYS:
-            keys.remove(key)
-
-        for group in cls.KEY_GROUPS:
-            for key in group:
-                keys.remove(key)
-                text = " - " + key
-                if key in cls.KEY_DESCRIPTIONS:
-                    text += ": " + cls.KEY_DESCRIPTIONS[key]
-                help.append(text)
-            help.append("")
-        assert not keys, str(keys)
-        return "\n".join(help)
-
-    def __init__(self, archive, format):
-        self.archive = archive
-        static_keys = {
-            'archivename': archive.name,
-            'archiveid': archive.fpr,
-        }
-        static_keys.update(self.FIXED_KEYS)
-        self.format = partial_format(format, static_keys)
-        self.format_keys = {f[1] for f in Formatter().parse(format)}
-        self.call_keys = {
-            'size': self.calculate_size,
-            'csize': self.calculate_csize,
-            'num_chunks': self.calculate_num_chunks,
-            'unique_chunks': self.calculate_unique_chunks,
-            'isomtime': partial(self.format_time, 'mtime'),
-            'isoctime': partial(self.format_time, 'ctime'),
-            'isoatime': partial(self.format_time, 'atime'),
-            'mtime': partial(self.time, 'mtime'),
-            'ctime': partial(self.time, 'ctime'),
-            'atime': partial(self.time, 'atime'),
-        }
-        for hash_function in hashlib.algorithms_guaranteed:
-            self.add_key(hash_function, partial(self.hash_item, hash_function))
-        self.used_call_keys = set(self.call_keys) & self.format_keys
-        self.item_data = static_keys
-
-    def add_key(self, key, callable_with_item):
-        self.call_keys[key] = callable_with_item
-        self.used_call_keys = set(self.call_keys) & self.format_keys
-
-    def get_item_data(self, item):
-        mode = stat.filemode(item.mode)
-        item_type = mode[0]
-        item_data = self.item_data
-
-        source = item.get('source', '')
-        extra = ''
-        if source:
-            source = remove_surrogates(source)
-            if item_type == 'l':
-                extra = ' -> %s' % source
-            else:
-                mode = 'h' + mode[1:]
-                extra = ' link to %s' % source
-        item_data['type'] = item_type
-        item_data['mode'] = mode
-        item_data['user'] = item.user or item.uid
-        item_data['group'] = item.group or item.gid
-        item_data['uid'] = item.uid
-        item_data['gid'] = item.gid
-        item_data['path'] = remove_surrogates(item.path)
-        item_data['bpath'] = item.path
-        item_data['source'] = source
-        item_data['linktarget'] = source
-        item_data['extra'] = extra
-        item_data['flags'] = item.get('bsdflags')
-        for key in self.used_call_keys:
-            item_data[key] = self.call_keys[key](item)
-        return item_data
-
-    def calculate_num_chunks(self, item):
-        return len(item.get('chunks', []))
-
-    def calculate_unique_chunks(self, item):
-        chunk_index = self.archive.cache.chunks
-        return sum(1 for c in item.get('chunks', []) if chunk_index[c.id].refcount == 1)
-
-    def calculate_size(self, item):
-        return sum(c.size for c in item.get('chunks', []))
-
-    def calculate_csize(self, item):
-        return sum(c.csize for c in item.get('chunks', []))
-
-    def hash_item(self, hash_function, item):
-        if 'chunks' not in item:
-            return ""
-        hash = hashlib.new(hash_function)
-        for _, data in self.archive.pipeline.fetch_many([c.id for c in item.chunks]):
-            hash.update(data)
-        return hash.hexdigest()
-
-    def format_time(self, key, item):
-        return format_time(safe_timestamp(item.get(key) or item.mtime))
-
-    def time(self, key, item):
-        return safe_timestamp(item.get(key) or item.mtime)
-
-
-class ChunkIteratorFileWrapper:
-    """File-like wrapper for chunk iterators"""
-
-    def __init__(self, chunk_iterator):
-        self.chunk_iterator = chunk_iterator
-        self.chunk_offset = 0
-        self.chunk = b''
-        self.exhausted = False
-
-    def _refill(self):
-        remaining = len(self.chunk) - self.chunk_offset
-        if not remaining:
-            try:
-                chunk = next(self.chunk_iterator)
-                self.chunk = memoryview(chunk.data)
-            except StopIteration:
-                self.exhausted = True
-                return 0  # EOF
-            self.chunk_offset = 0
-            remaining = len(self.chunk)
-        return remaining
-
-    def _read(self, nbytes):
-        if not nbytes:
-            return b''
-        remaining = self._refill()
-        will_read = min(remaining, nbytes)
-        self.chunk_offset += will_read
-        return self.chunk[self.chunk_offset - will_read:self.chunk_offset]
-
-    def read(self, nbytes):
-        parts = []
-        while nbytes and not self.exhausted:
-            read_data = self._read(nbytes)
-            nbytes -= len(read_data)
-            parts.append(read_data)
-        return b''.join(parts)
-
-
-def open_item(archive, item):
-    """Return file-like object for archived item (with chunks)."""
-    chunk_iterator = archive.pipeline.fetch_many([c.id for c in item.chunks])
-    return ChunkIteratorFileWrapper(chunk_iterator)
-
-
-def file_status(mode):
-    if stat.S_ISREG(mode):
-        return 'A'
-    elif stat.S_ISDIR(mode):
-        return 'd'
-    elif stat.S_ISBLK(mode):
-        return 'b'
-    elif stat.S_ISCHR(mode):
-        return 'c'
-    elif stat.S_ISLNK(mode):
-        return 's'
-    elif stat.S_ISFIFO(mode):
-        return 'f'
-    return '?'
-
-
-def consume(iterator, n=None):
-    """Advance the iterator n-steps ahead. If n is none, consume entirely."""
-    # Use functions that consume iterators at C speed.
-    if n is None:
-        # feed the entire iterator into a zero-length deque
-        deque(iterator, maxlen=0)
-    else:
-        # advance to the empty slice starting at position n
-        next(islice(iterator, n, n), None)
-
-# GenericDirEntry, scandir_generic (c) 2012 Ben Hoyt
-# from the python-scandir package (3-clause BSD license, just like us, so no troubles here)
-# note: simplified version
-
-
-class GenericDirEntry:
-    __slots__ = ('name', '_scandir_path', '_path')
-
-    def __init__(self, scandir_path, name):
-        self._scandir_path = scandir_path
-        self.name = name
-        self._path = None
-
-    @property
-    def path(self):
-        if self._path is None:
-            self._path = os.path.join(self._scandir_path, self.name)
-        return self._path
-
-    def stat(self, follow_symlinks=True):
-        assert not follow_symlinks
-        return os.lstat(self.path)
-
-    def _check_type(self, type):
-        st = self.stat(False)
-        return stat.S_IFMT(st.st_mode) == type
-
-    def is_dir(self, follow_symlinks=True):
-        assert not follow_symlinks
-        return self._check_type(stat.S_IFDIR)
-
-    def is_file(self, follow_symlinks=True):
-        assert not follow_symlinks
-        return self._check_type(stat.S_IFREG)
-
-    def is_symlink(self):
-        return self._check_type(stat.S_IFLNK)
-
-    def inode(self):
-        st = self.stat(False)
-        return st.st_ino
-
-    def __repr__(self):
-        return '<{0}: {1!r}>'.format(self.__class__.__name__, self.path)
-
-
-def scandir_generic(path='.'):
-    """Like os.listdir(), but yield DirEntry objects instead of returning a list of names."""
-    for name in sorted(os.listdir(path)):
-        yield GenericDirEntry(path, name)
-
-try:
-    from os import scandir
-except ImportError:
-    try:
-        # Try python-scandir on Python 3.4
-        from scandir import scandir
-    except ImportError:
-        # If python-scandir is not installed, then use a version that is just as slow as listdir.
-        scandir = scandir_generic
-
-
-def scandir_inorder(path='.'):
-    return sorted(scandir(path), key=lambda dirent: dirent.inode())
-
-
-def clean_lines(lines, lstrip=None, rstrip=None, remove_empty=True, remove_comments=True):
-    """
-    clean lines (usually read from a config file):
-
-    1. strip whitespace (left and right), 2. remove empty lines, 3. remove comments.
-
-    note: only "pure comment lines" are supported, no support for "trailing comments".
-
-    :param lines: input line iterator (e.g. list or open text file) that gives unclean input lines
-    :param lstrip: lstrip call arguments or False, if lstripping is not desired
-    :param rstrip: rstrip call arguments or False, if rstripping is not desired
-    :param remove_comments: remove comment lines (lines starting with "#")
-    :param remove_empty: remove empty lines
-    :return: yields processed lines
-    """
-    for line in lines:
-        if lstrip is not False:
-            line = line.lstrip(lstrip)
-        if rstrip is not False:
-            line = line.rstrip(rstrip)
-        if remove_empty and not line:
-            continue
-        if remove_comments and line.startswith('#'):
-            continue
-        yield line
-
-
-class CompressionDecider1:
-    def __init__(self, compression, compression_files):
-        """
-        Initialize a CompressionDecider instance (and read config files, if needed).
-
-        :param compression: default CompressionSpec (e.g. from --compression option)
-        :param compression_files: list of compression config files (e.g. from --compression-from) or
-                                  a list of other line iterators
-        """
-        self.compression = compression
-        if not compression_files:
-            self.matcher = None
-        else:
-            self.matcher = PatternMatcher(fallback=compression)
-            for file in compression_files:
-                try:
-                    for line in clean_lines(file):
-                        try:
-                            compr_spec, fn_pattern = line.split(':', 1)
-                        except:
-                            continue
-                        self.matcher.add([parse_pattern(fn_pattern)], CompressionSpec(compr_spec))
-                finally:
-                    if hasattr(file, 'close'):
-                        file.close()
-
-    def decide(self, path):
-        if self.matcher is not None:
-            return self.matcher.match(path)
-        return self.compression
-
-
-class CompressionDecider2:
-    def __init__(self, compression):
-        self.compression = compression
-
-    def decide(self, chunk):
-        # nothing fancy here yet: we either use what the metadata says or the default
-        # later, we can decide based on the chunk data also.
-        # if we compress the data here to decide, we can even update the chunk data
-        # and modify the metadata as desired.
-        compr_spec = chunk.meta.get('compress', self.compression)
-        compr_args = dict(buffer=COMPR_BUFFER)
-        compr_args.update(compr_spec)
-        if compr_args['name'] == 'auto':
-            # we did not decide yet, use heuristic:
-            compr_args, chunk = self.heuristic_lz4(compr_args, chunk)
-        return compr_args, chunk
-
-    def heuristic_lz4(self, compr_args, chunk):
-        meta, data = chunk
-        lz4 = get_compressor('lz4', buffer=compr_args['buffer'])
-        cdata = lz4.compress(data)
-        data_len = len(data)
-        cdata_len = len(cdata)
-        if cdata_len < data_len:
-            compr_spec = compr_args['spec']
-        else:
-            # uncompressible - we could have a special "uncompressible compressor"
-            # that marks such data as uncompressible via compression-type metadata.
-            compr_spec = CompressionSpec('none')
-        compr_args.update(compr_spec)
-        logger.debug("len(data) == %d, len(lz4(data)) == %d, choosing %s", data_len, cdata_len, compr_spec)
-        return compr_args, Chunk(data, **meta)
-
-
-@contextmanager
-def signal_handler(signo, handler):
-    old_signal_handler = signal.signal(signo, handler)
-    try:
-        yield
-    finally:
-        signal.signal(signo, old_signal_handler)

+ 4 - 0
src/borg/helpers/checks.py

@@ -1,4 +1,5 @@
 import os
+import sys
 
 from .errors import Error
 
@@ -8,6 +9,9 @@ class PythonLibcTooOld(Error):
 
 
 def check_python():
+    if sys.platform == 'win32':
+        # Does not support symlinks anyway
+        return
     required_funcs = {os.stat, os.utime, os.chown}
     if not os.supports_follow_symlinks.issuperset(required_funcs):
         raise PythonLibcTooOld

+ 29 - 13
src/borg/helpers/parseformat.py

@@ -2,6 +2,7 @@ import argparse
 import hashlib
 import json
 import os
+import sys
 import os.path
 import re
 import socket
@@ -180,19 +181,34 @@ def format_line(format, data):
 def replace_placeholders(text):
     """Replace placeholders in text with their values."""
     current_time = datetime.now()
-    data = {
-        'pid': os.getpid(),
-        'fqdn': socket.getfqdn(),
-        'hostname': socket.gethostname(),
-        'now': DatetimeWrapper(current_time.now()),
-        'utcnow': DatetimeWrapper(current_time.utcnow()),
-        'user': uid2user(os.getuid(), os.getuid()),
-        'uuid4': str(uuid.uuid4()),
-        'borgversion': borg_version,
-        'borgmajor': '%d' % borg_version_tuple[:1],
-        'borgminor': '%d.%d' % borg_version_tuple[:2],
-        'borgpatch': '%d.%d.%d' % borg_version_tuple[:3],
-    }
+    if sys.platform != 'win32':
+        data = {
+            'pid': os.getpid(),
+            'fqdn': socket.getfqdn(),
+            'hostname': socket.gethostname(),
+            'now': DatetimeWrapper(current_time.now()),
+            'utcnow': DatetimeWrapper(current_time.utcnow()),
+            'user': uid2user(os.getuid(), os.getuid()),
+            'uuid4': str(uuid.uuid4()),
+            'borgversion': borg_version,
+            'borgmajor': '%d' % borg_version_tuple[:1],
+            'borgminor': '%d.%d' % borg_version_tuple[:2],
+            'borgpatch': '%d.%d.%d' % borg_version_tuple[:3],
+        }
+    else:
+        data = {
+            'pid': os.getpid(),
+            'fqdn': socket.getfqdn(),
+            'hostname': socket.gethostname(),
+            'now': DatetimeWrapper(current_time.now()),
+            'utcnow': DatetimeWrapper(current_time.utcnow()),
+            'user': os.getlogin(),
+            'uuid4': str(uuid.uuid4()),
+            'borgversion': borg_version,
+            'borgmajor': '%d' % borg_version_tuple[:1],
+            'borgminor': '%d.%d' % borg_version_tuple[:2],
+            'borgpatch': '%d.%d.%d' % borg_version_tuple[:3],
+        }
     return format_line(text, data)
 
 

+ 4 - 2
src/borg/helpers/usergroup.py

@@ -1,5 +1,7 @@
-import grp
-import pwd
+import sys
+if sys.platform != 'win32':
+    import grp
+    import pwd
 from functools import lru_cache
 
 

+ 3 - 0
src/borg/item.pyx

@@ -184,6 +184,9 @@ class Item(PropDict):
     nlink = PropDict._make_property('nlink', int)
 
     part = PropDict._make_property('part', int)
+    win_dacl = PropDict._make_property('win_dacl', (str, type(None)), 'surrogate-escaped str or None', encode=safe_encode, decode=safe_decode)
+    win_sacl = PropDict._make_property('win_sacl', (str, type(None)), 'surrogate-escaped str or None', encode=safe_encode, decode=safe_decode)
+    user_sid = PropDict._make_property('user_sid', (str, type(None)), 'surrogate-escaped str or None', encode=safe_encode, decode=safe_decode)
 
     def get_size(self, hardlink_masters=None, memorize=False, compressed=False, from_chunks=False, consider_ids=None):
         """

+ 2 - 1
src/borg/platform/__init__.py

@@ -38,4 +38,5 @@ elif sys.platform == 'win32':  # pragma: windows only
     from .windows import get_owner, set_owner
     from .windows import get_ads
     from .windows import select
-
+    from .windows import get_process_id
+    from .windows import process_alive

+ 17 - 0
src/borg/platform/windows.pyx

@@ -99,6 +99,8 @@ cdef extern from 'windows.h':
     BOOL OpenThreadToken(HANDLE, DWORD, BOOL, HANDLE*)
     BOOL LookupPrivilegeValueW(wchar_t*, wchar_t*, _LUID*)
     BOOL AdjustTokenPrivileges(HANDLE, BOOL, _TOKEN_PRIVILEGES*, DWORD, _TOKEN_PRIVILEGES*, DWORD*)
+    
+    HANDLE OpenProcess(DWORD dwDesiredAccess, BOOL bInheritHandle, DWORD dwProcessId)
 
     HANDLE GetCurrentThread()
     HANDLE GetCurrentProcess()
@@ -134,6 +136,8 @@ cdef extern from 'windows.h':
 
     cdef extern int TOKEN_ADJUST_PRIVILEGES
     cdef extern int TOKEN_QUERY
+    
+    cdef extern int PROCESS_QUERY_INFORMATION
 
 cdef extern from 'accctrl.h':
     ctypedef enum _SE_OBJECT_TYPE:
@@ -567,3 +571,16 @@ def select(rlist, wlist, xlist, timeout=0):
         if size.value > 0:
             retRlist.append(pipe)
     return retRlist, wlist, retXlist
+
+
+def get_process_id():
+    #return hostname, pid, thread_id
+    return platform.node().lower(), os.getpid(), 0
+
+
+def process_alive(host, pid, thread):
+    if host.lower() != platform.node().lower():
+        return True
+    return (OpenProcess(PROCESS_QUERY_INFORMATION, False, pid) != NULL)
+        
+    

+ 1 - 0
src/borg/remote.py

@@ -43,6 +43,7 @@ from .algorithms.checksums import xxh64
 if sys.platform == 'win32':
     from .platform import select as windowsSelect
     select.select = windowsSelect
+
     class NoSSHClient(Error):
         """Could not find supported ssh client. Supported  clients are {}."""
 

+ 19 - 17
src/borg/repository.py

@@ -1,6 +1,7 @@
 import errno
 import mmap
 import os
+import sys
 import shutil
 import struct
 from binascii import hexlify, unhexlify
@@ -619,23 +620,24 @@ class Repository:
             else:
                 # Keep one full worst-case segment free in non-append-only mode
                 required_free_space += full_segment_size
-        try:
-            st_vfs = os.statvfs(self.path)
-        except OSError as os_error:
-            logger.warning('Failed to check free space before committing: ' + str(os_error))
-            return
-        # f_bavail: even as root - don't touch the Federal Block Reserve!
-        free_space = st_vfs.f_bavail * st_vfs.f_bsize
-        logger.debug('check_free_space: required bytes {}, free bytes {}'.format(required_free_space, free_space))
-        if free_space < required_free_space:
-            if self.created:
-                logger.error('Not enough free space to initialize repository at this location.')
-                self.destroy()
-            else:
-                self._rollback(cleanup=True)
-            formatted_required = format_file_size(required_free_space)
-            formatted_free = format_file_size(free_space)
-            raise self.InsufficientFreeSpaceError(formatted_required, formatted_free)
+        if sys.platform != 'win32':
+            try:
+                st_vfs = os.statvfs(self.path)
+            except OSError as os_error:
+                logger.warning('Failed to check free space before committing: ' + str(os_error))
+                return
+            # f_bavail: even as root - don't touch the Federal Block Reserve!
+            free_space = st_vfs.f_bavail * st_vfs.f_bsize
+            logger.debug('check_free_space: required bytes {}, free bytes {}'.format(required_free_space, free_space))
+            if free_space < required_free_space:
+                if self.created:
+                    logger.error('Not enough free space to initialize repository at this location.')
+                    self.destroy()
+                else:
+                    self._rollback(cleanup=True)
+                formatted_required = format_file_size(required_free_space)
+                formatted_free = format_file_size(free_space)
+                raise self.InsufficientFreeSpaceError(formatted_required, formatted_free)
 
     def log_storage_quota(self):
         if self.storage_quota:

+ 2 - 0
src/borg/testsuite/__init__.py

@@ -101,6 +101,8 @@ def are_fifos_supported():
 
 @functools.lru_cache()
 def is_utime_fully_supported():
+    if sys.platform == 'win32':
+        return False
     with unopened_tempfile() as filepath:
         # Some filesystems (such as SSHFS) don't support utime on symlinks
         if are_symlinks_supported():

+ 0 - 2
src/borg/testsuite/archiver.py

@@ -460,7 +460,6 @@ class ArchiverTestCase(ArchiverTestCaseBase):
             self.cmd('extract', self.repository_location + '::test')
             assert not os.path.exists('input/unix-socket')
 
-
     @pytest.mark.skipif(not are_symlinks_supported(), reason='symlinks not supported')
     def test_symlink_extract(self):
         self.create_test_files()
@@ -1460,7 +1459,6 @@ class ArchiverTestCase(ArchiverTestCaseBase):
             # Restore permissions so shutil.rmtree is able to delete it
             os.system('chmod -R u+w ' + self.repository_path)
 
-
     @pytest.mark.skipif('BORG_TESTS_IGNORE_MODES' in os.environ, reason='modes unreliable')
     def test_umask(self):
         self.create_regular_file('file1', size=1024 * 80)

+ 4 - 324
src/borg/testsuite/helpers.py

@@ -32,9 +32,6 @@ from ..helpers import dash_open
 
 from . import BaseTestCase, FakeInputs
 
-if sys.platform == 'win32':
-    import posixpath
-
 
 class BigIntTestCase(BaseTestCase):
 
@@ -106,12 +103,12 @@ class TestLocationWithoutEnv:
                 "Location(proto='file', user=None, host=None, port=None, path='/some/path', archive='archive')"
             assert repr(Location('file:///some/path')) == \
                 "Location(proto='file', user=None, host=None, port=None, path='/some/path', archive=None)"
+            assert Location('file:///some/path').to_key_filename() == keys_dir + 'some_path'
         else:
             assert repr(Location('file://C:/some/path::archive')).replace('\\\\', '/') == \
                 "Location(proto='file', user=None, host=None, port=None, path='C:/some/path', archive='archive')"
             assert repr(Location('file://C:/some/path')).replace('\\\\', '/') == \
                 "Location(proto='file', user=None, host=None, port=None, path='C:/some/path', archive=None)"
-            assert Location('file:///some/path').to_key_filename() == keys_dir + 'some_path'
 
     def test_scp(self, monkeypatch, keys_dir):
         monkeypatch.delenv('BORG_REPO', raising=False)
@@ -228,9 +225,9 @@ class TestLocationWithoutEnv:
         test_pid = os.getpid()
         assert repr(Location('/some/path::archive{pid}')) == \
             "Location(proto='file', user=None, host=None, port=None, path='/some/path', archive='archive{}')".format(test_pid)
-        location_time1 = Location('/some/path::archive{now}')
+        location_time1 = Location('/some/path::archive{now:%s}')
         sleep(1.1)
-        location_time2 = Location('/some/path::archive{now}')
+        location_time2 = Location('/some/path::archive{now:%s}')
         assert location_time1.archive != location_time2.archive
 
     def test_bad_syntax(self):
@@ -263,7 +260,7 @@ class TestLocationWithEnv:
             assert repr(Location('::archive')).replace('\\\\', '/') == \
                 "Location(proto='file', user=None, host=None, port=None, path='C:/some/path', archive='archive')"
             assert repr(Location()).replace('\\\\', '/') == \
-                "Location(proto='file', user=None, host=None, port=None, path='C:/some/path', archive=None)"'
+                "Location(proto='file', user=None, host=None, port=None, path='C:/some/path', archive=None)"
             assert repr(Location()) == \
                    "Location(proto='file', user=None, host=None, port=None, path='file://C:/some/path', archive=None)"
 
@@ -329,323 +326,6 @@ class FormatTimedeltaTestCase(BaseTestCase):
         )
 
 
-def check_patterns(files, pattern, expected):
-    """Utility for testing patterns.
-    """
-    assert all([f == os.path.normpath(f) for f in files]), "Pattern matchers expect normalized input paths"
-
-    matched = [f for f in files if pattern.match(f)]
-
-    assert matched == (files if expected is None else expected)
-
-
-@pytest.mark.parametrize("pattern, expected", [
-    # "None" means all files, i.e. all match the given pattern
-    ("/", None),
-    ("/./", None),
-    ("", []),
-    ("/home/u", []),
-    ("/home/user", ["/home/user/.profile", "/home/user/.bashrc"]),
-    ("/etc", ["/etc/server/config", "/etc/server/hosts"]),
-    ("///etc//////", ["/etc/server/config", "/etc/server/hosts"]),
-    ("/./home//..//home/user2", ["/home/user2/.profile", "/home/user2/public_html/index.html"]),
-    ("/srv", ["/srv/messages", "/srv/dmesg"]),
-    ])
-@pytest.mark.skipif(sys.platform == 'win32', reason='Need some windows path tests')
-def test_patterns_prefix(pattern, expected):
-    files = [
-        "/etc/server/config", "/etc/server/hosts", "/home", "/home/user/.profile", "/home/user/.bashrc",
-        "/home/user2/.profile", "/home/user2/public_html/index.html", "/srv/messages", "/srv/dmesg",
-    ]
-
-    check_patterns(files, PathPrefixPattern(pattern), expected)
-
-
-@pytest.mark.parametrize("pattern, expected", [
-    # "None" means all files, i.e. all match the given pattern
-    ("", []),
-    ("foo", []),
-    ("relative", ["relative/path1", "relative/two"]),
-    ("more", ["more/relative"]),
-    ])
-@pytest.mark.skipif(sys.platform == 'win32', reason='Need some windows path tests')
-def test_patterns_prefix_relative(pattern, expected):
-    files = ["relative/path1", "relative/two", "more/relative"]
-
-    check_patterns(files, PathPrefixPattern(pattern), expected)
-
-
-@pytest.mark.parametrize("pattern, expected", [
-    # "None" means all files, i.e. all match the given pattern
-    ("/*", None),
-    ("/./*", None),
-    ("*", None),
-    ("*/*", None),
-    ("*///*", None),
-    ("/home/u", []),
-    ("/home/*",
-     ["/home/user/.profile", "/home/user/.bashrc", "/home/user2/.profile", "/home/user2/public_html/index.html",
-      "/home/foo/.thumbnails", "/home/foo/bar/.thumbnails"]),
-    ("/home/user/*", ["/home/user/.profile", "/home/user/.bashrc"]),
-    ("/etc/*", ["/etc/server/config", "/etc/server/hosts"]),
-    ("*/.pr????e", ["/home/user/.profile", "/home/user2/.profile"]),
-    ("///etc//////*", ["/etc/server/config", "/etc/server/hosts"]),
-    ("/./home//..//home/user2/*", ["/home/user2/.profile", "/home/user2/public_html/index.html"]),
-    ("/srv*", ["/srv/messages", "/srv/dmesg"]),
-    ("/home/*/.thumbnails", ["/home/foo/.thumbnails", "/home/foo/bar/.thumbnails"]),
-    ])
-@pytest.mark.skipif(sys.platform == 'win32', reason='Need some windows path tests')
-def test_patterns_fnmatch(pattern, expected):
-    files = [
-        "/etc/server/config", "/etc/server/hosts", "/home", "/home/user/.profile", "/home/user/.bashrc",
-        "/home/user2/.profile", "/home/user2/public_html/index.html", "/srv/messages", "/srv/dmesg",
-        "/home/foo/.thumbnails", "/home/foo/bar/.thumbnails",
-    ]
-
-    check_patterns(files, FnmatchPattern(pattern), expected)
-
-
-@pytest.mark.parametrize("pattern, expected", [
-    # "None" means all files, i.e. all match the given pattern
-    ("*", None),
-    ("**/*", None),
-    ("/**/*", None),
-    ("/./*", None),
-    ("*/*", None),
-    ("*///*", None),
-    ("/home/u", []),
-    ("/home/*",
-     ["/home/user/.profile", "/home/user/.bashrc", "/home/user2/.profile", "/home/user2/public_html/index.html",
-      "/home/foo/.thumbnails", "/home/foo/bar/.thumbnails"]),
-    ("/home/user/*", ["/home/user/.profile", "/home/user/.bashrc"]),
-    ("/etc/*/*", ["/etc/server/config", "/etc/server/hosts"]),
-    ("/etc/**/*", ["/etc/server/config", "/etc/server/hosts"]),
-    ("/etc/**/*/*", ["/etc/server/config", "/etc/server/hosts"]),
-    ("*/.pr????e", []),
-    ("**/.pr????e", ["/home/user/.profile", "/home/user2/.profile"]),
-    ("///etc//////*", ["/etc/server/config", "/etc/server/hosts"]),
-    ("/./home//..//home/user2/", ["/home/user2/.profile", "/home/user2/public_html/index.html"]),
-    ("/./home//..//home/user2/**/*", ["/home/user2/.profile", "/home/user2/public_html/index.html"]),
-    ("/srv*/", ["/srv/messages", "/srv/dmesg", "/srv2/blafasel"]),
-    ("/srv*", ["/srv", "/srv/messages", "/srv/dmesg", "/srv2", "/srv2/blafasel"]),
-    ("/srv/*", ["/srv/messages", "/srv/dmesg"]),
-    ("/srv2/**", ["/srv2", "/srv2/blafasel"]),
-    ("/srv2/**/", ["/srv2/blafasel"]),
-    ("/home/*/.thumbnails", ["/home/foo/.thumbnails"]),
-    ("/home/*/*/.thumbnails", ["/home/foo/bar/.thumbnails"]),
-    ])
-@pytest.mark.skipif(sys.platform == 'win32', reason='Need some windows path tests')
-def test_patterns_shell(pattern, expected):
-    files = [
-        "/etc/server/config", "/etc/server/hosts", "/home", "/home/user/.profile", "/home/user/.bashrc",
-        "/home/user2/.profile", "/home/user2/public_html/index.html", "/srv", "/srv/messages", "/srv/dmesg",
-        "/srv2", "/srv2/blafasel", "/home/foo/.thumbnails", "/home/foo/bar/.thumbnails",
-    ]
-
-    check_patterns(files, ShellPattern(pattern), expected)
-
-
-@pytest.mark.parametrize("pattern, expected", [
-    # "None" means all files, i.e. all match the given pattern
-    ("", None),
-    (".*", None),
-    ("^/", None),
-    ("^abc$", []),
-    ("^[^/]", []),
-    ("^(?!/srv|/foo|/opt)",
-     ["/home", "/home/user/.profile", "/home/user/.bashrc", "/home/user2/.profile",
-      "/home/user2/public_html/index.html", "/home/foo/.thumbnails", "/home/foo/bar/.thumbnails", ]),
-    ])
-@pytest.mark.skipif(sys.platform == 'win32', reason='Need some windows path tests')
-def test_patterns_regex(pattern, expected):
-    files = [
-        '/srv/data', '/foo/bar', '/home',
-        '/home/user/.profile', '/home/user/.bashrc',
-        '/home/user2/.profile', '/home/user2/public_html/index.html',
-        '/opt/log/messages.txt', '/opt/log/dmesg.txt',
-        "/home/foo/.thumbnails", "/home/foo/bar/.thumbnails",
-    ]
-
-    obj = RegexPattern(pattern)
-    assert str(obj) == pattern
-    assert obj.pattern == pattern
-
-    check_patterns(files, obj, expected)
-
-
-def test_regex_pattern():
-    # The forward slash must match the platform-specific path separator
-    assert RegexPattern("^/$").match("/")
-    assert RegexPattern("^/$").match(os.path.sep)
-    assert not RegexPattern(r"^\\$").match("/")
-
-
-def use_normalized_unicode():
-    return sys.platform in ("darwin",)
-
-
-def _make_test_patterns(pattern):
-    return [PathPrefixPattern(pattern),
-            FnmatchPattern(pattern),
-            RegexPattern("^{}/foo$".format(pattern)),
-            ShellPattern(pattern),
-            ]
-
-
-@pytest.mark.parametrize("pattern", _make_test_patterns("b\N{LATIN SMALL LETTER A WITH ACUTE}"))
-def test_composed_unicode_pattern(pattern):
-    assert pattern.match("b\N{LATIN SMALL LETTER A WITH ACUTE}/foo")
-    assert pattern.match("ba\N{COMBINING ACUTE ACCENT}/foo") == use_normalized_unicode()
-
-
-@pytest.mark.parametrize("pattern", _make_test_patterns("ba\N{COMBINING ACUTE ACCENT}"))
-def test_decomposed_unicode_pattern(pattern):
-    assert pattern.match("b\N{LATIN SMALL LETTER A WITH ACUTE}/foo") == use_normalized_unicode()
-    assert pattern.match("ba\N{COMBINING ACUTE ACCENT}/foo")
-
-
-@pytest.mark.parametrize("pattern", _make_test_patterns(str(b"ba\x80", "latin1")))
-def test_invalid_unicode_pattern(pattern):
-    assert not pattern.match("ba/foo")
-    assert pattern.match(str(b"ba\x80/foo", "latin1"))
-
-
-@pytest.mark.parametrize("lines, expected", [
-    # "None" means all files, i.e. none excluded
-    ([], None),
-    (["# Comment only"], None),
-    (["*"], []),
-    (["# Comment",
-      "*/something00.txt",
-      "  *whitespace*  ",
-      # Whitespace before comment
-      " #/ws*",
-      # Empty line
-      "",
-      "# EOF"],
-     ["/more/data", "/home", " #/wsfoobar"]),
-    (["re:.*"], []),
-    (["re:\s"], ["/data/something00.txt", "/more/data", "/home"]),
-    ([r"re:(.)(\1)"], ["/more/data", "/home", "\tstart/whitespace", "/whitespace/end\t"]),
-    (["", "", "",
-      "# This is a test with mixed pattern styles",
-      # Case-insensitive pattern
-      "re:(?i)BAR|ME$",
-      "",
-      "*whitespace*",
-      "fm:*/something00*"],
-     ["/more/data"]),
-    ([r"  re:^\s  "], ["/data/something00.txt", "/more/data", "/home", "/whitespace/end\t"]),
-    ([r"  re:\s$  "], ["/data/something00.txt", "/more/data", "/home", " #/wsfoobar", "\tstart/whitespace"]),
-    (["pp:./"], None),
-    (["pp:/"], [" #/wsfoobar", "\tstart/whitespace"]),
-    (["pp:aaabbb"], None),
-    (["pp:/data", "pp: #/", "pp:\tstart", "pp:/whitespace"], ["/more/data", "/home"]),
-    ])
-def test_patterns_from_file(tmpdir, lines, expected):
-    files = [
-        '/data/something00.txt', '/more/data', '/home',
-        ' #/wsfoobar',
-        '\tstart/whitespace',
-        '/whitespace/end\t',
-    ]
-
-    def evaluate(filename):
-        matcher = PatternMatcher(fallback=True)
-        matcher.add(load_excludes(open(filename, "rt")), False)
-        return [path for path in files if matcher.match(path)]
-
-    exclfile = tmpdir.join("exclude.txt")
-
-    with exclfile.open("wt") as fh:
-        fh.write("\n".join(lines))
-
-    assert evaluate(str(exclfile)) == (files if expected is None else expected)
-
-
-@pytest.mark.parametrize("pattern, cls", [
-    ("", FnmatchPattern),
-
-    # Default style
-    ("*", FnmatchPattern),
-    ("/data/*", FnmatchPattern),
-
-    # fnmatch style
-    ("fm:", FnmatchPattern),
-    ("fm:*", FnmatchPattern),
-    ("fm:/data/*", FnmatchPattern),
-    ("fm:fm:/data/*", FnmatchPattern),
-
-    # Regular expression
-    ("re:", RegexPattern),
-    ("re:.*", RegexPattern),
-    ("re:^/something/", RegexPattern),
-    ("re:re:^/something/", RegexPattern),
-
-    # Path prefix
-    ("pp:", PathPrefixPattern),
-    ("pp:/", PathPrefixPattern),
-    ("pp:/data/", PathPrefixPattern),
-    ("pp:pp:/data/", PathPrefixPattern),
-
-    # Shell-pattern style
-    ("sh:", ShellPattern),
-    ("sh:*", ShellPattern),
-    ("sh:/data/*", ShellPattern),
-    ("sh:sh:/data/*", ShellPattern),
-    ])
-def test_parse_pattern(pattern, cls):
-    assert isinstance(parse_pattern(pattern), cls)
-
-
-@pytest.mark.parametrize("pattern", ["aa:", "fo:*", "00:", "x1:abc"])
-def test_parse_pattern_error(pattern):
-    with pytest.raises(ValueError):
-        parse_pattern(pattern)
-
-
-def test_pattern_matcher():
-    pm = PatternMatcher()
-
-    assert pm.fallback is None
-
-    for i in ["", "foo", "bar"]:
-        assert pm.match(i) is None
-
-    pm.add([RegexPattern("^a")], "A")
-    pm.add([RegexPattern("^b"), RegexPattern("^z")], "B")
-    pm.add([RegexPattern("^$")], "Empty")
-    pm.fallback = "FileNotFound"
-
-    assert pm.match("") == "Empty"
-    assert pm.match("aaa") == "A"
-    assert pm.match("bbb") == "B"
-    assert pm.match("ccc") == "FileNotFound"
-    assert pm.match("xyz") == "FileNotFound"
-    assert pm.match("z") == "B"
-
-    assert PatternMatcher(fallback="hey!").fallback == "hey!"
-
-
-def test_compression_specs():
-    with pytest.raises(ValueError):
-        CompressionSpec('')
-    assert CompressionSpec('none') == dict(name='none')
-    assert CompressionSpec('lz4') == dict(name='lz4')
-    assert CompressionSpec('zlib') == dict(name='zlib', level=6)
-    assert CompressionSpec('zlib,0') == dict(name='zlib', level=0)
-    assert CompressionSpec('zlib,9') == dict(name='zlib', level=9)
-    with pytest.raises(ValueError):
-        CompressionSpec('zlib,9,invalid')
-    assert CompressionSpec('lzma') == dict(name='lzma', level=6)
-    assert CompressionSpec('lzma,0') == dict(name='lzma', level=0)
-    assert CompressionSpec('lzma,9') == dict(name='lzma', level=9)
-    with pytest.raises(ValueError):
-        CompressionSpec('lzma,9,invalid')
-    with pytest.raises(ValueError):
-        CompressionSpec('invalid')
-
-
 def test_chunkerparams():
     assert ChunkerParams('19,23,21,4095') == (19, 23, 21, 4095)
     assert ChunkerParams('10,23,16,4095') == (10, 23, 16, 4095)

+ 4 - 1
src/borg/testsuite/platform.py

@@ -4,7 +4,8 @@ import random
 import shutil
 import sys
 import tempfile
-import pwd
+if sys.platform != 'win32':
+    import pwd
 import unittest
 
 from ..platform import acl_get, acl_set, swidth
@@ -43,6 +44,8 @@ def fakeroot_detected():
 
 
 def user_exists(username):
+    if sys.platform == 'win32':
+        return True
     try:
         pwd.getpwnam(username)
         return True

+ 1 - 1
src/borg/testsuite/repository.py

@@ -925,7 +925,7 @@ class RemoteRepositoryCheckTestCase(RepositoryCheckTestCase):
     def open(self, create=False):
         if sys.platform != 'win32':
             return RemoteRepository(Location('__testsuite__:' + os.path.join(self.tmppath, 'repository')),
-			                        exclusive=True, create=create)
+                                    exclusive=True, create=create)
         else:
             return RemoteRepository(Location(os.path.join(self.tmppath, 'repository')), create=create)