from pathlib import Path from pymobiledevice3.lockdown import LockdownClient from pymobiledevice3.cli.mounter import MobileImageMounterService from pymobiledevice3.cli.developer import DtSimulateLocation from pymobiledevice3.services.diagnostics import DiagnosticsService from pymobiledevice3.exceptions import PasscodeRequiredError import logging import requests import tempfile import zipfile from tqdm import tqdm from ispoof.utils import get_home_folder DEVELOPER_DISK_IMAGE_URL = 'https://github.com/pdso/DeveloperDiskImage/raw/master/{ios_version}/{ios_version}.zip' logger = logging.getLogger(__name__) def download_developer_disk_image(ios_version, directory): url = DEVELOPER_DISK_IMAGE_URL.format(ios_version=ios_version) with requests.get(url, stream=True) as r: r.raise_for_status() total_size_in_bytes = int(r.headers.get('content-length', 0)) with tqdm(total=total_size_in_bytes, unit='iB', unit_scale=True, dynamic_ncols=True) as progress_bar: with tempfile.NamedTemporaryFile('wb+') as f: for chunk in r.iter_content(chunk_size=8192): progress_bar.update(len(chunk)) f.write(chunk) zip_file = zipfile.ZipFile(f) zip_file.extractall(directory) class Device: def __init__(self): self.lockdown = LockdownClient() self.diagnostics = DiagnosticsService(self.lockdown) self.mounter = None self.location = None def mount_image(self): try: self.mounter = MobileImageMounterService(self.lockdown) except PasscodeRequiredError as e: logger.error(e) image_type = 'Developer' if not self.mounter.is_image_mounted(image_type): logger.debug("Image is not mounted yet") logger.debug('trying to figure out the best suited DeveloperDiskImage') version = self.lockdown.sanitized_ios_version image_dir = f'{get_home_folder()}/DevDiskImage/' image_path = f'{image_dir}/DeveloperDiskImage.dmg' signature = f'{image_path}.signature' developer_disk_image_dir = Path(image_path).parent if not developer_disk_image_dir.exists(): try: download_developer_disk_image(version, developer_disk_image_dir) except PermissionError: logger.error( f'DeveloperDiskImage could not be saved to path ({developer_disk_image_dir}). ' f'Please make sure your user has the necessary permissions') return image_path = Path(image_path) signature = Path(signature) image_path = image_path.read_bytes() signature = signature.read_bytes() self.mounter.upload_image(image_type, image_path, signature) self.mounter.mount(image_type, signature) logger.info('DeveloperDiskImage mounted successfully') else: logger.debug("Image is mounted.") def spoof_gps(self, destination): if self.location is None: self.location = DtSimulateLocation(self.lockdown) self.location.set(destination.latitude, destination.longitude) return True def stop_spoofing(self): self.location.clear() self.diagnostics.restart()