Example #1
0
def test_decrypt_fail(os, open, GPG):
    """Unsuccessful decryption sets 'ok' attribute to False."""
    crypt_result = MagicMock()
    crypt_result.ok = False
    GPG.return_value.decrypt = crypt_result

    g = GPGOps(gpg_recipient='B183CAFE')
    result = g.decrypt('encryptedfile', 'plainfile', 'phrase')
    not ok_(result.ok)
Example #2
0
def test_encrypt_ok(open, GPG):
    """Successful encryption sets 'ok' attribute to True."""
    crypt_result = MagicMock()
    crypt_result.ok = True
    GPG.return_value.encrypt = crypt_result

    g = GPGOps(gpg_recipient='B183CAFE')
    result = g.encrypt('plainfile', 'encryptedfile')
    ok_(result.ok)
Example #3
0
def test_encrypt(gpg):
    g = GPGOps(gpg_recipient='B183CAFE')

    fsrc = tempfile.NamedTemporaryFile()
    plainfile = fsrc.name
    encryptedfile = 'encryptedfile'
    g.encrypt(plainfile, encryptedfile)

    eq_(g.gpg.encrypt.call_count, 1)
Example #4
0
def test_decrypt(os, gpg):
    g = GPGOps(gpg_recipient='B183CAFE')

    fsrc = tempfile.NamedTemporaryFile()
    encryptedfile = fsrc.name
    plainfile = 'plainfile'
    phrase = 'trustno1'
    g.decrypt(encryptedfile, plainfile, phrase)

    eq_(g.gpg.decrypt.call_count, 1)
Example #5
0
    def __init__(self, args, database=None):
        """Set program parameters and initialize GPG operations object."""
        if database is None:
            self.database = 'inventory.sqlite'
        else:
            self.database = database

        self.set_parameters(args)
        self.gpg = GPGOps(gpg_binary=self.gpg_binary,
                          gpg_recipient=self.public_id,
                          gpg_keyring=self.gpg_keyring)
