* feat(springboard): add get_icon_state method for reading home screen layout Add get_icon_state() method to SpringBoardServicesClient that retrieves the current home screen icon layout from iOS devices. Features: - Read complete home screen layout including icon positions and folders - Support for optional formatVersion parameter - Works on all iOS versions (tested on iOS 18.7.3) - Comprehensive documentation with usage examples Note: This PR intentionally does NOT include set_icon_state() as that functionality is non-operational on iOS 18+ (see issue #62 for details). Tested on: - Device: iPhone 16,2 (iPhone 15 Pro) - iOS: 18.7.3 (Build 22H217) * feat(springboard): add set_icon_state method with date precision fix - Implement set_icon_state() to modify home screen layout - Implement set_icon_state_with_version() with format_version parameter - Add truncate_dates_to_seconds() to convert nanosecond precision dates to second precision - Fix iOS compatibility issue where high-precision dates were rejected - Successfully tested on iOS 18.7.3 (previously believed to be restricted) - Follows pymobiledevice3 implementation pattern * refactor(utils): extract truncate_dates_to_seconds to utils::plist module - Move date truncation logic from springboardservices to reusable utils::plist module - Add comprehensive unit tests for date truncation functionality - Add public API documentation for the utility function - This makes the date normalization logic available for other services that may need it * perf(springboard): normalize dates on read instead of write - Move date truncation from set_icon_state to get_icon_state - Eliminates unnecessary clone() operation in set_icon_state - Better performance when setting icon state multiple times - Cleaner API: data from get_icon_state is directly usable in set_icon_state - Users don't need to worry about date precision issues * refactor(springboard): address PR feedback - use Option<&str> and add error validation - Change format_version parameter from Option<String> to Option<&str> for consistency - Remove outdated iOS 18+ restriction comments since setIconState works on iOS 18+ - Add error validation to get_icon_state method similar to get_icon_pngdata - Update documentation to reflect accurate iOS compatibility * Fix cargo clippy warnings * Fix clippy warnings in plist.rs * Add springboard CLI commands --------- Co-authored-by: Jackson Coxson <jkcoxson@gmail.com>
idevice
A pure Rust library for interacting with iOS services. Inspired by libimobiledevice pymobiledevice3, and go-ios this library interfaces with lockdownd, usbmuxd, and RSD to perform actions on an iOS device that a Mac normally would.
For help and information, join the idevice Discord
State
IMPORTANT: Breaking changes will happen at each point release until 0.2.0.
Pin your Cargo.toml to a specific version to avoid breakage.
This library is in development and research stage. Releases are being published to crates.io for use in other projects, but the API and feature-set are far from final or even planned.
Why use this?
libimobiledevice is a groundbreaking library. Unfortunately, it hasn't been seriously updated in a long time, and does not support many modern iOS features.
Libraries such as pymobiledevice3 and go-ios have popped up to fill that gap, but both lacked the support I needed for embedding into applications and server programs. Python requires an interpreter, and Go's current ability to be embedded in other languages is lacking.
This library is currently used in popular apps such as
StikDebug,
CrossCode
and
Protokolle.
idevice has proven there is a need. It's currently deployed on tens of
thousands of devices, all across the world.
Features
To keep dependency bloat and compile time down, everything is contained in features.
| Feature | Description |
|---|---|
afc |
Apple File Conduit for file system access. |
amfi |
Apple mobile file integrity service |
core_device_proxy |
Start a secure tunnel to access protected services. |
crashreportcopymobile |
Copy crash reports. |
debug_proxy |
Send GDB commands to the device. |
diagnostics_relay |
Access device diagnostics information (IORegistry, MobileGestalt, battery, NAND, device control). |
dvt |
Access Apple developer tools (e.g. Instruments). |
heartbeat |
Maintain a heartbeat connection. |
house_arrest |
Manage files in app containers |
installation_proxy |
Manage app installation and uninstallation. |
springboardservices |
Control SpringBoard (e.g. UI interactions). Partial support. |
misagent |
Manage provisioning profiles on the device. |
mobilebackup2 |
Manage backups. |
mobile_image_mounter |
Manage DDI images. |
location_simulation |
Simulate GPS locations on the device. |
pair |
Pair the device. |
syslog_relay |
Relay system logs from the device |
tcp |
Connect to devices over TCP. |
tunnel_tcp_stack |
Naive in-process TCP stack for core_device_proxy. |
tss |
Make requests to Apple's TSS servers. Partial support. |
tunneld |
Interface with pymobiledevice3's tunneld. |
usbmuxd |
Connect using the usbmuxd daemon. |
xpc |
Access protected services via XPC over RSD. |
Planned/TODO
Finish the following:
- springboard
Implement the following:
- companion_proxy
- diagnostics
- mobilebackup2
- notification_proxy
- screenshot
- webinspector
As this project is done in my free time within my busy schedule, there is no ETA for any of these. Feel free to contribute or donate!
Usage
idevice is purposefully verbose to allow for powerful configurations. No size fits all, but effort is made to reduce boilerplate via providers.
// enable the usbmuxd feature
use idevice::{lockdown::LockdowndClient, IdeviceService};
use idevice::usbmuxd::{UsbmuxdAddr, UsbmuxdConnection},
#[tokio::main]
async fn main() {
// usbmuxd is Apple's daemon for connecting to devices over USB.
// We'll ask usbmuxd for a device
let mut usbmuxd = UsbmuxdConnection::default()
.await
.expect("Unable to connect to usbmuxd");
let devs = usbmuxd.get_devices().unwrap();
if devs.is_empty() {
eprintln!("No devices connected!");
return;
}
// Create a provider to automatically create connections to the device.
// Many services require opening multiple connections to get where you want.
let provider = devs[0].to_provider(UsbmuxdAddr::from_env_var().unwrap(), 0, "example-program")
// ``connect`` takes an object with the provider trait
let mut lockdown_client = match LockdowndClient::connect(&provider).await {
Ok(l) => l,
Err(e) => {
eprintln!("Unable to connect to lockdown: {e:?}");
return;
}
};
println!("{:?}", lockdown_client.get_value("ProductVersion").await);
println!(
"{:?}",
lockdown_client
.start_session(
&provider
.get_pairing_file()
.await
.expect("failed to get pairing file")
)
.await
);
println!("{:?}", lockdown_client.idevice.get_type().await.unwrap());
println!("{:#?}", lockdown_client.get_all_values().await);
}
More examples are in the tools crate and in the crate documentation.
FFI
For use in other languages, a small FFI crate has been created to start exposing
idevice. Example C programs can be found in the ffi/examples directory.
C++
"Hey wait a second, there's a lot of C++ code in this library!!" C++ bindings have been made for many of idevice's features. This allows smooth and safer usage in C++ and Swift codebases.
Technical Explanation
There are so many layers and protocols in this library, many stacked on top of one another. It's difficult to describe the magnitude that is Apple's interfaces.
I would recommend reading the DeepWiki explanations and overviews to get an idea of how this library and their associated protocols work. But a general overview is:
Lockdown
- A lockdown service is accessible via a port given by lockdown
- Lockdown is accessible by USB or TCP via TLS
- USB is accessible via usbmuxd
- usbmuxd is accessed through a unix socket
- That Unix socket has its own protocol
RemoteXPC/RSD
- An RSD service is discovered through a RemoteXPC handshake response
- RemoteXPC is transferred over non-compliant HTTP/2
- That HTTP/2 is accessed through an NCM USB interface or CoreDeviceProxy
- CoreDeviceProxy is a lockdown service, see above
This doesn't even touch RPPairing, which is still a mystery as of writing.
License
MIT