mirror of
https://github.com/jkcoxson/idevice.git
synced 2026-03-02 14:36:16 +01:00
Implement new library for tools
This commit is contained in:
34
Cargo.lock
generated
34
Cargo.lock
generated
@@ -225,6 +225,33 @@ version = "0.2.1"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724"
|
checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "clap"
|
||||||
|
version = "4.5.27"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "769b0145982b4b48713e01ec42d61614425f27b7058bda7180a3a41f30104796"
|
||||||
|
dependencies = [
|
||||||
|
"clap_builder",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "clap_builder"
|
||||||
|
version = "4.5.27"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1b26884eb4b57140e4d2d93652abfa49498b938b3c9179f9fc487b0acc3edad7"
|
||||||
|
dependencies = [
|
||||||
|
"anstream",
|
||||||
|
"anstyle",
|
||||||
|
"clap_lex",
|
||||||
|
"strsim",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "clap_lex"
|
||||||
|
version = "0.7.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f46ad14479a25103f283c0f10005961cf086d8dc42205bb44c46ac563475dca6"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "colorchoice"
|
name = "colorchoice"
|
||||||
version = "1.0.3"
|
version = "1.0.3"
|
||||||
@@ -679,6 +706,7 @@ dependencies = [
|
|||||||
name = "idevice-tools"
|
name = "idevice-tools"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"clap",
|
||||||
"env_logger",
|
"env_logger",
|
||||||
"idevice",
|
"idevice",
|
||||||
"log",
|
"log",
|
||||||
@@ -1229,6 +1257,12 @@ version = "1.2.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3"
|
checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "strsim"
|
||||||
|
version = "0.11.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "subtle"
|
name = "subtle"
|
||||||
version = "2.6.1"
|
version = "2.6.1"
|
||||||
|
|||||||
@@ -42,3 +42,4 @@ env_logger = { version = "0.11" }
|
|||||||
tun-rs = { version = "1.5", features = ["async"] }
|
tun-rs = { version = "1.5", features = ["async"] }
|
||||||
sha2 = { version = "0.10" }
|
sha2 = { version = "0.10" }
|
||||||
ureq = { version = "3" }
|
ureq = { version = "3" }
|
||||||
|
clap = { version = "4.5" }
|
||||||
|
|||||||
65
tools/src/common.rs
Normal file
65
tools/src/common.rs
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
// Jackson Coxson
|
||||||
|
// Common functions between tools
|
||||||
|
|
||||||
|
use std::{net::IpAddr, str::FromStr};
|
||||||
|
|
||||||
|
use idevice::{
|
||||||
|
pairing_file::PairingFile,
|
||||||
|
provider::{IdeviceProvider, TcpProvider},
|
||||||
|
usbmuxd::{UsbmuxdAddr, UsbmuxdConnection},
|
||||||
|
};
|
||||||
|
|
||||||
|
pub async fn get_provider(
|
||||||
|
udid: Option<&String>,
|
||||||
|
host: Option<&String>,
|
||||||
|
pairing_file: Option<&String>,
|
||||||
|
label: &str,
|
||||||
|
) -> Result<Box<dyn IdeviceProvider>, String> {
|
||||||
|
let provider: Box<dyn IdeviceProvider> = if udid.is_some() {
|
||||||
|
let udid = udid.unwrap();
|
||||||
|
let mut usbmuxd = UsbmuxdConnection::default()
|
||||||
|
.await
|
||||||
|
.expect("Unable to connect to usbmxud");
|
||||||
|
let dev = match usbmuxd.get_device(udid).await {
|
||||||
|
Ok(d) => d,
|
||||||
|
Err(e) => {
|
||||||
|
return Err(format!("Device not found: {e:?}"));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
Box::new(dev.to_provider(UsbmuxdAddr::default(), 0, label))
|
||||||
|
} else if host.is_some() && pairing_file.is_some() {
|
||||||
|
let host = match IpAddr::from_str(host.unwrap()) {
|
||||||
|
Ok(h) => h,
|
||||||
|
Err(e) => {
|
||||||
|
return Err(format!("Invalid host: {e:?}"));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let pairing_file = match PairingFile::read_from_file(pairing_file.unwrap()) {
|
||||||
|
Ok(p) => p,
|
||||||
|
Err(e) => {
|
||||||
|
return Err(format!("Unable to read pairing file: {e:?}"));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Box::new(TcpProvider {
|
||||||
|
addr: host,
|
||||||
|
pairing_file,
|
||||||
|
label: "ideviceinfo-jkcoxson".to_string(),
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
let mut usbmuxd = UsbmuxdConnection::default()
|
||||||
|
.await
|
||||||
|
.expect("Unable to connect to usbmxud");
|
||||||
|
let devs = match usbmuxd.get_devices().await {
|
||||||
|
Ok(d) => d,
|
||||||
|
Err(e) => {
|
||||||
|
return Err(format!("Unable to get devices from usbmuxd: {e:?}"));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if devs.is_empty() {
|
||||||
|
return Err("No devices connected!".to_string());
|
||||||
|
}
|
||||||
|
Box::new(devs[0].to_provider(UsbmuxdAddr::default(), 0, label))
|
||||||
|
};
|
||||||
|
Ok(provider)
|
||||||
|
}
|
||||||
@@ -1,92 +1,74 @@
|
|||||||
// Jackson Coxson
|
// Jackson Coxson
|
||||||
|
|
||||||
|
use clap::{Arg, Command};
|
||||||
use idevice::{
|
use idevice::{
|
||||||
core_device_proxy::{self},
|
core_device_proxy::{self},
|
||||||
lockdownd::{self, LockdowndClient},
|
IdeviceService,
|
||||||
pairing_file::PairingFile,
|
|
||||||
Idevice,
|
|
||||||
};
|
};
|
||||||
use tun_rs::AbstractDevice;
|
use tun_rs::AbstractDevice;
|
||||||
|
|
||||||
use std::{
|
mod common;
|
||||||
net::{Ipv4Addr, SocketAddrV4},
|
|
||||||
str::FromStr,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
env_logger::init();
|
env_logger::init();
|
||||||
let mut host = None;
|
let matches = Command::new("core_device_proxy_tun")
|
||||||
let mut pairing_file = None;
|
.about("Start a tunnel")
|
||||||
|
.arg(
|
||||||
|
Arg::new("host")
|
||||||
|
.long("host")
|
||||||
|
.value_name("HOST")
|
||||||
|
.help("IP address of the device"),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("pairing_file")
|
||||||
|
.long("pairing-file")
|
||||||
|
.value_name("PATH")
|
||||||
|
.help("Path to the pairing file"),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("udid")
|
||||||
|
.value_name("UDID")
|
||||||
|
.help("UDID of the device (overrides host/pairing file)")
|
||||||
|
.index(1),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("about")
|
||||||
|
.long("about")
|
||||||
|
.help("Show about information")
|
||||||
|
.action(clap::ArgAction::SetTrue),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("help")
|
||||||
|
.short('h')
|
||||||
|
.long("help")
|
||||||
|
.help("Show this help message")
|
||||||
|
.action(clap::ArgAction::SetTrue),
|
||||||
|
)
|
||||||
|
.get_matches();
|
||||||
|
|
||||||
// Loop through args
|
if matches.get_flag("about") {
|
||||||
let mut i = 0;
|
println!("core_device_proxy - Start a lockdown tunnel on the device");
|
||||||
while i < std::env::args().len() {
|
println!("Copyright (c) 2025 Jackson Coxson");
|
||||||
match std::env::args().nth(i).unwrap().as_str() {
|
|
||||||
"--host" => {
|
|
||||||
host = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
|
||||||
i += 2;
|
|
||||||
}
|
|
||||||
"--pairing-file" => {
|
|
||||||
pairing_file = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
|
||||||
i += 2;
|
|
||||||
}
|
|
||||||
"-h" | "--help" => {
|
|
||||||
println!("core_device_proxy_tun - start a tunnel");
|
|
||||||
println!("Usage:");
|
|
||||||
println!(" core_device_proxy_tun [options]");
|
|
||||||
println!("Options:");
|
|
||||||
println!(" --host <host>");
|
|
||||||
println!(" --pairing_file <path>");
|
|
||||||
println!(" -h, --help");
|
|
||||||
println!(" --about");
|
|
||||||
println!("\n\nSet RUST_LOG to info, debug, warn, error, or trace to see more logs. Default is error.");
|
|
||||||
std::process::exit(0);
|
|
||||||
}
|
|
||||||
"--about" => {
|
|
||||||
println!("ideviceinfo - get information from the idevice. Reimplementation of libimobiledevice's binary.");
|
|
||||||
println!("Copyright (c) 2025 Jackson Coxson");
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
i += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if host.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the IP of the device with --host");
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if pairing_file.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the path the the pairing file with --pairing-file");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
let ip = Ipv4Addr::from_str(host.unwrap().as_str()).unwrap();
|
|
||||||
let socket = SocketAddrV4::new(ip, lockdownd::LOCKDOWND_PORT);
|
|
||||||
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
let udid = matches.get_one::<String>("udid");
|
||||||
let socket = Box::new(socket);
|
let host = matches.get_one::<String>("host");
|
||||||
let idevice = Idevice::new(socket, "heartbeat_client");
|
let pairing_file = matches.get_one::<String>("pairing_file");
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.as_ref().unwrap()).unwrap();
|
let provider =
|
||||||
|
match common::get_provider(udid, host, pairing_file, "core_device_proxy-jkcoxson").await {
|
||||||
|
Ok(p) => p,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("{e}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let mut lockdown_client = LockdowndClient { idevice };
|
let mut tun_proxy = core_device_proxy::CoreDeviceProxy::connect(&*provider)
|
||||||
lockdown_client.start_session(&p).await.unwrap();
|
|
||||||
|
|
||||||
let (port, _) = lockdown_client
|
|
||||||
.start_service(core_device_proxy::SERVCE_NAME)
|
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.expect("Unable to connect");
|
||||||
|
|
||||||
let socket = SocketAddrV4::new(ip, port);
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
|
||||||
let socket = Box::new(socket);
|
|
||||||
let mut idevice = Idevice::new(socket, "core_device_proxy_tun");
|
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.unwrap()).unwrap();
|
|
||||||
|
|
||||||
idevice.start_session(&p).await.unwrap();
|
|
||||||
|
|
||||||
let mut tun_proxy = core_device_proxy::CoreDeviceProxy::new(idevice);
|
|
||||||
let response = tun_proxy.establish_tunnel().await.unwrap();
|
let response = tun_proxy.establish_tunnel().await.unwrap();
|
||||||
|
|
||||||
let dev = tun_rs::create(&tun_rs::Configuration::default()).unwrap();
|
let dev = tun_rs::create(&tun_rs::Configuration::default()).unwrap();
|
||||||
|
|||||||
@@ -1,92 +1,70 @@
|
|||||||
// Jackson Coxson
|
// Jackson Coxson
|
||||||
// Heartbeat client
|
// Heartbeat client
|
||||||
|
|
||||||
use idevice::{
|
use clap::{Arg, Command};
|
||||||
heartbeat::HeartbeatClient,
|
use idevice::{heartbeat::HeartbeatClient, IdeviceService};
|
||||||
lockdownd::{self, LockdowndClient},
|
|
||||||
pairing_file::PairingFile,
|
|
||||||
Idevice,
|
|
||||||
};
|
|
||||||
|
|
||||||
use std::{
|
mod common;
|
||||||
net::{Ipv4Addr, SocketAddrV4},
|
|
||||||
str::FromStr,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
env_logger::init();
|
env_logger::init();
|
||||||
let mut host = None;
|
let matches = Command::new("core_device_proxy_tun")
|
||||||
let mut pairing_file = None;
|
.about("Start a tunnel")
|
||||||
|
.arg(
|
||||||
|
Arg::new("host")
|
||||||
|
.long("host")
|
||||||
|
.value_name("HOST")
|
||||||
|
.help("IP address of the device"),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("pairing_file")
|
||||||
|
.long("pairing-file")
|
||||||
|
.value_name("PATH")
|
||||||
|
.help("Path to the pairing file"),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("udid")
|
||||||
|
.value_name("UDID")
|
||||||
|
.help("UDID of the device (overrides host/pairing file)")
|
||||||
|
.index(1),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("about")
|
||||||
|
.long("about")
|
||||||
|
.help("Show about information")
|
||||||
|
.action(clap::ArgAction::SetTrue),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("help")
|
||||||
|
.short('h')
|
||||||
|
.long("help")
|
||||||
|
.help("Show this help message")
|
||||||
|
.action(clap::ArgAction::SetTrue),
|
||||||
|
)
|
||||||
|
.get_matches();
|
||||||
|
|
||||||
// Loop through args
|
if matches.get_flag("about") {
|
||||||
let mut i = 0;
|
println!("heartbeat_client - heartbeat a device");
|
||||||
while i < std::env::args().len() {
|
println!("Copyright (c) 2025 Jackson Coxson");
|
||||||
match std::env::args().nth(i).unwrap().as_str() {
|
|
||||||
"--host" => {
|
|
||||||
host = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
|
||||||
i += 2;
|
|
||||||
}
|
|
||||||
"--pairing-file" => {
|
|
||||||
pairing_file = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
|
||||||
i += 2;
|
|
||||||
}
|
|
||||||
"-h" | "--help" => {
|
|
||||||
println!("ideviceinfo - get information from the idevice");
|
|
||||||
println!("Usage:");
|
|
||||||
println!(" ideviceinfo [options]");
|
|
||||||
println!("Options:");
|
|
||||||
println!(" --host <host>");
|
|
||||||
println!(" --pairing_file <path>");
|
|
||||||
println!(" -h, --help");
|
|
||||||
println!(" --about");
|
|
||||||
println!("\n\nSet RUST_LOG to info, debug, warn, error, or trace to see more logs. Default is error.");
|
|
||||||
std::process::exit(0);
|
|
||||||
}
|
|
||||||
"--about" => {
|
|
||||||
println!("ideviceinfo - get information from the idevice. Reimplementation of libimobiledevice's binary.");
|
|
||||||
println!("Copyright (c) 2025 Jackson Coxson");
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
i += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if host.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the IP of the device with --host");
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if pairing_file.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the path the the pairing file with --pairing-file");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
let ip = Ipv4Addr::from_str(host.unwrap().as_str()).unwrap();
|
|
||||||
let socket = SocketAddrV4::new(ip, lockdownd::LOCKDOWND_PORT);
|
|
||||||
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
let udid = matches.get_one::<String>("udid");
|
||||||
let socket = Box::new(socket);
|
let host = matches.get_one::<String>("host");
|
||||||
let idevice = Idevice::new(socket, "heartbeat_client");
|
let pairing_file = matches.get_one::<String>("pairing_file");
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.as_ref().unwrap()).unwrap();
|
let provider =
|
||||||
|
match common::get_provider(udid, host, pairing_file, "heartbeat_client-jkcoxson").await {
|
||||||
let mut lockdown_client = LockdowndClient { idevice };
|
Ok(p) => p,
|
||||||
lockdown_client.start_session(&p).await.unwrap();
|
Err(e) => {
|
||||||
|
eprintln!("{e}");
|
||||||
let (port, _) = lockdown_client
|
return;
|
||||||
.start_service("com.apple.mobile.heartbeat")
|
}
|
||||||
|
};
|
||||||
|
let mut heartbeat_client = HeartbeatClient::connect(&*provider)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.expect("Unable to connect to heartbeat");
|
||||||
|
|
||||||
let socket = SocketAddrV4::new(ip, port);
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
|
||||||
let socket = Box::new(socket);
|
|
||||||
let mut idevice = Idevice::new(socket, "heartbeat_client");
|
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.unwrap()).unwrap();
|
|
||||||
|
|
||||||
idevice.start_session(&p).await.unwrap();
|
|
||||||
|
|
||||||
let mut heartbeat_client = HeartbeatClient { idevice };
|
|
||||||
|
|
||||||
let mut interval = 15;
|
let mut interval = 15;
|
||||||
loop {
|
loop {
|
||||||
|
|||||||
@@ -1,72 +1,77 @@
|
|||||||
// Jackson Coxson
|
// Jackson Coxson
|
||||||
// idevice Rust implementation of libimobiledevice's ideviceinfo
|
// idevice Rust implementation of libimobiledevice's ideviceinfo
|
||||||
|
|
||||||
use std::{
|
use clap::{Arg, Command};
|
||||||
net::{Ipv4Addr, SocketAddrV4},
|
use idevice::{lockdownd::LockdowndClient, pairing_file::PairingFile, IdeviceService};
|
||||||
str::FromStr,
|
|
||||||
};
|
|
||||||
|
|
||||||
use idevice::{
|
mod common;
|
||||||
lockdownd::{self, LockdowndClient},
|
|
||||||
pairing_file::PairingFile,
|
|
||||||
Idevice,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
env_logger::init();
|
env_logger::init();
|
||||||
let mut host = None;
|
|
||||||
let mut pairing_file = None;
|
|
||||||
|
|
||||||
// Loop through args
|
let matches = Command::new("core_device_proxy_tun")
|
||||||
let mut i = 0;
|
.about("Start a tunnel")
|
||||||
while i < std::env::args().len() {
|
.arg(
|
||||||
match std::env::args().nth(i).unwrap().as_str() {
|
Arg::new("host")
|
||||||
"--host" => {
|
.long("host")
|
||||||
host = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
.value_name("HOST")
|
||||||
i += 2;
|
.help("IP address of the device"),
|
||||||
}
|
)
|
||||||
"--pairing-file" => {
|
.arg(
|
||||||
pairing_file = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
Arg::new("pairing_file")
|
||||||
i += 2;
|
.long("pairing-file")
|
||||||
}
|
.value_name("PATH")
|
||||||
"-h" | "--help" => {
|
.help("Path to the pairing file"),
|
||||||
println!("ideviceinfo - get information from the idevice");
|
)
|
||||||
println!("Usage:");
|
.arg(
|
||||||
println!(" ideviceinfo [options]");
|
Arg::new("udid")
|
||||||
println!("Options:");
|
.value_name("UDID")
|
||||||
println!(" --host <host>");
|
.help("UDID of the device (overrides host/pairing file)")
|
||||||
println!(" --pairing_file <path>");
|
.index(1),
|
||||||
println!(" -h, --help");
|
)
|
||||||
println!(" --about");
|
.arg(
|
||||||
println!("\n\nSet RUST_LOG to info, debug, warn, error, or trace to see more logs. Default is error.");
|
Arg::new("about")
|
||||||
std::process::exit(0);
|
.long("about")
|
||||||
}
|
.help("Show about information")
|
||||||
"--about" => {
|
.action(clap::ArgAction::SetTrue),
|
||||||
println!("ideviceinfo - get information from the idevice. Reimplementation of libimobiledevice's binary.");
|
)
|
||||||
println!("Copyright (c) 2025 Jackson Coxson");
|
.arg(
|
||||||
}
|
Arg::new("help")
|
||||||
_ => {
|
.short('h')
|
||||||
i += 1;
|
.long("help")
|
||||||
|
.help("Show this help message")
|
||||||
|
.action(clap::ArgAction::SetTrue),
|
||||||
|
)
|
||||||
|
.get_matches();
|
||||||
|
|
||||||
|
if matches.get_flag("about") {
|
||||||
|
println!("ideviceinfo - get information from the idevice. Reimplementation of libimobiledevice's binary.");
|
||||||
|
println!("Copyright (c) 2025 Jackson Coxson");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let udid = matches.get_one::<String>("udid");
|
||||||
|
let host = matches.get_one::<String>("host");
|
||||||
|
let pairing_file = matches.get_one::<String>("pairing_file");
|
||||||
|
|
||||||
|
let provider =
|
||||||
|
match common::get_provider(udid, host, pairing_file, "ideviceinfo-jkcoxson").await {
|
||||||
|
Ok(p) => p,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("{e}");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut lockdown_client = match LockdowndClient::connect(&*provider).await {
|
||||||
|
Ok(l) => l,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Unable to connect to lockdown: {e:?}");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
if host.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the IP of the device with --host");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if pairing_file.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the path the the pairing file with --pairing-file");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
let ip = Ipv4Addr::from_str(host.unwrap().as_str()).unwrap();
|
|
||||||
let socket = SocketAddrV4::new(ip, lockdownd::LOCKDOWND_PORT);
|
|
||||||
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
|
||||||
let socket = Box::new(socket);
|
|
||||||
let idevice = Idevice::new(socket, "ideviceinfo-jkcoxson");
|
|
||||||
|
|
||||||
let mut lockdown_client = LockdowndClient::new(idevice);
|
|
||||||
println!("{:?}", lockdown_client.get_value("ProductVersion").await);
|
println!("{:?}", lockdown_client.get_value("ProductVersion").await);
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.unwrap()).unwrap();
|
let p = PairingFile::read_from_file(pairing_file.unwrap()).unwrap();
|
||||||
|
|||||||
@@ -1,92 +1,72 @@
|
|||||||
// Jackson Coxson
|
// Jackson Coxson
|
||||||
// Just lists apps for now
|
// Just lists apps for now
|
||||||
|
|
||||||
use idevice::{
|
use clap::{Arg, Command};
|
||||||
installation_proxy::InstallationProxyClient,
|
use idevice::{installation_proxy::InstallationProxyClient, IdeviceService};
|
||||||
lockdownd::{self, LockdowndClient},
|
|
||||||
pairing_file::PairingFile,
|
|
||||||
Idevice,
|
|
||||||
};
|
|
||||||
|
|
||||||
use std::{
|
mod common;
|
||||||
net::{Ipv4Addr, SocketAddrV4},
|
|
||||||
str::FromStr,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
env_logger::init();
|
env_logger::init();
|
||||||
let mut host = None;
|
|
||||||
let mut pairing_file = None;
|
|
||||||
|
|
||||||
// Loop through args
|
let matches = Command::new("core_device_proxy_tun")
|
||||||
let mut i = 0;
|
.about("Start a tunnel")
|
||||||
while i < std::env::args().len() {
|
.arg(
|
||||||
match std::env::args().nth(i).unwrap().as_str() {
|
Arg::new("host")
|
||||||
"--host" => {
|
.long("host")
|
||||||
host = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
.value_name("HOST")
|
||||||
i += 2;
|
.help("IP address of the device"),
|
||||||
}
|
)
|
||||||
"--pairing-file" => {
|
.arg(
|
||||||
pairing_file = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
Arg::new("pairing_file")
|
||||||
i += 2;
|
.long("pairing-file")
|
||||||
}
|
.value_name("PATH")
|
||||||
"-h" | "--help" => {
|
.help("Path to the pairing file"),
|
||||||
println!("ideviceinfo - get information from the idevice");
|
)
|
||||||
println!("Usage:");
|
.arg(
|
||||||
println!(" ideviceinfo [options]");
|
Arg::new("udid")
|
||||||
println!("Options:");
|
.value_name("UDID")
|
||||||
println!(" --host <host>");
|
.help("UDID of the device (overrides host/pairing file)")
|
||||||
println!(" --pairing_file <path>");
|
.index(1),
|
||||||
println!(" -h, --help");
|
)
|
||||||
println!(" --about");
|
.arg(
|
||||||
println!("\n\nSet RUST_LOG to info, debug, warn, error, or trace to see more logs. Default is error.");
|
Arg::new("about")
|
||||||
std::process::exit(0);
|
.long("about")
|
||||||
}
|
.help("Show about information")
|
||||||
"--about" => {
|
.action(clap::ArgAction::SetTrue),
|
||||||
println!("ideviceinfo - get information from the idevice. Reimplementation of libimobiledevice's binary.");
|
)
|
||||||
println!("Copyright (c) 2025 Jackson Coxson");
|
.arg(
|
||||||
}
|
Arg::new("help")
|
||||||
_ => {
|
.short('h')
|
||||||
i += 1;
|
.long("help")
|
||||||
}
|
.help("Show this help message")
|
||||||
}
|
.action(clap::ArgAction::SetTrue),
|
||||||
}
|
)
|
||||||
if host.is_none() {
|
.get_matches();
|
||||||
println!("Invalid arguments! Pass the IP of the device with --host");
|
|
||||||
|
if matches.get_flag("about") {
|
||||||
|
println!("instproxy - query and manage apps installed on a device. Reimplementation of libimobiledevice's binary.");
|
||||||
|
println!("Copyright (c) 2025 Jackson Coxson");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if pairing_file.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the path the the pairing file with --pairing-file");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
let ip = Ipv4Addr::from_str(host.unwrap().as_str()).unwrap();
|
|
||||||
let socket = SocketAddrV4::new(ip, lockdownd::LOCKDOWND_PORT);
|
|
||||||
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
let udid = matches.get_one::<String>("udid");
|
||||||
let socket = Box::new(socket);
|
let host = matches.get_one::<String>("host");
|
||||||
let idevice = Idevice::new(socket, "heartbeat_client");
|
let pairing_file = matches.get_one::<String>("pairing_file");
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.as_ref().unwrap()).unwrap();
|
let provider =
|
||||||
|
match common::get_provider(udid, host, pairing_file, "ideviceinfo-jkcoxson").await {
|
||||||
|
Ok(p) => p,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("{e}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let mut lockdown_client = LockdowndClient { idevice };
|
let mut instproxy_client = InstallationProxyClient::connect(&*provider)
|
||||||
lockdown_client.start_session(&p).await.unwrap();
|
|
||||||
|
|
||||||
let (port, _) = lockdown_client
|
|
||||||
.start_service("com.apple.mobile.installation_proxy")
|
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.expect("Unable to connect to instproxy");
|
||||||
|
|
||||||
let socket = SocketAddrV4::new(ip, port);
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
|
||||||
let socket = Box::new(socket);
|
|
||||||
let mut idevice = Idevice::new(socket, "instproxy-client");
|
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.unwrap()).unwrap();
|
|
||||||
|
|
||||||
idevice.start_session(&p).await.unwrap();
|
|
||||||
|
|
||||||
let mut instproxy_client = InstallationProxyClient::new(idevice);
|
|
||||||
let apps = instproxy_client.get_apps(None, None).await.unwrap();
|
let apps = instproxy_client.get_apps(None, None).await.unwrap();
|
||||||
for app in apps.keys() {
|
for app in apps.keys() {
|
||||||
println!("{app}");
|
println!("{app}");
|
||||||
|
|||||||
@@ -1,94 +1,75 @@
|
|||||||
// Jackson Coxson
|
// Jackson Coxson
|
||||||
// Just lists apps for now
|
// Just lists apps for now
|
||||||
|
|
||||||
use idevice::{
|
use clap::{Arg, Command};
|
||||||
lockdownd::{self, LockdowndClient},
|
use idevice::{mounter::ImageMounter, IdeviceService};
|
||||||
mounter::ImageMounter,
|
|
||||||
pairing_file::PairingFile,
|
|
||||||
Idevice,
|
|
||||||
};
|
|
||||||
|
|
||||||
use sha2::{Digest, Sha384};
|
use sha2::{Digest, Sha384};
|
||||||
|
|
||||||
use std::{
|
mod common;
|
||||||
net::{Ipv4Addr, SocketAddrV4},
|
|
||||||
str::FromStr,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
env_logger::init();
|
env_logger::init();
|
||||||
let mut host = None;
|
|
||||||
let mut pairing_file = None;
|
|
||||||
|
|
||||||
// Loop through args
|
let matches = Command::new("core_device_proxy_tun")
|
||||||
let mut i = 0;
|
.about("Start a tunnel")
|
||||||
while i < std::env::args().len() {
|
.arg(
|
||||||
match std::env::args().nth(i).unwrap().as_str() {
|
Arg::new("host")
|
||||||
"--host" => {
|
.long("host")
|
||||||
host = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
.value_name("HOST")
|
||||||
i += 2;
|
.help("IP address of the device"),
|
||||||
}
|
)
|
||||||
"--pairing-file" => {
|
.arg(
|
||||||
pairing_file = Some(std::env::args().nth(i + 1).unwrap().to_string());
|
Arg::new("pairing_file")
|
||||||
i += 2;
|
.long("pairing-file")
|
||||||
}
|
.value_name("PATH")
|
||||||
"-h" | "--help" => {
|
.help("Path to the pairing file"),
|
||||||
println!("ideviceinfo - get information from the idevice");
|
)
|
||||||
println!("Usage:");
|
.arg(
|
||||||
println!(" ideviceinfo [options]");
|
Arg::new("udid")
|
||||||
println!("Options:");
|
.value_name("UDID")
|
||||||
println!(" --host <host>");
|
.help("UDID of the device (overrides host/pairing file)")
|
||||||
println!(" --pairing_file <path>");
|
.index(1),
|
||||||
println!(" -h, --help");
|
)
|
||||||
println!(" --about");
|
.arg(
|
||||||
println!("\n\nSet RUST_LOG to info, debug, warn, error, or trace to see more logs. Default is error.");
|
Arg::new("about")
|
||||||
|
.long("about")
|
||||||
|
.help("Show about information")
|
||||||
|
.action(clap::ArgAction::SetTrue),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("help")
|
||||||
|
.short('h')
|
||||||
|
.long("help")
|
||||||
|
.help("Show this help message")
|
||||||
|
.action(clap::ArgAction::SetTrue),
|
||||||
|
)
|
||||||
|
.get_matches();
|
||||||
|
|
||||||
|
if matches.get_flag("about") {
|
||||||
|
println!("mounter - query and manage images mounted on a device. Reimplementation of libimobiledevice's binary.");
|
||||||
|
println!("Copyright (c) 2025 Jackson Coxson");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let udid = matches.get_one::<String>("udid");
|
||||||
|
let host = matches.get_one::<String>("host");
|
||||||
|
let pairing_file = matches.get_one::<String>("pairing_file");
|
||||||
|
|
||||||
|
let provider =
|
||||||
|
match common::get_provider(udid, host, pairing_file, "ideviceinfo-jkcoxson").await {
|
||||||
|
Ok(p) => p,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("{e}");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
"--about" => {
|
};
|
||||||
println!("mounter - mount developer disk images on the device. Reimplementation of pymobiledevice3's program.");
|
|
||||||
println!("Copyright (c) 2025 Jackson Coxson");
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
i += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if host.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the IP of the device with --host");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if pairing_file.is_none() {
|
|
||||||
println!("Invalid arguments! Pass the path the the pairing file with --pairing-file");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
let ip = Ipv4Addr::from_str(host.unwrap().as_str()).unwrap();
|
|
||||||
let socket = SocketAddrV4::new(ip, lockdownd::LOCKDOWND_PORT);
|
|
||||||
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
let mut mounter_client = ImageMounter::connect(&*provider)
|
||||||
let socket = Box::new(socket);
|
|
||||||
let idevice = Idevice::new(socket, "mounter_client");
|
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.as_ref().unwrap()).unwrap();
|
|
||||||
|
|
||||||
let mut lockdown_client = LockdowndClient { idevice };
|
|
||||||
lockdown_client.start_session(&p).await.unwrap();
|
|
||||||
|
|
||||||
let (port, _) = lockdown_client
|
|
||||||
.start_service("com.apple.mobile.mobile_image_mounter")
|
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.expect("Unable to connect to image mounter");
|
||||||
|
|
||||||
let socket = SocketAddrV4::new(ip, port);
|
|
||||||
let socket = tokio::net::TcpStream::connect(socket).await.unwrap();
|
|
||||||
let socket = Box::new(socket);
|
|
||||||
let mut idevice = Idevice::new(socket, "mounter_client");
|
|
||||||
|
|
||||||
let p = PairingFile::read_from_file(pairing_file.unwrap()).unwrap();
|
|
||||||
|
|
||||||
idevice.start_session(&p).await.unwrap();
|
|
||||||
|
|
||||||
let mut mounter_client = ImageMounter::new(idevice);
|
|
||||||
let images = mounter_client.copy_devices().await.unwrap();
|
let images = mounter_client.copy_devices().await.unwrap();
|
||||||
println!("Images: {images:#?}");
|
println!("Images: {images:#?}");
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user