Example #6
0
class DirEncryption(object):
    """DirEncryption encrypts and decrypts files between two directories.

    One directory serves as a repository of unencrypted files, and the
    other contains encrypted version of the same files.

    DirEncryption uses a SQLite database as a register of known files.
    Files from the source (unencrypted) directory will be encrypted
    into the destination directory if the modified time of a file
    is newer than the time of the last run.
    """
    def __init__(self, args, database=None):
        """Set program parameters and initialize GPG operations object."""
        if database is None:
            self.database = 'inventory.sqlite'
        else:
            self.database = database

        self.set_parameters(args)
        self.gpg = GPGOps(gpg_binary=self.gpg_binary,
                          gpg_recipient=self.public_id,
                          gpg_keyring=self.gpg_keyring)

    def set_parameters(self, args):
        """Set parameters based on database config and passed args."""
        self.verbose = False
        if args and args.verbose:
            self.verbose = True

        with Inventory(self.database) as i:
            parameters = i.read_parameters()
        for parameter, value in parameters.items():
            if self.verbose:
                printit('Parameters: {:<15} : {}', parameter, value)

        self.last_timestamp = parameters['last_timestamp']

        self.plaindir = os.path.expanduser(parameters['plaindir'])
        self.securedir = os.path.expanduser(parameters['securedir'])
        self.public_id = parameters['public_id']
        self.gpg_keyring = parameters['gpg_keyring']
        self.gpg_homedir = os.path.expanduser(parameters['gpg_homedir'])
        self.gpg_binary = os.path.expanduser(parameters['gpg_binary'])

        if args is None:
            return

        if args.plaindir:
            self.plaindir = os.path.expanduser(args.plaindir)
        if args.securedir:
            self.securedir = os.path.expanduser(args.securedir)
        if args.public_id:
            self.public_id = args.public_id
        if args.gpg_keyring:
            self.gpg_keyring = args.gpg_keyring
        if args.gpg_homedir:
            self.gpg_homedir = os.path.expanduser(args.gpg_homedir)
        if args.gpg_binary:
            self.gpg_binary = os.path.expanduser(args.gpg_binary)

    def encrypt_all(self):
        """Encrypt all new files from unencrypted directory.

        New files are those that have modified timestamp newer than
        the timestamp of the last run. At the start of run, the timestamp
        is updated.

        The files are recursively searched for in the source directory.
        """
        register = {}
        with Inventory(self.database) as inv:
            register = inv.read_register()
            inv.update_last_timestamp()
            files = self.find_unencrypted_files(register)
            for plainfile, val in files.items():
                if not val['is_new']:
                    # remove old file in secure directory
                    encfile = inv.read_line_from_register(plainfile)
                    FileOps.delete_file(self.securedir, encfile)
                encryptedfile = self.generate_name()
                self.encrypt(plainfile, encryptedfile, inv)
                if self.verbose:
                    printit('Encrypted: {} ---> {}', plainfile, encryptedfile)

    def encrypt(self, plainfile, encfile, inventory):
        """Encrypt the file and register input and output filenames."""
        plain_path = os.path.join(self.plaindir, plainfile)
        encrypted_path = os.path.join(self.securedir, encfile)

        self.gpg.encrypt(plain_path, encrypted_path)
        inventory.register(plainfile, encfile, self.public_id)

    def decrypt_all(self, passphrase):
        """Decrypt all files from encrypted source.

        Files that are being decrypted must be registered under the same
        public id in the database, so the passed passphrase would work
        for dectyption process.
        """
        register = {}
        with Inventory(self.database) as i:
            register = i.read_register()
            for filename, record in register.items():
                if record['public_id'] != self.public_id:
                    continue
                try:
                    self.decrypt(record['encrypted_file'],
                                 record['unencrypted_file'], passphrase)
                except IOError as e:
                    logging.warning('decrypt_all: {}'.format(e))

    def decrypt(self, encfile, plainfile, phrase):
        """Decrypt the file using a supplied passphrase."""
        encrypted_path = os.path.join(self.securedir, encfile)
        plain_path = os.path.join(self.plaindir, plainfile)
        if self.verbose:
            printit('Decrypt: {} ---> {}', encrypted_path, plain_path)
        self.gpg.decrypt(encrypted_path, plain_path, phrase)

    def find_unencrypted_files(self, register):
        """List all files that need to be encrypted.

        os.walk does not expand tilde in paths, so the walk directory
        is explicitly expanded.

        register is the currently known list of encrypted files.

        Returns a dict, with relative path of the unencrypted files
        for keys, and a dict with modified time key-value pair and
        is_new boolean flag for values.
        """
        files = {}
        if self.verbose:
            printit('Walking: {}', self.plaindir)

        for (dirpath, dirnames, filenames) in os.walk(self.plaindir):
            for name in filenames:
                filepath = os.path.join(dirpath, name)
                statinfo = os.stat(filepath)
                mtime = statinfo.st_mtime
                relative_path = filepath[(len(self.plaindir) + 1):]
                if relative_path not in register:
                    # new file
                    enc_flag = '*'
                    files[relative_path] = {
                        'modified_time': mtime,
                        'is_new': True
                    }
                elif relative_path in register and mtime > int(
                        self.last_timestamp):
                    # file exists and has changed since last run
                    enc_flag = '*'
                    files[relative_path] = {
                        'modified_time': mtime,
                        'is_new': False
                    }
                else:
                    # file is not changed since last run
                    enc_flag = ' '
                if self.verbose:
                    printit('List files: {} {} ({}): {}', enc_flag, int(mtime),
                            self.last_timestamp, relative_path)
        return files

    def generate_name(self):
        """Return a unique file name for encrypted file."""
        return str(uuid.uuid4())
