scripts/pvlbackup-rsync-snapshot
author Tero Marttila <terom@paivola.fi>
Fri, 02 Mar 2012 17:16:16 +0200
changeset 37 9103a9456087
parent 36 6070507e09ce
child 38 24cdf1372cab
permissions -rwxr-xr-x
rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     1
#!/usr/bin/python
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     2
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     3
"""
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     4
    Manage rsync --link-dest based snapshots.
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     5
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     6
    rsync's from <src> to <dst>/snapshots/YYYY-MM-DD-HH-MM-SS using --link-dest <dst>/current.
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     7
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     8
    Updates symlink <dst>/current -> <dst>/snapshots/...
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     9
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    10
    Then archives <dst>/current to <dst>/<period>/<date> using --link-dest.
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    11
"""
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    12
30
29b60df79122 version: 0.2.3; move version to pvl.backup.__version__; add --version opt
Tero Marttila <terom@paivola.fi>
parents: 22
diff changeset
    13
from pvl.backup import __version__
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    14
from pvl.backup import rsync
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    15
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    16
import optparse, ConfigParser
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    17
import os, os.path, stat
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    18
import shutil
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    19
import datetime
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    20
import logging
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    21
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    22
log = logging.getLogger()
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    23
36
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
    24
# command-line options, global state
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    25
options = None
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    26
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    27
def parse_options (argv, defaults) :
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    28
    """
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    29
        Parse command-line arguments.
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    30
    """
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    31
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    32
    parser = optparse.OptionParser(
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    33
            prog        = argv[0],
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    34
            usage       = '%prog: [options] [ --config <path> | --target <path> [ --source <src> ] [ --interval <name> ] ]',
30
29b60df79122 version: 0.2.3; move version to pvl.backup.__version__; add --version opt
Tero Marttila <terom@paivola.fi>
parents: 22
diff changeset
    35
            version     = __version__,
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    36
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    37
            # module docstring
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    38
            # XXX: breaks multi-line descriptions..
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    39
            description = __doc__,
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    40
    )
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    41
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    42
    # logging
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    43
    general = optparse.OptionGroup(parser, "General Options")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    44
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    45
    general.add_option('-q', '--quiet',      dest='loglevel', action='store_const', const=logging.WARNING, help="Less output")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    46
    general.add_option('-v', '--verbose',    dest='loglevel', action='store_const', const=logging.INFO,  help="More output")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    47
    general.add_option('-D', '--debug',      dest='loglevel', action='store_const', const=logging.DEBUG, help="Even more output")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    48
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    49
    parser.add_option_group(general)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    50
