#ios #libimobiledevice #developer-tools #lockdown

rusty_libimobiledevice

An ergonomic library to communicate with iOS devices

14 releases

new 0.2.3 Jan 7, 2025
0.2.2 Nov 15, 2024
0.2.0 Oct 3, 2024
0.1.11 Sep 20, 2024
0.1.2 Jul 7, 2022

#260 in Cryptography

Download history 224/week @ 2024-09-20 223/week @ 2024-09-27 61/week @ 2024-10-04 24/week @ 2024-10-11 8/week @ 2024-10-18 2/week @ 2024-10-25 4/week @ 2024-11-01 111/week @ 2024-11-08 123/week @ 2024-11-15 10/week @ 2024-11-22 21/week @ 2024-11-29 41/week @ 2024-12-06 4/week @ 2024-12-13 143/week @ 2025-01-03

166 downloads per month

LGPL-2.1-or-later

1MB
21K SLoC

rusty_libimobiledevice

Rusty Libimobiledevice - An ergonomic library to communicate with iOS devices.

Tested level: 0.00% - Contribute to this library by battle testing methods! All functions should be tested for possible segfaults and memory leaks.

If you see the Verified: False, that means that function needs your help to make sure it's safe. Open a PR with your testing code to change a function's status.

Building

Build and install the following packages:

Note: Package managers do not ship static libraries, so you will need to build them yourself if compiling statically.

To cross compile this crate, you can use the vendored feature and the build script will attempt to clone and build them for the specified target.

Usage

Add the crate and path to your cargo.toml, and add either static or dynamic to the features list. This will determine how the library is linked. By default this is dynamic. You can also use the vendored feature to build libimobiledevice at compile time.

Check the tools directory for full examples of how to use this library. It has many common use-cases.

To list devices detected by a usbmuxd daemon, you can use the following example.

// Include the idevice module. Will be needed in most scenarios.
use rusty_libimobiledevice::idevice;

fn main() {
    // Get all devices attatched from the usbmuxd daemon
    let devices = match idevice::get_devices() {
        Ok(devices) => devices,
        Err(e) => {
            // If the daemon is not running or does not behave as expected, this returns an error
            println!("Error getting devices: {:?}", e);
            return;
        }
    };

    // Devices support the display trait and can be viewed as such
    println!("Devices found: {:?}", devices);
}

More complicated code can skip fetching devices from usbmuxd and attach straight to a network device.

Services

This library implements methods for a handful of an iOS device's services These can be useful for manipulating functions on the device. For example, you can get a list of apps installed on a device using the following example

use rusty_libimobiledevice::idevice;

fn print_apps(udid: String) {
    // Get the device from usbmuxd using the given UDID
    let device = match idevice::get_device(udid.to_string()) {
        Ok(device) => device,
        Err(e) => {
            println!("Error: Could not find device: {:?}", e);
            return;
        }
    };

    // Start an instproxy service on the device
    let instproxy_client = match device.new_instproxy_client("idevicelistapps".to_string()) {
        Ok(instproxy) => {
            println!("Successfully started instproxy");
            instproxy
        }
        Err(e) => {
            println!("Error starting instproxy: {:?}", e);
            return;
        }
    };

    // Create a request to be sent using the service
    let mut client_opts = InstProxyClient::options_new();
    InstProxyClient::options_add(
        &mut client_opts,
        vec![("ApplicationType".to_string(), Plist::new_string("Any"))],
    );
    InstProxyClient::options_set_return_attributes(
        &mut client_opts,
        vec![
            "CFBundleIdentifier".to_string(),
            "CFBundleExecutable".to_string(),
            "Container".to_string(),
        ],
    );

    // Send the request and get the lookup results as a plist
    let lookup_results = match instproxy_client.lookup(vec![], client_opts) {
        Ok(apps) => {
            println!("Successfully looked up apps");
            apps
        }
        Err(e) => {
            println!("Error looking up apps: {:?}", e);
            return;
        }
    };

    println!("{}", lookup_results.to_string());
}

Dependencies

~2.2–5.5MB
~120K SLoC