Example #7
0
class DirEncryption(object):
    """DirEncryption encrypts and decrypts files between two directories.

    One directory serves as a repository of unencrypted files, and the
    other contains encrypted version of the same files.

    DirEncryption uses a SQLite database as a register of known files.
    Files from the source (unencrypted) directory will be encrypted
    into the destination directory if the modified time of a file
    is newer than the time of the last run.
    """
    def __init__(self, args, database=None):
        """Set program parameters and initialize GPG operations object."""
        if database is None:
            self.database = 'inventory.sqlite'
        else:
            self.database = database

        self.set_parameters(args)
        self.gpg = GPGOps(gpg_binary=self.gpg_binary,
                          gpg_recipient=self.public_id,
                          gpg_keyring=self.gpg_keyring)

    def set_parameters(self, args):
        """Set parameters based on database config and passed args."""
        self.verbose = False
        if args and args.verbose:
            self.verbose = True

        with Inventory(self.database) as i:
            parameters = i.read_parameters()
        for parameter, value in parameters.items():
            if self.verbose:
                print('Parameters: {:<15} : {}'.format(parameter, value))

        self.last_timestamp = parameters['last_timestamp']

        self.plaindir = os.path.expanduser(parameters['plaindir'])
        self.securedir = os.path.expanduser(parameters['securedir'])
        self.restoredir = os.path.expanduser(parameters['restoredir'])
        self.public_id = parameters['public_id']
        self.gpg_keyring = parameters['gpg_keyring']
        self.gpg_homedir = os.path.expanduser(parameters['gpg_homedir'])
        self.gpg_binary = os.path.expanduser(parameters['gpg_binary'])

        if args is None:
            return

        if args.plaindir:
            self.plaindir = os.path.expanduser(args.plaindir)
        if args.securedir:
            self.securedir = os.path.expanduser(args.securedir)
        if args.restoredir:
            self.restoredir = os.path.expanduser(args.restoredir)
        if args.public_id:
            self.public_id = args.public_id
        if args.gpg_keyring:
            self.gpg_keyring = args.gpg_keyring
        if args.gpg_homedir:
            self.gpg_homedir = os.path.expanduser(args.gpg_homedir)
        if args.gpg_binary:
            self.gpg_binary = os.path.expanduser(args.gpg_binary)

    def encrypt_all(self):
        """Encrypt all new files from unencrypted directory.

        New files are those that have modified timestamp newer than
        the timestamp of the last run. At the start of run, the timestamp
        is updated.

        The files are recursively searched for in the source directory.
        """
        register = {}
        print("Encrypting all directory '{}', please wait...".format(
            self.plaindir))
        with Inventory(self.database) as inv:
            register = inv.read_register("all")

            # treat regular files first
            self.encrypt_regular_files(register, inv)
            # then treat empty directories
            self.register_empty_dirs(register, inv)
            # finally treat symlinks
            self.register_symlinks(register, inv)

            self.do_inv_maintenance(inv)
            print("Done !")

    def encrypt_regular_files(self, register, inventory):
        """Encrypt all regular files."""
        files = self.find_unencrypted_files(register)
        for plainfile, val in files.items():
            if not val['is_new']:
                # remove old file in secure directory
                encfile = inventory.read_line_from_register(plainfile)
                FileOps.delete_file(self.securedir, encfile)
            encryptedfile = self.generate_name()
            encrypted_ok = self.encrypt(plainfile, encryptedfile, inventory)
            if encrypted_ok and self.verbose:
                print('Encrypted file: {} ---> {}'.format(
                    plainfile, encryptedfile))

    def register_empty_dirs(self, register, inventory):
        """Register all empty directories."""
        dirs = self.find_unregistered_empty_dirs(register)
        for dir_name, val in dirs.items():
            if not val['is_new']:
                # remove old dir in register
                inventory.clean_record(dir_name)
            self.register(dir_name, inventory, False)
            if self.verbose:
                print('Registered empty directory: {}'.format(dir_name))

    def register_symlinks(self, register, inventory):
        """Register all symlinks."""
        links = self.find_unregistered_links(register)
        for link_name, val in links.items():
            if not val['is_new']:
                # remove old link in register
                inventory.clean_record(link_name)
            self.register(link_name, inventory, True, val['target'])
            if self.verbose:
                print('Registered symlink: {} ---> {}'.format(
                    link_name, val['target']))

    def do_inv_maintenance(self, inventory):
        """Clean register and update timestamp."""
        self.clean(inventory)
        inventory.update_last_timestamp()

    def encrypt(self, plainfile, encfile, inventory, is_link=False):
        """Encrypt the file and register input and output filenames."""
        plain_path = os.path.join(self.plaindir, plainfile)
        encrypted_path = os.path.join(self.securedir, encfile)
        result = self.gpg.encrypt(plain_path, encrypted_path)
        if result.ok:
            inventory.register(plainfile, encfile, self.public_id, is_link, '')
        else:
            message = result.stderr.replace('\n', '\n\t')
            print('FAILED encryption of: {}\n\t{}'.format(plainfile, message))
        return result.ok

    def register(self, plainfile, inventory, is_link, link_target=None):
        """Register symlinks and empty directories."""
        inventory.register(plainfile, '', '', is_link, link_target)

    def clean(self, inv):
        """Clean register and file system

        If some files, links or empty directories have been
        deleted in the root directory, they must be removed
        from the register
        """

        print("Clean register and file system...")
        registered_files = inv.read_register("files")
        registered_links = inv.read_register("links")
        registered_dirs = inv.read_register("dirs")

        # Clean regular files
        for filename, record in registered_files.items():
            unenc_filename_path = os.path.join(self.plaindir, filename)
            enc_filename_path = os.path.join(self.securedir,
                                             record['encrypted_file'])
            if not os.path.isfile(unenc_filename_path):
                if os.path.isfile(enc_filename_path):
                    print(
                        "  --> Delete encrypted file {} and unregister".format(
                            record['encrypted_file']))
                    FileOps.delete_file(self.securedir,
                                        record['encrypted_file'])
                inv.clean_record(filename)

        # Clean symlinks
        for filename, record in registered_links.items():
            unenc_filename_path = os.path.join(self.plaindir, filename)
            if not os.path.islink(unenc_filename_path):
                print("  --> Unregister symlink {}".format(filename))
                inv.clean_record(filename)

        # Clean empty directories
        for filename, record in registered_dirs.items():
            unenc_filename_path = os.path.join(self.plaindir, filename)
            if not os.path.isdir(unenc_filename_path):
                print("  --> Unregister empty directory {}".format(filename))
                inv.clean_record(filename)
            elif os.path.isdir(unenc_filename_path):
                # unregister if the directory has been filled
                if os.listdir(unenc_filename_path):
                    print("  --> Unregister  non empty directory {}".format(
                        filename))
                    inv.clean_record(filename)

    def decrypt_all(self, passphrase):
        """Decrypt all files from encrypted source.

        Files that are being decrypted must be registered under the same
        public id in the database, so the passed passphrase would work
        for decryption process.
        """
        register = {}
        with Inventory(self.database) as i:
            register = i.read_register("all")
            for filename, record in register.items():
                # first decrypt regular files
                if record['is_link'] == 0 and record[
                        'encrypted_file'] and record['public_id']:
                    if record['public_id'] != self.public_id:
                        continue
                    try:
                        self.decrypt(record['encrypted_file'],
                                     record['unencrypted_file'], passphrase)
                    except IOError as e:
                        logging.warning('decrypt_all: {}'.format(e))
                        print('Failed to create file {} : {}'.format(
                            record['unencrypted_file'], str(e)))
                # then restore empty directories
                elif record['is_link'] == 0 and not record[
                        'encrypted_file'] and not record['public_id']:
                    FileOps.create_directory(self.restoredir,
                                             record['unencrypted_file'])
                # finally restore symlinks
                elif record['is_link'] == 1:
                    FileOps.create_symlink(self.restoredir,
                                           record['unencrypted_file'],
                                           record['target'])

    def decrypt(self, encfile, plainfile, phrase):
        """Decrypt the file using a supplied passphrase."""
        encrypted_path = os.path.join(self.securedir, encfile)
        restored_path = os.path.join(self.restoredir, plainfile)
        if self.verbose:
            print('Decrypt: {} ---> {}'.format(encrypted_path, restored_path))
        result = self.gpg.decrypt(encrypted_path, restored_path, phrase)
        if not result.ok:
            message = result.stderr.replace('\n', '\n\t')
            print('FAILED decryption of: {} to {}\n\t{}'.format(
                encfile, plainfile, message))
        return result.ok

    def find_unencrypted_files(self, register):
        """List all files that need to be encrypted.

        os.walk does not expand tilde in paths, so the walk directory
        is explicitly expanded.

        register is the currently known list of encrypted files.

        Returns a dict, with relative path of the unencrypted files
        for keys, and is_new boolean flag for values.
        """
        files = {}
        if self.verbose:
            print('Walking: {}'.format(self.plaindir))

        for (dirpath, dirnames, filenames) in os.walk(self.plaindir):
            for name in filenames:
                filepath = os.path.join(dirpath, name)
                if os.path.islink(filepath):
                    continue
                statinfo = os.stat(filepath)
                mtime = statinfo.st_mtime
                relative_path = filepath[(len(self.plaindir) + 1):]
                if relative_path not in register:
                    # new file
                    enc_flag = '*'
                    files[relative_path] = {'is_new': True}
                elif relative_path in register and mtime > int(
                        self.last_timestamp):
                    # file exists and has changed since last run
                    enc_flag = '*'
                    files[relative_path] = {'is_new': False}
                else:
                    # file has not changed since last run
                    enc_flag = ' '
                if self.verbose:
                    print('List files: {} {} ({}): {}'.format(
                        enc_flag, int(mtime), self.last_timestamp,
                        relative_path))
        return files

    def find_unregistered_links(self, register):
        """List all links that need to be registered.

        Returns a dict, with relative path of the unregistered links
        for keys, having a dict with target of the link and
        is_new boolean flag for values.
        """
        links = {}
        if self.verbose:
            print('Walking: {}'.format(self.plaindir))
        links_to_treat = list()
        # get symlinks to directories
        for (dirpath, dirnames, filenames) in os.walk(self.plaindir,
                                                      followlinks=True):
            if os.path.islink(dirpath):
                links_to_treat.append(dirpath)
        # get symlinks to files
        for (dirpath, dirnames, filenames) in os.walk(self.plaindir):
            for name in filenames:
                linkpath = os.path.join(dirpath, name)
                if not os.path.islink(linkpath):
                    continue
                links_to_treat.append(linkpath)
        for link in links_to_treat:
            # stat to the link and not his target
            statinfo = os.stat(link, follow_symlinks=False)
            mtime = statinfo.st_mtime
            linkto = os.readlink(link)
            relative_path = link[(len(self.plaindir) + 1):]
            if relative_path not in register:
                # new link
                enc_flag = '*'
                links[relative_path] = {'target': linkto, 'is_new': True}
            elif relative_path in register and mtime > int(
                    self.last_timestamp):
                # link exists and has changed since last run
                enc_flag = '*'
                links[relative_path] = {'target': linkto, 'is_new': False}
            else:
                # link has not changed since last run
                enc_flag = ' '
            if self.verbose:
                print('List links: {} {} ({}): {}'.format(
                    enc_flag, int(mtime), self.last_timestamp, relative_path))
        return links

    def find_unregistered_empty_dirs(self, register):
        """List all empty directories that need to be registered.

        Returns a dict, with relative path of the unregistered directories
        for keys, having a dict with is_new boolean flag for values.
        """
        result = {}
        if self.verbose:
            print('Walking: {}'.format(self.plaindir))
        empty_dirs = list()
        for (dirpath, dirnames, filenames) in os.walk(self.plaindir):
            if not dirnames and not filenames:
                empty_dirs.append(dirpath)
        if not empty_dirs:
            return result
        for dir in empty_dirs:
            statinfo = os.stat(dir)
            mtime = statinfo.st_mtime
            relative_path = dir[(len(self.plaindir) + 1):]
            if relative_path not in register:
                # new dir
                enc_flag = '*'
                result[relative_path] = {'is_new': True}
            elif relative_path in register and mtime > int(
                    self.last_timestamp):
                # dir exists and has changed since last run
                enc_flag = '*'
                result[relative_path] = {'is_new': False}
            else:
                # dir has not changed since last run
                enc_flag = ' '
            if self.verbose:
                print('List empty directories: {} {} ({}): {}'.format(
                    enc_flag, int(mtime), self.last_timestamp, relative_path))
        return result

    def generate_name(self):
        """Return a unique file name for encrypted file."""
        return str(uuid.uuid4())