# Mobile Verification Toolkit (MVT) # Copyright (c) 2021-2022 Claudio Guarnieri. # Use of this software is governed by the MVT License 1.1 that can be found at # https://license.mvt.re/1.1/ import getpass import io import logging import os import sys import tarfile from pathlib import Path from mvt.android.parsers.backup import (AndroidBackupParsingError, InvalidBackupPassword, parse_ab_header, parse_backup_file) from mvt.common.command import Command from .modules.backup import BACKUP_MODULES log = logging.getLogger(__name__) class CmdAndroidCheckBackup(Command): name = "check-backup" modules = BACKUP_MODULES def __init__(self, target_path=None, results_path=None, ioc_files=[], module_name=None, serial=None, fast_mode=False): super().__init__(target_path=target_path, results_path=results_path, ioc_files=ioc_files, module_name=module_name, serial=serial, fast_mode=fast_mode, log=log) self.backup_type = None self.backup_archive = None self.backup_files = [] def init(self): if os.path.isfile(self.target_path): self.backup_type = "ab" with open(self.target_path, "rb") as handle: data = handle.read() header = parse_ab_header(data) if not header["backup"]: log.critical("Invalid backup format, file should be in .ab format") sys.exit(1) password = None if header["encryption"] != "none": password = getpass.getpass(prompt="Backup Password: ", stream=None) try: tardata = parse_backup_file(data, password=password) except InvalidBackupPassword: log.critical("Invalid backup password") sys.exit(1) except AndroidBackupParsingError as e: log.critical("Impossible to parse this backup file: %s", e) log.critical("Please use Android Backup Extractor (ABE) instead") sys.exit(1) dbytes = io.BytesIO(tardata) self.backup_archive = tarfile.open(fileobj=dbytes) for member in self.backup_archive: self.backup_files.append(member.name) elif os.path.isdir(self.target_path): self.backup_type = "folder" self.target_path = Path(self.target_path).absolute().as_posix() for root, subdirs, subfiles in os.walk(os.path.abspath(self.target_path)): for fname in subfiles: self.backup_files.append(os.path.relpath(os.path.join(root, fname), self.target_path)) else: log.critical("Invalid backup path, path should be a folder or an Android Backup (.ab) file") sys.exit(1) def module_init(self, module): if self.backup_type == "folder": module.from_folder(self.target_path, self.backup_files) else: module.from_ab(self.target_path, self.backup_archive, self.backup_files)