%PDF- %PDF-
Direktori : /lib/python3/dist-packages/duplicity/backends/ |
Current File : //lib/python3/dist-packages/duplicity/backends/giobackend.py |
# -*- Mode:Python; indent-tabs-mode:nil; tab-width:4; encoding:utf-8 -*- # # Copyright 2009 Michael Terry <mike@mterry.name> # # This file is part of duplicity. # # Duplicity is free software; you can redistribute it and/or modify it # under the terms of the GNU General Public License as published by the # Free Software Foundation; either version 2 of the License, or (at your # option) any later version. # # Duplicity is distributed in the hope that it will be useful, but # WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # General Public License for more details. # # You should have received a copy of the GNU General Public License # along with duplicity; if not, write to the Free Software Foundation, # Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA import atexit import os import signal import subprocess import duplicity.backend from duplicity import log from duplicity import util def ensure_dbus(): # GIO requires a dbus session bus which can start the gvfs daemons # when required. So we make sure that such a bus exists and that our # environment points to it. if "DBUS_SESSION_BUS_ADDRESS" not in os.environ: p = subprocess.Popen(["dbus-launch"], stdout=subprocess.PIPE, universal_newlines=True) output = p.communicate()[0].decode("utf8", errors="replace") lines = output.split("\n") for line in lines: parts = line.split("=", 1) if len(parts) == 2: if parts[0] == "DBUS_SESSION_BUS_PID": # cleanup at end atexit.register(os.kill, int(parts[1]), signal.SIGTERM) os.environ[parts[0]] = parts[1] class GIOBackend(duplicity.backend.Backend): """Use this backend when saving to a GIO URL. This is a bit of a meta-backend, in that it can handle multiple schemas. URLs look like schema://user@server/path. """ def __init__(self, parsed_url): from gi.repository import Gio # pylint: disable=import-error from gi.repository import GLib # pylint: disable=import-error class DupMountOperation(Gio.MountOperation): """A simple MountOperation that grabs the password from the environment or the user. """ def __init__(self, backend): Gio.MountOperation.__init__(self) self.backend = backend self.connect("ask-password", self.ask_password_cb) self.connect("ask-question", self.ask_question_cb) def ask_password_cb(self, *args, **kwargs): self.set_password(self.backend.get_password()) self.reply(Gio.MountOperationResult.HANDLED) def ask_question_cb(self, *args, **kwargs): # Obviously just always answering with the first choice is a naive # approach. But there's no easy way to allow for answering questions # in duplicity's typical run-from-cron mode with environment variables. # And only a couple gvfs backends ask questions: 'sftp' does about # new hosts and 'afc' does if the device is locked. 0 should be a # safe choice. self.set_choice(0) self.reply(Gio.MountOperationResult.HANDLED) duplicity.backend.Backend.__init__(self, parsed_url) ensure_dbus() self.remote_file = Gio.File.new_for_uri(parsed_url.url_string) # Now we make sure the location is mounted op = DupMountOperation(self) loop = GLib.MainLoop() self.remote_file.mount_enclosing_volume(Gio.MountMountFlags.NONE, op, None, self.__done_with_mount, loop) loop.run() # halt program until we're done mounting # Now make the directory if it doesn't exist if not self.remote_file.query_exists(): self.remote_file.make_directory_with_parents(None) def __done_with_mount(self, fileobj, result, loop): from gi.repository import Gio # pylint: disable=import-error from gi.repository import GLib # pylint: disable=import-error try: fileobj.mount_enclosing_volume_finish(result) except GLib.GError as e: # check for NOT_SUPPORTED because some schemas (e.g. file://) validly don't if e.code != Gio.IOErrorEnum.ALREADY_MOUNTED and e.code != Gio.IOErrorEnum.NOT_SUPPORTED: log.FatalError( _("Connection failed, please check your password: %s") % util.uexc(e), log.ErrorCode.connection_failed, ) loop.quit() def __copy_progress(self, *args, **kwargs): pass def __copy_file(self, source, target): from gi.repository import Gio # pylint: disable=import-error # Don't pass NOFOLLOW_SYMLINKS here. Some backends (e.g. google-drive:) # use symlinks internally for all files. In the normal course of # events, we never deal with symlinks anyway, just tarballs. source.copy(target, Gio.FileCopyFlags.OVERWRITE, None, self.__copy_progress, None) def _error_code(self, operation, e): from gi.repository import Gio # pylint: disable=import-error from gi.repository import GLib # pylint: disable=import-error if isinstance(e, GLib.GError): if e.code == Gio.IOErrorEnum.FAILED and operation == "delete": # Sometimes delete will return a generic failure on a file not # found (notably the FTP does that) return log.ErrorCode.backend_not_found elif e.code == Gio.IOErrorEnum.PERMISSION_DENIED: return log.ErrorCode.backend_permission_denied elif e.code == Gio.IOErrorEnum.NOT_FOUND: return log.ErrorCode.backend_not_found elif e.code == Gio.IOErrorEnum.NO_SPACE: return log.ErrorCode.backend_no_space def _put(self, source_path, remote_filename): from gi.repository import Gio # pylint: disable=import-error source_file = Gio.File.new_for_path(source_path.name) target_file = self.remote_file.get_child_for_display_name(os.fsdecode(remote_filename)) self.__copy_file(source_file, target_file) def _get(self, filename, local_path): from gi.repository import Gio # pylint: disable=import-error source_file = self.remote_file.get_child_for_display_name(os.fsdecode(filename)) target_file = Gio.File.new_for_path(local_path.name) self.__copy_file(source_file, target_file) def _list(self): from gi.repository import Gio # pylint: disable=import-error files = [] # We grab display name, rather than file name because some backends # (e.g. google-drive:) use filesystem-specific IDs as file names and # only expose the "normal" name as display names. We need the display # name, because we try to parse them. enum = self.remote_file.enumerate_children( Gio.FILE_ATTRIBUTE_STANDARD_DISPLAY_NAME, Gio.FileQueryInfoFlags.NONE, None ) info = enum.next_file(None) while info: files.append(info.get_display_name()) info = enum.next_file(None) return files def _delete(self, filename): target_file = self.remote_file.get_child_for_display_name(os.fsdecode(filename)) target_file.delete(None) def _query(self, filename): from gi.repository import Gio # pylint: disable=import-error target_file = self.remote_file.get_child_for_display_name(os.fsdecode(filename)) info = target_file.query_info(Gio.FILE_ATTRIBUTE_STANDARD_SIZE, Gio.FileQueryInfoFlags.NONE, None) return {"size": info.get_size()} duplicity.backend.register_backend_prefix("gio", GIOBackend)