duplitab 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. #!/usr/bin/env python3
  2. # PYTHON_ARGCOMPLETE_OK
  3. import hashlib
  4. import os
  5. import shlex
  6. import subprocess
  7. import tempfile
  8. import time
  9. import urllib.parse
  10. import yaml
  11. def command_join(args):
  12. return ' '.join([shlex.quote(a) for a in args])
  13. def sshfs_mount(url, path):
  14. url_attr = urllib.parse.urlparse(url)
  15. assert url_attr.scheme in ['sftp']
  16. mount_command = [
  17. 'sshfs',
  18. '{}:{}'.format(url_attr.netloc, url_attr.path),
  19. path,
  20. ]
  21. print('+ {}'.format(command_join(mount_command)))
  22. subprocess.check_call(mount_command)
  23. def sshfs_unmount(path):
  24. unmount_command = [
  25. 'fusermount',
  26. '-u',
  27. path,
  28. ]
  29. print('+ {}'.format(command_join(unmount_command)))
  30. subprocess.check_call(unmount_command)
  31. def backup(config, no_print_config, no_print_statistics, tab_dry):
  32. for backup in config:
  33. if not no_print_config:
  34. print(yaml.dump({"backup": backup}, default_flow_style = False))
  35. backup_command = ['duplicity']
  36. # encryption
  37. try:
  38. encryption = backup['encryption']
  39. except KeyError:
  40. encryption = True
  41. if 'encryption' in backup and not backup['encryption']:
  42. backup_command += ['--no-encryption']
  43. else:
  44. if 'encrypt_key' in backup:
  45. backup_command += ['--encrypt-key', backup['encrypt_key']]
  46. # selectors
  47. try:
  48. selectors = backup['selectors']
  49. except KeyError:
  50. selectors = []
  51. for selector in selectors:
  52. if selector['option'] in ['include', 'exclude']:
  53. backup_command += ['--{}'.format(selector['option']), selector['shell_pattern']]
  54. else:
  55. raise Exception("unsupported selector option '{}'".format(selector['option']))
  56. # statistics
  57. if no_print_statistics:
  58. backup_command.append('--no-print-statistics')
  59. # source
  60. try:
  61. source_type = backup['source_type']
  62. except KeyError:
  63. source_type = 'local'
  64. if source_type == 'local':
  65. backup_command += [backup['source_path']]
  66. else:
  67. raise Exception("unsupported source type '{}'".format(source_type))
  68. # target
  69. try:
  70. target_mount_path = None
  71. if 'target_via_sshfs' in backup and backup['target_via_sshfs']:
  72. target_mount_path = tempfile.mkdtemp(prefix = 'duplitab-target-sshfs-')
  73. backup_command += ['file://' + target_mount_path]
  74. sshfs_mount(backup['target_url'], target_mount_path)
  75. # set backup name to make archive dir persistent
  76. # (default name: hash of target url)
  77. backup_command += ['--name', hashlib.sha1(backup['target_url'].encode('utf-8')).hexdigest()]
  78. else:
  79. backup_command += [backup['target_url']]
  80. try:
  81. if tab_dry:
  82. print('* {}'.format(command_join(backup_command)))
  83. else:
  84. print('+ {}'.format(command_join(backup_command)))
  85. subprocess.check_call(backup_command)
  86. finally:
  87. if target_mount_path:
  88. try:
  89. sshfs_unmount(target_mount_path)
  90. except subprocess.CalledProcessError:
  91. time.sleep(1)
  92. sshfs_unmount(target_mount_path)
  93. finally:
  94. if target_mount_path:
  95. os.rmdir(target_mount_path)
  96. def run(command, config_path, no_print_config, no_print_statistics, tab_dry):
  97. with open(config_path) as config_file:
  98. config = yaml.load(config_file.read())
  99. if command == 'backup':
  100. backup(
  101. config = config,
  102. no_print_config = no_print_config,
  103. no_print_statistics = no_print_statistics,
  104. tab_dry = tab_dry,
  105. )
  106. def _init_argparser():
  107. import argparse
  108. argparser = argparse.ArgumentParser(description = None)
  109. argparser.add_argument(
  110. '-c',
  111. '--config',
  112. dest = 'config_path',
  113. default = '/etc/duplitab',
  114. )
  115. subparsers = argparser.add_subparsers(
  116. dest = 'command',
  117. )
  118. subparser_backup = subparsers.add_parser('backup')
  119. subparser_backup.add_argument(
  120. '--no-print-config',
  121. action = 'store_true',
  122. )
  123. subparser_backup.add_argument(
  124. '--no-print-statistics',
  125. action = 'store_true',
  126. )
  127. subparser_backup.add_argument(
  128. '--tab-dry',
  129. action = 'store_true',
  130. )
  131. return argparser
  132. def main(argv):
  133. argparser = _init_argparser()
  134. try:
  135. import argcomplete
  136. argcomplete.autocomplete(argparser)
  137. except ImportError:
  138. pass
  139. args = argparser.parse_args(argv)
  140. run(**vars(args))
  141. return 0
  142. if __name__ == "__main__":
  143. import sys
  144. sys.exit(main(sys.argv[1:]))