14
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    51
    # rsync
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    52
    rsync = optparse.OptionGroup(parser, "rsync Options")
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    53
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    54
    rsync.add_option('--exclude-from',       metavar='FILE',
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    55
        help="Read exclude rules from given file")
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    56
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    57
    rsync.add_option('--include-from',       metavar='FILE',
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    58
        help="Read include rules from given file")
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    59
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    60
    parser.add_option_group(rsync)
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
    61
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    62
    # global
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    63
    parser.add_option('--clean-intervals',  action='store_true',
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    64
        help="Clean out old interval links")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    65
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    66
    parser.add_option('--clean-snapshots',  action='store_true',
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    67
        help="Clean out unused snapshots (those not linked to)")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    68
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    69
    parser.add_option('--clean',             action='store_true',
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    70
        help="Clean out both intervals and snapshots")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    71
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    72
    parser.add_option('-n', '--dry-run',    action='store_true',
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    73
        help="Don't actually clean anything")
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    74
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    75
    #
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    76
    parser.add_option('-c', '--config',     metavar='FILE',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    77
        help="Load configuration file")
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    78
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    79
    #
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    80
    parser.add_option('-T', '--target',    metavar='PATH',
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    81
        help="Target path")
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    82
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    83
    parser.add_option('-s', '--source',     metavar='RSYNC-PATH', dest='target_source', default=False,
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    84
        help="Run target backup from source in rsync-syntax")
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    85
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    86
    parser.add_option('--interval',         metavar='NAME', action='append', dest='target_intervals',
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    87
        help="Run target with given given interval(s)")
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    88
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    89
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    90
    # defaults
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    91
    parser.set_defaults(
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
    92
        loglevel            = logging.INFO,
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    93
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
    94
        target_intervals    = [],
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    95
    )
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    96
    parser.set_defaults(**defaults)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    97
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
    98
    
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    99
    # parse
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   100
    options, args = parser.parse_args(argv[1:])
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   101
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   102
    # configure
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   103
    logging.basicConfig(
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   104
        format  = '%(processName)s: %(name)s: %(levelname)s %(funcName)s : %(message)s',
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   105
        level   = options.loglevel,
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   106
    )
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   107
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   108
    if options.clean :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   109
        options.clean_intervals = options.clean_snapshots = options.clean
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   110
14
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   111
    if options.include_from :
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   112
        options.rsync_options['include-from'] = options.include_from
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   113
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   114
    if options.exclude_from :
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   115
        options.rsync_options['exclude-from'] = options.exclude_from
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   116
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   117
    return options, args
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   118
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   119
## Configuration
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   120
class ConfigError (Exception) :
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   121
    pass
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   122
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   123
def process_config_name (name) :
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   124
    """
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   125
        Process config file name into python version
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   126
    """
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   127
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   128
    return name.replace('-', '_')
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   129
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   130
def parse_config (path, defaults) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   131
    """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   132
        Parse given config file
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   133
    """
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   134
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   135
    log.debug("loading config: %s", path)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   136
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   137
    config = dict(defaults)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   138
    config_file = ConfigParser.RawConfigParser()
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   139
    config_file.read([path])
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   140
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   141
    # handle each section
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   142
    for section in config_file.sections() :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   143
        # mangle
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   144
        section_name = process_config_name(section)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   145
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   146
        log.debug("section: %s", section_name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   147
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   148
        # subsections
36
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   149
        if ':' in section_name :
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   150
            # legacy!
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   151
            section_path = section_name.split(':')
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   152
        else :
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   153
            # new! shiny!
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   154
            section_path = section_name.split('/')
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   155
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   156
        # lookup section dict from config
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   157
        lookup = config
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   158
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   159
        # XXX: sections are not in order, so we can't rely on the parent section being created before we handle the sub-section
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   160
        for name in section_path :
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   161
            # possibly create
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   162
            if name not in lookup :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   163
                lookup[name] = {}
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   164
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   165
            lookup = lookup[name]
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   166
 
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   167
        # found dict for this section
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   168
        config_section = lookup
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   169
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   170
        # values
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   171
        for name, value in config_file.items(section) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   172
            # mangle
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   173
            name = process_config_name(name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   174
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   175
            log.debug("section: %s: %s = %s", '/'.join(section_path), name, value)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   176
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   177
            config_section[name] = value
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   178
    
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   179
    log.debug("config: %s", config)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   180
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   181
    return config
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   182
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   183
def config_bool (name, value, strict=True) :
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   184
    if value.lower() in ('yes', 'true', '1', 'on') :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   185
        return True
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   186
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   187
    elif value.lower() in ('no', 'false', '0', 'off') :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   188
        return False
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   189
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   190
    elif strict :
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   191
        raise ConfigError("Unrecognized boolean value: {name} = {value}".format(name=name, value=value))
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   192
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   193
    else :
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   194
        # allow non-boolean values
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   195
        return value
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   196
37
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   197
def config_int (name, value, default=False) :
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   198
    if not value and default is not False:
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   199
        # returning default value if one is given
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   200
        return default
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   201
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   202
    try :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   203
        return int(value)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   204
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   205
    except ValueError, e:
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   206
        raise ConfigError("Invalid integer value: {name} = {value}".format(name=name, value=value))
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   207
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   208
def config_list (name, value) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   209
    return value.split()
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   210
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   211
def walk_symlinks (tree, ignore=False) :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   212
    """
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   213
        Walk through all symlinks in given dir, yielding:
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   214
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   215
            (dirpath, name, target)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   216
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   217
        Passes through errors from os.listdir/os.lstat.
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   218
    """
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   219
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   220
    for name in os.listdir(tree) :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   221
        if ignore and name in ignore :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   222
            log.debug("%s: ignore: %s", tree, name)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   223
            continue
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   224
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   225
        path = os.path.join(tree, name)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   226
        
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   227
        # stat symlink itself
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   228
        st = os.lstat(path)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   229
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   230
        if stat.S_ISDIR(st.st_mode) :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   231
            # recurse
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   232
            log.debug("%s: tree: %s", tree, name)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   233
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   234
            for item in walk_symlinks(path) :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   235
                yield item
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   236
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   237
        elif stat.S_ISLNK(st.st_mode) :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   238
            # found
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   239
            target = os.readlink(path)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   240
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   241
            log.debug("%s: link: %s -> %s", tree, name, target)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   242
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   243
            yield tree, name, target
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   244
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   245
        else :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   246
            log.debug("%s: skip: %s", tree, name)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   247
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   248
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   249
class Interval (object) :
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   250
    """
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   251
        An interval definition.
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   252
    """
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   253
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   254
    @classmethod
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   255
    def from_config (cls, options, name,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   256
        format,
37
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   257
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   258
        # deprecated
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   259
        keep    = None,
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   260
    ) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   261
        if not format :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   262
            # magic to use snapshot name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   263
            _format = None
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   264
        else :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   265
            _format = format
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   266
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   267
        return cls(name, 
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   268
            format  = _format, 
37
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   269
            keep    = config_int('keep', keep, default=None),
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   270
        )
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   271
16
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   272
    @classmethod
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   273
    def from_target_config (cls, name, base, arg) :
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   274
        if isinstance(arg, dict) :
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   275
            # full instance
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   276
            return cls(name,
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   277
                format  = arg.get('format', base.format if base else None),
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   278
                keep    = arg.get('keep', base.keep if base else None),
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   279
            )
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   280
        else :
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   281
            # partial instance with keep
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   282
            return cls(name,
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   283
                format  = base.format,
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   284
                keep    = config_int('keep', arg) if arg else base.keep,
16
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   285
            )
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   286
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   287
    def __init__ (self, name, format, keep) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   288
        self.name = name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   289
        self.format = format
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   290
        self.keep = keep
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   291
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   292
    def __str__ (self) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   293
        return self.name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   294
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   295
class Target (object) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   296
    """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   297
        A target run, i.e. a rsync-snapshot destination dir
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   298
            
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   299
        [target:...]
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   300
    """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   301
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   302
    @classmethod
35
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   303
    def config_intervals (cls, name, intervals) :
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   304
        for interval, arg in intervals.iteritems() :
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   305
            # lookup base from options.intervals
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   306
            try :
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   307
                base = options.intervals[interval]
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   308
            except KeyError:
36
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   309
                raise ConfigError("Unknown interval for [target/{target}]: {interval}".format(target=name, interval=interval))
35
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   310
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   311
            # parse
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   312
            yield Interval.from_target_config(interval, base, arg)
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   313
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   314
    @classmethod
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   315
    def from_config (cls, options, name,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   316
        path            = False,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   317
        source          = None,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   318
        enable          = 'no',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   319
        exclude_from    = None,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   320
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   321
        # subsections
16
d4b9954273a1 [target:$] intervals -> [target:$:intervals]
Tero Marttila <terom@paivola.fi>
parents: 15
diff changeset
   322
        intervals       = None,
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   323
        rsync_options   = None,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   324
    ) :
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   325
        if not source and source is not False :
36
6070507e09ce change rsync-snapshot.conf to use [foo/bar] syntax, because
Tero Marttila <terom@paivola.fi>
parents: 35
diff changeset
   326
            raise ConfigError("Missing required option: source for [target/{name}]".format(name=name))
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   327
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   328
        # global defaults
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   329
        _rsync_options = dict(options.rsync_options)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   330
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   331
        if rsync_options :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   332
            # override
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   333
            _rsync_options.update([
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   334
                # parse
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   335
                (option, config_bool(option, value, strict=False)) for option, value in rsync_options.iteritems()
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   336
            ])
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   337
37
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   338
        if not intervals :
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   339
            raise ConfigError("Missing required [target/{name}/intervals]".format(name=name))
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   340
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   341
        # lookup intervals
35
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   342
        _intervals = list(cls.config_intervals(name, intervals))
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   343
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   344
        return cls(name, 
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   345
            path            = path if path else name,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   346
            source          = source,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   347
            enable          = config_bool('enable', enable),
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   348
            intervals       = _intervals,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   349
            rsync_options   = _rsync_options,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   350
            exclude_from    = exclude_from,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   351
        )
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   352
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   353
    def __init__ (self, name,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   354
        path,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   355
        source, 
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   356
        enable          = False, 
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   357
        intervals       = [],
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   358
        rsync_options   = {},
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   359
        exclude_from    = None
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   360
    ) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   361
        self.name = name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   362
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   363
        self.path = path
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   364
        self.source = source
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   365
        self.enable = enable
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   366
        
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   367
        self.intervals = intervals
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   368
        
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   369
        self.rsync_options = rsync_options
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   370
        self.exclude_from = exclude_from
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   371
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   372
        # this snapshot?
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   373
        self.snapshots_dir = os.path.join(self.path, 'snapshots')
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   374
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   375
        # 'current' symlink
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   376
        self.current_path = os.path.join(self.path, 'current')
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   377
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   378
    def prepare (self, options) :
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   379
        """
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   380
            Prepare dir for usage
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   381
        """
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   382
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   383
        if not os.path.exists(self.path) :
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   384
            raise Exception("Missing target dir: {path}".format(path=self.path))
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   385
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   386
        if not os.path.exists(self.snapshots_dir) :
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   387
            log.warn("Creating snapshots dir: %s", self.snapshots_dir)
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   388
            os.mkdir(self.snapshots_dir)
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   389
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   390
    def snapshot (self, options, now) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   391
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   392
            Perform the rsync from our source to self.snapshot_dir.
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   393
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   394
            XXX: allocate snapshot_name here?
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   395
        """
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   396
       
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   397
        # new snapshot
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   398
        snapshot_name = now.strftime(options.snapshot_format)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   399
        snapshot_path = os.path.join(self.snapshots_dir, snapshot_name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   400
        temp_path = os.path.join(self.snapshots_dir, 'tmp')
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   401
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   402
        if os.path.exists(temp_path) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   403
            raise Exception("Old temp snapshot dir remains, please clean up: {path}".format(path=temp_path))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   404
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   405
        log.info("Perform main snapshot: %s -> %s", self.source, snapshot_path)
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   406
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   407
        # build rsync options
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   408
        opts = dict(self.rsync_options)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   409
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   410
        if os.path.exists(self.current_path) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   411
            # real path to target
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   412
            target = os.readlink(self.current_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   413
            target_path = os.path.join(os.path.dirname(self.current_path), target)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   414
            target_abs = os.path.abspath(target_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   415
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   416
            log.info("Using current -> %s as base", target_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   417
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   418
            # use as link-dest base; hardlinks unchanged files; target directory must be empty
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   419
            # rsync links absolute paths..
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   420
            opts['link-dest'] = target_abs
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   421
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   422
        # go
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   423
        log.debug("rsync %s -> %s", self.source, temp_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   424
        rsync.rsync(self.source, temp_path, **opts)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   425
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   426
        # move in to final name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   427
        log.debug("rename %s -> %s", temp_path, snapshot_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   428
        os.rename(temp_path, snapshot_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   429
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   430
        return snapshot_name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   431
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   432
    def update_interval (self, options, interval, now, snapshot_name) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   433
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   434
            Update given <interval>/... links for this target, using the given new snapshot
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   435
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   436
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   437
        dir_path = os.path.join(self.path, interval.name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   438
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   439
        if not os.path.exists(dir_path) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   440
            log.warn("Creating interval dir: %s", dir_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   441
            os.mkdir(dir_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   442
        
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   443
        
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   444
        # name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   445
        if interval.format is None :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   446
            # per-snapshot
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   447
            name = snapshot_name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   448
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   449
            log.debug("%s: using snapshot_name: %s", interval, name)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   450
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   451
        else :
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   452
            # by date
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   453
            name = now.strftime(interval.format)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   454
            
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   455
            log.debug("%s: using interval.format: %s -> %s", interval, interval.format, name)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   456
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   457
        # path
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   458
        path_name = os.path.join(interval.name, name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   459
        path = os.path.join(self.path, path_name)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   460
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   461
        log.debug("%s: processing %s", interval, path_name)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   462
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   463
        # already there?
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   464
        if os.path.exists(path) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   465
            target = os.readlink(path)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   466
35
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   467
            log.info("%s: Keeping existing: %s -> %s", interval, name, target)
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   468
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   469
        else :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   470
            # update
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   471
            target = os.path.join('..', 'snapshots', snapshot_name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   472
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   473
            log.info("%s: Updating: %s -> %s", interval, name, target)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   474
            log.debug("%s -> %s", path, target)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   475
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   476
            os.symlink(target, path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   477
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   478
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   479
    def clean_interval (self, options, interval) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   480
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   481
            Clean out given <interval>/... dir for this target.
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   482
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   483
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   484
        # path
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   485
        dir_path = os.path.join(self.path, interval.name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   486
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   487
        if not os.path.exists(dir_path) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   488
            log.warn("%s: Skipping, no interval dir: %s", interval, dir_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   489
            return
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   490
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   491
        # configured
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   492
        keep = interval.keep
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   493
37
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   494
        if not keep :
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   495
            log.info("%s: Zero keep given, not cleaning up anything", interval)
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   496
            return
9103a9456087 rsync-snapshot: fail nicer-ly if no intervals are given; deprecate default keep intervals, just don't cleanup old snapshots if no keep is given
Tero Marttila <terom@paivola.fi>
parents: 36
diff changeset
   497
31
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   498
        # items to clean?
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   499
        items = os.listdir(dir_path)
31
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   500
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   501
        # sort newest -> oldest
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   502
        items.sort(reverse=True)
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   503
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   504
        log.info("%s: Have %d / %d items", interval, len(items), keep)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   505
        log.debug("%s: items: %s", interval, ' '.join(items))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   506
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   507
        if len(items) > keep :
31
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   508
            # select oldest ones
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   509
            clean = items[keep:]
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   510
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   511
            log.info("%s: Cleaning out %d items", interval, len(clean))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   512
            log.debug("%s: cleaning out: %s", interval, ' '.join(clean))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   513
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   514
            for item in clean :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   515
                path = os.path.join(dir_path, item)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   516
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   517
                log.info("%s: Clean: %s", interval, path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   518
31
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   519
                if not options.dry_run :
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   520
                    log.debug("rmtree: %s", path)
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   521
                    os.unlink(path)
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   522
                else :
e946c741c500 rsync-snapshot: fix clean_interval to actually clean up the oldest snapshots, not the newest ones..
Tero Marttila <terom@paivola.fi>
parents: 30
diff changeset
   523
                    log.debug("dryrun: %s", path)
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   524
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   525
    def clean_snapshots (self, options) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   526
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   527
            Clean out all snapshots for this target not linked to from within our root.
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   528
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   529
            Fails without doing anything if unable to read the destination dir.
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   530
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   531
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   532
        # real path to snapshots
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   533
        snapshots_path = os.path.realpath(os.path.abspath(self.snapshots_dir))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   534
        log.debug("real snapshots_path: %s", snapshots_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   535
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   536
        # set of found targets
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   537
        found = set()
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   538
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   539
        # walk all symlinks
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   540
        for dirpath, name, target in walk_symlinks(self.path, ignore=set(['snapshots'])) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   541
            # target dir
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   542
            target_path = os.path.realpath(os.path.join(dirpath, target))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   543
            target_dir = os.path.dirname(target_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   544
            target_name = os.path.basename(target_path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   545
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   546
            if target_dir == snapshots_path :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   547
                log.debug("%s: found: %s -> %s", dirpath, name, target_name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   548
                found.add(target_name)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   549
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   550
            else :
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   551
                log.debug("%s: ignore: %s -> %s", dirpath, name, target_path)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   552
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   553
        # discover all snapshots
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   554
        snapshots = set(os.listdir(snapshots_path))
14
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   555
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   556
        # clean out special names
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   557
        snapshots = snapshots - set(['new'])
14
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   558
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   559
        ## compare
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   560
        used = snapshots & found
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   561
        unused = snapshots - found
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   562
        broken = found - snapshots
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   563
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   564
        log.info("Found used=%d, unused=%d, broken=%d snapshot symlinks", len(used), len(unused), len(broken))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   565
        log.debug("used=%s, unused=%s", used, unused)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   566
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   567
        if broken :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   568
            log.warn("Found broken symlinks to snapshots: %s", ' '.join(broken))
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   569
        
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   570
        if unused :
35
b6c0bdd6d8d4 rsync-snapshot: better ConfigError for referencing invalid [interval:..] from [target:...:intervals]; tidy up INFO output a little
Tero Marttila <terom@paivola.fi>
parents: 31
diff changeset
   571
            log.info("Cleaning out %d unused snapshots:", len(unused))
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   572
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   573
            for name in unused :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   574
                path = os.path.join(snapshots_path, name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   575
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   576
                log.info("Clean: %s", name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   577
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   578
                if not options.dry_run :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   579
                    log.debug("rmtree: %s", path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   580
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   581
                    # nuke
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   582
                    shutil.rmtree(path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   583
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   584
                else :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   585
                    log.debug("dry-run: %s", path)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   586
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   587
    def run_snapshot (self, options, now) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   588
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   589
            Run snapshot + update current.
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   590
        """
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   591
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   592
        # initial rsync
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   593
        snapshot_name = self.snapshot(options, now)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   594
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   595
        # update current
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   596
        log.info("Updating current -> %s", snapshot_name)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   597
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   598
        if os.path.islink(self.current_path) :
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   599
            # replace
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   600
            os.unlink(self.current_path)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   601
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   602
        os.symlink(os.path.join('snapshots', snapshot_name), self.current_path)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   603
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   604
        return snapshot_name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   605
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   606
    def run_intervals (self, options, now, snapshot_name) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   607
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   608
            Run our intervals.
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   609
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   610
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   611
        if not self.intervals :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   612
            log.info("No intervals given; not running any")
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   613
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   614
        else :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   615
            # maintain intervals
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   616
            log.info("Updating %d intervals...", len(self.intervals))
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   617
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   618
            for interval in self.intervals :
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   619
                log.debug("%s", interval)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   620
14
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   621
                log.info("Updating interval: %s", interval)
2a7b87dc6c45 rsync-snapshot: clean before update, fix link-dest with abspath, include/exclude
Tero Marttila <terom@paivola.fi>
parents: 12
diff changeset
   622
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   623
                # update
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   624
                self.update_interval(options, interval, now, snapshot_name)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   625
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   626
    def run (self, options) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   627
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   628
            Execute
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   629
        """
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   630
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   631
        # prep
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   632
        self.prepare(options)
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   633
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   634
        # clean intervals?
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   635
        if options.clean_intervals:
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   636
            for interval in self.intervals :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   637
                log.info("Cleaning interval: %s...", interval)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   638
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   639
                self.clean_interval(options, interval)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   640
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   641
        # clean snapshots?
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   642
        if options.clean_snapshots :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   643
            log.info("Cleaning snapshots...")
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   644
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   645
            self.clean_snapshots(options)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   646
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   647
        # snapshot from source?
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   648
        if self.source :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   649
            # timestamp for run
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   650
            now = datetime.datetime.now()
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   651
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   652
            log.info("Started snapshot run at: %s", now)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   653
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   654
            # snapshot + current
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   655
            snapshot_name = self.run_snapshot(options, now)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   656
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   657
            # intervals?
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   658
            self.run_intervals(options, now, snapshot_name)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   659
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   660
        # ok
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   661
        return 1
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   662
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   663
    def __str__ (self) :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   664
        return self.name
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   665
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   666
def run (options, targets) :
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   667
    # default config
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   668
    config = dict(
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   669
        rsync_options   = {},
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   670
        intervals       = {},
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   671
        targets         = {},
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   672
    )
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   673
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   674
    if options.config :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   675
        # load
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   676
        try :
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   677
            config = parse_config(options.config, config)
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   678
        except ConfigError as e:
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   679
            log.error("Configuration error: %s: %s", options.config, e)
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   680
            return 2
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   681
 
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   682
    # manual?
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   683
    if options.target :
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   684
        config['targets'][options.target] = dict(
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   685
            path        = options.target,
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   686
            source      = options.target_source,
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   687
            intervals   = dict((name, None) for name in options.target_intervals),
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   688
        )
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   689
  
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   690
    # intervals
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   691
    for name in config['intervals'] :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   692
        interval_config = config['intervals'][name]
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   693
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   694
        # parse
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   695
        interval = Interval.from_config(options, name, **interval_config)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   696
        
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   697
        log.debug("config interval: %s", name)
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   698
        
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   699
        # store
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   700
        options.intervals[name] = interval
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   701
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   702
    for option in config['rsync_options'] :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   703
        value = config['rsync_options'][option]
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   704
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   705
        # parse, allowing non-boolean values as well...
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   706
        value = config_bool(option, value, strict=False)
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   707
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   708
        log.debug("rsync option: %s=%s", option, value)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   709
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   710
        # store
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   711
        options.rsync_options[option] = value
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   712
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   713
    # what targets?
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   714
    if not targets :
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   715
        # default to all defined targets
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   716
        targets = list(config['targets'])
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   717
    
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   718
    else :
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   719
        # given ones, but verify they exist
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   720
        for target in targets :
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   721
            if target not in config['targets'] :
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   722
                log.error("Unknown target given: %s", target)
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   723
                log.info("Defined targets: %s", ' '.join(config['targets']))
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   724
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   725
                return 2
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   726
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   727
    # targets
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   728
    for name in targets :
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   729
        target_config = config['targets'][name]
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   730
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   731
        # parse
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   732
        target = Target.from_config(options, name, **target_config)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   733
21
de69e9ba8f22 rsync-snapshot: fix [rsync-options] handling for non-flag options
Tero Marttila <terom@paivola.fi>
parents: 17
diff changeset
   734
        log.info("Config target: %s", name)
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   735
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   736
        # run
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   737
        target.run(options)
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   738
22
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   739
    # ok
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   740
    return 0
a2bc5cc9de4d 0.2.1: slightly improved error handling
Tero Marttila <terom@paivola.fi>
parents: 21
diff changeset
   741
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   742
def config_defaults () :
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   743
    return dict(
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   744
        # snapshots/ naming
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   745
        snapshot_format = '%Y%m%d-%H%M%S',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   746
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   747
        # rsync options, in invoke.optargs format
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   748
        rsync_options = {
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   749
            'archive':          True,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   750
            'hard-links':       True,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   751
            'one-file-system':  True,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   752
            'numeric-ids':      True,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   753
            'delete':           True,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   754
        },
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   755
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   756
        # defined intervals
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   757
        intervals       = dict((i.name, i) for i in [
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   758
            Interval('recent',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   759
                format  = None,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   760
                keep    = 4,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   761
            ),
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   762
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   763
            Interval('day',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   764
                format  = '%Y-%m-%d',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   765
                keep    = 7,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   766
            ),
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   767
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   768
            Interval('week',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   769
                format  = '%Y-%W',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   770
                keep    = 4,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   771
            ),
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   772
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   773
            Interval('month',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   774
                format  = '%Y-%m',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   775
                keep    = 4,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   776
            ),
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   777
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   778
            Interval('year',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   779
                format  = '%Y',
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   780
                keep    = 1,
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   781
            )
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   782
        ]),
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   783
    )
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   784
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   785
def main (argv) :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   786
    global options
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   787
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   788
    # option defaults
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   789
    defaults = config_defaults()
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   790
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   791
    # global options + args
15
61f6d0ca0432 rsync-snapshot: some kind of --config support
Tero Marttila <terom@paivola.fi>
parents: 14
diff changeset
   792
    options, args = parse_options(argv, defaults)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   793
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   794
    # args: filter targets
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   795
    # XXX: fix name mangling
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   796
    targets = [target.replace('-', '_') for target in args]
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   797
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   798
    try :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   799
        # handle it
17
b88653920e7a fix up --destination/--source to --target/--source to run manually; specify list of targets to run as args
Tero Marttila <terom@paivola.fi>
parents: 16
diff changeset
   800
        return run(options, targets)
12
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   801
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   802
    except Exception, e:
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   803
        log.error("Internal error:", exc_info=e)
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   804
        return 3
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   805
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   806
    # ok
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   807
    return 0
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   808
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   809
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   810
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   811
if __name__ == '__main__' :
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   812
    import sys
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   813
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   814
    sys.exit(main(sys.argv))
fbfdde7326f4 rsync-snapshot: manage --link-dest'd interval snapshots
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
   815