See NEWS.
pymobiledevice3
is a pure python3 implementation for working with iDevices (iPhone, etc...). This means this tool is
both architecture and platform generic and is supported and tested on:
- Windows
- Linux
- macOS
Main features include:
- Device discovery over bonjour
- TCP port forwarding
- Viewing syslog lines (including debug)
- Profile management
- Application management
- File system management (AFC)
- Crash reports management
- Network sniffing (PCAP)
- Firmware update
- Mounting images
- Notification listening and triggering (
notify_post()
api) - Querying and setting SpringBoard options
- Automating WebInspector features
- DeveloperDiskImage features:
- Taking screenshots
- Simulate locations
- Process management
- Sniffing KDebug messages (strace capabilities++)
- Process monitoring (
top
like) - Accessibility features
- Sniffing oslog which includes both syslog and signposts
- Backup
Install the last released version using pip
:
python3 -m pip install -U pymobiledevice3
Or install the latest version from sources:
git clone git@github.com:doronz88/pymobiledevice3.git
cd pymobiledevice3
python3 -m pip install -U -e .
You can also install auto-completion for all available sub-commands by adding the following into your ~/.zshrc
:
# python-click<8.0
eval "$(_PYMOBILEDEVICE3_COMPLETE=source_zsh pymobiledevice3)"
# python-click>=8.0
eval "$(_PYMOBILEDEVICE3_COMPLETE=zsh_source pymobiledevice3)"
Currently, openssl is explicitly required if using on older iOS version (<13).
On macOS:
brew install openssl
On Linux:
sudo apt install openssl
The CLI subcommands are divided roughly by the protocol layer used for interacting in the device. For example, all
features derived from the DeveloperDiskImage will be accessible from the developer
subcommand. This also means that every feature which isn't there won't require it.
This is the main CLI usage:
Usage: python -m pymobiledevice3 [OPTIONS] COMMAND [ARGS]...
Options:
-h, --help Show this message and exit.
Commands:
activation activation options
afc FileSystem utils
amfi amfi options
apps application options
backup2 backup utils
bonjour bonjour options
companion companion options
crash crash report options
developer developer options.
diagnostics diagnostics options
lockdown lockdown options
mounter mounter options
notification notification options
pcap sniff device traffic
power-assertion Create a power assertion (wraps...
processes processes cli
profile profile options
provision privision options
remote remote options
restore restore options
springboard springboard options
syslog syslog options
usbmux usbmuxd options
webinspector webinspector options
version get installed package version
You could also import the modules and use the API yourself:
from pymobiledevice3.remote.remote_service_discovery import RemoteServiceDiscoveryService
from pymobiledevice3.lockdown import create_using_usbmux
from pymobiledevice3.services.syslog import SyslogService
# Connecting via usbmuxd
lockdown = create_using_usbmux()
for line in SyslogService(service_provider=lockdown).watch():
# just print all syslog lines as is
print(line)
# Or via remoted (iOS>=17)
# First, create a tunnel using:
# $ sudo pymobiledevice3 remote start-tunnel
# You can of course implement it yourself by copying the same pieces of code from:
# https://github.com/doronz88/pymobiledevice3/blob/master/pymobiledevice3/cli/remote.py#L68
# Now you can simply connect to the created tunnel's host and port
host = 'fded:c26b:3d2f::1' # randomized
port = 65177 # randomized
with RemoteServiceDiscoveryService((host, port)) as rsd:
for line in SyslogService(service_provider=rsd).watch():
# just print all syslog lines as is
print(line)
NOTE: Currently, this is only supported on macOS
Starting at iOS 17.0, Apple introduced the new CoreDevice framework to work with iOS devices. This framework relies on the RemoteXPC protocol. In order to communicate with the developer services you'll be required to first create trusted tunnel as follows:
sudo python3 -m pymobiledevice3 remote start-tunnel
The root permissions are required since this will create a new TUN/TAP device which is a high privilege operation. The output should be something similar to:
Interface: utun6
RSD Address: fd7b:e5b:6f53::1
RSD Port: 64337
Use the follow connection option:
--rsd fd7b:e5b:6f53::1 64337
Now, (almost) all of pymobiledevice3 accept an additional --rsd
option for connecting to the service over this new
tunnel. You can now try to execute any of them as follows:
# Accessing the DVT services
python3 -m pymobiledevice3 developer dvt ls / --rsd fd7b:e5b:6f53::1 64337
# Or any of the "normal" ones
python3 -m pymobiledevice3 syslog live --rsd fd7b:e5b:6f53::1 64337
The Tunneld Server is responsible for automatically creating a QUIC tunnel for Remote Service Discovery (RSD) when a device is connected.
To start the Tunneld Server, use the following command (with root privileges):
sudo python3 -m pymobiledevice3 remote tunneld
Once the Tunneld Server is running, you can use it for RSD over the created QUIC tunnel.
To specify a device by its UDID:
python3 -m pymobiledevice3 remote rsd-info --tunnel UDID
To let Tunneld automatically select a device (if only one is connected):
python3 -m pymobiledevice3 remote rsd-info --tunnel ''
If no UDID is specified and multiple devices are connected, a prompt will appear for device selection.
A recorded example for using a variety of features can be viewed at: https://terminalizer.com/view/18920b405193
There is A LOT you may do on the device using pymobiledevice3
. This is just a TL;DR of some common operations:
- Listing connected devices:
pymobiledevice3 usbmux list
- Discover network devices using bonjour:
pymobiledevice3 bonjour browse
- View all syslog lines (including debug messages):
pymobiledevice3 syslog live
- Filter out only messages containing the word "SpringBoard":
pymobiledevice3 syslog live -m SpringBoard
- Restart device:
pymobiledevice3 diagnostics restart
- Pull all crash reports from device:
pymobiledevice3 crash pull /path/to/crashes
- Manage the media directory:
pymobiledevice3 afc shell
- List all installed applications and their details:
pymobiledevice3 apps list --user --system
- Create a TCP tunnel from your HOST to the device:
pymobiledevice3 usbmux forward HOST_PORT DEVICE_PORT
- Create a full backup of the device:
pymobiledevice3 backup2 backup --full DIRECTORY
- Restore a given backup:
pymobiledevice3 backup2 restore DIRECTORY
- The following will require Web Inspector feature to be turned on:
- Get interactive JavaScript shell on any open tab:
pymobiledevice3 webinspector js_shell
- List currently opened tabs is device's browser:
pymobiledevice3 webinspector opened-tabs
- The following will require also the Remote Automation feature to be turned on:
- Get interactive JavaScript shell on new remote automation tab:
pymobiledevice3 webinspector js_shell --automation
- Launch an automation session to view a given URL:
pymobiledevice3 webinspector launch URL
- Get a a selenium-like shell:
pymobiledevice3 webinspector shell
- Get interactive JavaScript shell on new remote automation tab:
- Get interactive JavaScript shell on any open tab:
- Mount DeveloperDiskImage (On iOS>=17.0, each command will require an additional
--rsd
option):pymobiledevice3 mounter auto-mount
- The following will assume the DeveloperDiskImage is already mounted:
- Simulate an
x y
location:pymobiledevice3 developer simulate-location set x y
- Or the following for iOS>=17.0:
pymobiledevice3 developer dvt simulate-location set --rsd HOST PORT -- x y
- Taking a screenshot from the device:
pymobiledevice3 developer dvt screenshot /path/to/screen.png
- View detailed process list (including ppid, uid, guid, sandboxed, etc...):
pymobiledevice3 developer dvt sysmon process single
- Sniffing oslog:
pymobiledevice3 developer dvt oslog
- Kill a process:
pymobiledevice3 developer dvt kill PID
- List files in a given directory (un-chrooted):
pymobiledevice3 developer dvt ls PATH
- Launch an app by its bundle name:
pymobiledevice3 developer dvt launch com.apple.mobilesafari
- Sniff all KDebug events to get an
strace
-like output:pymobiledevice3 developer dvt core-profile-session parse-live
- Sniff all KDebug events into a file for parsing later with tools such
as
pykdebugparser
,fs_usage
and so on...pymobiledevice3 developer dvt core-profile-session save FILENAME
- Get device extended information (kernel name, chipset, etc...):
pymobiledevice3 developer dvt device-information
- Monitor energy-consumption for a specific PID:
pymobiledevice3 developer dvt energy PID1 PID2 ...
- Simulate an
To understand the bits and bytes of the communication with lockdownd you are advised to take a look at this article:
This is the list of all the services from lockdownd
which we reversed and implemented API wrappers for. A click on
each will lead to each one's implementation, where you can learn more about.
com.apple.mobile.heartbeat
- Just a ping to
lockdownd
service. - Used to keep an active connection with
lockdownd
- Just a ping to
com.apple.mobileactivationd
- Activation services
com.apple.afc
- File access for
/var/mobile/Media
- Based on afcd's protocol
- File access for
com.apple.crashreportcopymobile
- File access for
/var/mobile/Library/Logs/CrashReports
- Based on afcd's protocol
- File access for
com.apple.pcapd
- Sniff device's network traffic
com.apple.syslog_relay
- Just streams syslog lines as raw strings
- For a more robust structural parsing, it's better to access the
com.apple.os_trace_relay
relay.
com.apple.os_trace_relay
- More extensive syslog monitoring
com.apple.mobile.diagnostics_relay
- General diagnostic tools
com.apple.mobile.notification_proxy
&com.apple.mobile.insecure_notification_proxy
- API wrapper for
notify_post()
¬ify_register_dispatch()
- API wrapper for
com.apple.crashreportmover
- Just trigger
crash_mover
to move all crash reports into crash directory
- Just trigger
com.apple.mobile.MCInstall
- Profile management (MDM)
com.apple.misagent
- Provisioning Profiles management
com.apple.companion_proxy
- Companion features (watches and etc.)
com.apple.mobilebackup2
- Local backup management
com.apple.mobile.assertion_agent
- Create power assertion to prevent different kinds of sleep
com.apple.springboardservices
- Play with device's button layout
com.apple.mobile.mobile_image_mounter
- Image mounter service (used for DeveloperDiskImage mounting)
com.apple.mobile.house_arrest
- Get AFC utils (file management per application bundle)
com.apple.mobile.installation_proxy
- Application management
com.apple.instruments.remoteserver
- Developer instrumentation service, iOS<14 (DeveloperDiskImage)
com.apple.instruments.remoteserver.DVTSecureSocketProxy
- Developer instrumentation service, iOS>=14 (DeveloperDiskImage)
com.apple.mobile.screenshotr
- Take screenshot into a PNG format (DeveloperDiskImage)
com.apple.accessibility.axAuditDaemon.remoteserver
- Accessibility features (DeveloperDiskImage)
com.apple.dt.simulatelocation
- Allows to simulate locations (DeveloperDiskImage)
com.apple.dt.fetchsymbols
- Allows fetching of
dyld
and dyld shared cache files (DeveloperDiskImage)
- Allows fetching of
com.apple.webinspector
- Used to debug WebViews
com.apple.amfi.lockdown
- Used to enable developer-mode
This is the list of services we haven't dedicated time in implementing. If you feel the need to use one of them or any other that is not listed in here, feel free to create us an issue request .
com.apple.idamd
- Allows settings the IDAM configuration (something to do with loading of AppleUSBDeviceAudioDevice)
com.apple.atc
- AirTraffic related
com.apple.atc2
com.apple.ait.aitd
- AirTraffic related
com.apple.mobile.file_relay
(Deprecated)- On older iOS versions (iOS <= 8), this was the main relay used for file operations, which was later replaced with AFC.
com.apple.mobilesync
com.apple.purpletestr
(Deprecated)com.apple.PurpleReverseProxy.Conn
- Something to do with tethering internet connection to restored devices
com.apple.PurpleReverseProxy.Ctrl
- Something to do with tethering internet connection to restored devices
com.apple.dt.remotepairingdeviced.lockdown
com.apple.commcenter.mobile-helper-cbupdateservice
com.apple.carkit.service
- Used to transfer data to accessories. Data is transferred using iAP2 packets.
com.apple.bluetooth.BTPacketLogger
com.apple.streaming_zip_conduit
- Another relay used to install IPAs
Every such subcommand may wrap several relay requests underneath. If you wish to try and play with some the relays yourself, you can run:
pymobiledevice3 lockdown service <service-name>
This will start an IPython shell where you already have the connection established using the client
variable and you
can send & receive messages.
# This shell allows you to communicate directly with every service layer behind the lockdownd daemon.
# For example, you can do the following:
client.send_plist({"Command": "DoSomething"})
# and view the reply
print(client.recv_plist())
# or just send raw message
client.sendall(b"hello")
# and view the result
print(client.recvall(20))
If you want to play with DTServiceHub
which lies behind the developer
options, you can also use:
pymobiledevice3 developer shell
To also get an IPython shell, which lets you call ObjC methods from the exported objects in the instruments' namespace like so:
# This shell allows you to send messages to the DVTSecureSocketProxy and receive answers easily.
# Generally speaking, each channel represents a group of actions.
# Calling actions is done using a selector and auxiliary (parameters).
# Receiving answers is done by getting a return value and seldom auxiliary (private / extra parameters).
# To see the available channels, type the following:
developer.supported_identifiers
# In order to send messages, you need to create a channel:
channel = developer.make_channel('com.apple.instruments.server.services.deviceinfo')
# After creating the channel you can call allowed selectors:
channel.runningProcesses()
# If an answer is expected, you can receive it using the receive method:
processes = channel.receive_plist()
# Sometimes the selector requires parameters, You can add them using MessageAux. For example lets kill a process:
channel = developer.make_channel('com.apple.instruments.server.services.processcontrol')
args = MessageAux().append_obj(80) # This will kill pid 80
channel.killPid_(args, expects_reply=False) # Killing a process doesn't require an answer.
# In some rare cases, you might want to receive the auxiliary and the selector return value.
# For that cases you can use the recv_plist method.
return_value, auxiliary = developer.recv_plist()
See CONTRIBUTING.
Please see misc