mirror of
https://github.com/jkcoxson/idevice.git
synced 2026-03-02 14:36:16 +01:00
Implement remaining app services
This commit is contained in:
@@ -73,6 +73,63 @@ pub struct ProcessToken {
|
|||||||
pub executable_url: Option<ExecutableUrl>,
|
pub executable_url: Option<ExecutableUrl>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Clone, Debug)]
|
||||||
|
pub struct SignalResponse {
|
||||||
|
pub process: ProcessToken,
|
||||||
|
#[serde(rename = "deviceTimestamp")]
|
||||||
|
pub device_timestamp: plist::Date,
|
||||||
|
pub signal: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Icon data is in a proprietary format.
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// 0000: 06 00 00 00 40 06 00 00 00 00 00 00 01 00 00 00 - header
|
||||||
|
/// 0010: 00 00 a0 41 00 00 a0 41 00 00 00 00 00 00 00 00 - width x height as float
|
||||||
|
/// 0020: 00 00 a0 41 00 00 a0 41 00 00 00 00 00 00 00 00 - wdith x height (again?)
|
||||||
|
/// 0030: 00 00 00 00 03 08 08 09 2a 68 6f 7d 44 a9 b7 d0 - start of image data
|
||||||
|
/// <snip>
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// The data can be parsed like so in Python
|
||||||
|
///
|
||||||
|
/// ```python
|
||||||
|
/// from PIL import Image
|
||||||
|
///
|
||||||
|
/// width, height = 20, 20 (from the float sizes)
|
||||||
|
/// with open("icon.raw", "rb") as f:
|
||||||
|
/// f.seek(0x30)
|
||||||
|
/// raw = f.read(width * height * 4)
|
||||||
|
///
|
||||||
|
/// img = Image.frombytes("RGBA", (width, height), raw)
|
||||||
|
/// img.save("icon.png")
|
||||||
|
/// ```
|
||||||
|
#[derive(Deserialize, Clone, Debug)]
|
||||||
|
pub struct IconData {
|
||||||
|
pub data: plist::Data,
|
||||||
|
#[serde(rename = "iconSize.height")]
|
||||||
|
pub icon_height: f64,
|
||||||
|
#[serde(rename = "iconSize.width")]
|
||||||
|
pub icon_width: f64,
|
||||||
|
#[serde(rename = "minimumSize.height")]
|
||||||
|
pub minimum_height: f64,
|
||||||
|
#[serde(rename = "minimumSize.width")]
|
||||||
|
pub minimum_width: f64,
|
||||||
|
#[serde(rename = "$classes")]
|
||||||
|
pub classes: Vec<String>,
|
||||||
|
#[serde(rename = "validationToken")]
|
||||||
|
pub validation_token: plist::Data,
|
||||||
|
pub uuid: IconUuid,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Clone, Debug)]
|
||||||
|
pub struct IconUuid {
|
||||||
|
#[serde(rename = "NS.uuidbytes")]
|
||||||
|
pub bytes: plist::Data,
|
||||||
|
#[serde(rename = "$classes")]
|
||||||
|
pub classes: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
impl<R: ReadWrite> AppServiceClient<R> {
|
impl<R: ReadWrite> AppServiceClient<R> {
|
||||||
pub async fn new(stream: R) -> Result<Self, IdeviceError> {
|
pub async fn new(stream: R) -> Result<Self, IdeviceError> {
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
@@ -181,12 +238,10 @@ impl<R: ReadWrite> AppServiceClient<R> {
|
|||||||
.invoke("com.apple.coredevice.feature.listprocesses", None)
|
.invoke("com.apple.coredevice.feature.listprocesses", None)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
println!("{}", pretty_print_plist(&res));
|
|
||||||
|
|
||||||
let res = match res
|
let res = match res
|
||||||
.as_dictionary()
|
.as_dictionary()
|
||||||
.and_then(|x| x.get("processTokens"))
|
.and_then(|x| x.get("processTokens"))
|
||||||
.and_then(|x| plist::from_value(x).unwrap())
|
.and_then(|x| plist::from_value(x).ok())
|
||||||
{
|
{
|
||||||
Some(r) => r,
|
Some(r) => r,
|
||||||
None => {
|
None => {
|
||||||
@@ -197,4 +252,107 @@ impl<R: ReadWrite> AppServiceClient<R> {
|
|||||||
|
|
||||||
Ok(res)
|
Ok(res)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Gives no response on failure or success
|
||||||
|
pub async fn uninstall_app(
|
||||||
|
&mut self,
|
||||||
|
bundle_id: impl Into<String>,
|
||||||
|
) -> Result<(), IdeviceError> {
|
||||||
|
let bundle_id = bundle_id.into();
|
||||||
|
self.inner
|
||||||
|
.invoke(
|
||||||
|
"com.apple.coredevice.feature.uninstallapp",
|
||||||
|
Some(
|
||||||
|
crate::plist!({"bundleIdentifier": bundle_id})
|
||||||
|
.into_dictionary()
|
||||||
|
.unwrap(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn send_signal(
|
||||||
|
&mut self,
|
||||||
|
pid: u32,
|
||||||
|
signal: u32,
|
||||||
|
) -> Result<SignalResponse, IdeviceError> {
|
||||||
|
let res = self
|
||||||
|
.inner
|
||||||
|
.invoke(
|
||||||
|
"com.apple.coredevice.feature.sendsignaltoprocess",
|
||||||
|
Some(
|
||||||
|
crate::plist!({
|
||||||
|
"process": { "processIdentifier": pid as i64},
|
||||||
|
"signal": signal as i64,
|
||||||
|
})
|
||||||
|
.into_dictionary()
|
||||||
|
.unwrap(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let res = match plist::from_value(&res) {
|
||||||
|
Ok(r) => r,
|
||||||
|
Err(e) => {
|
||||||
|
warn!("Could not parse signal response: {e:?}");
|
||||||
|
return Err(IdeviceError::UnexpectedResponse);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(res)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn fetch_app_icon(
|
||||||
|
&mut self,
|
||||||
|
bundle_id: impl Into<String>,
|
||||||
|
width: f32,
|
||||||
|
height: f32,
|
||||||
|
scale: f32,
|
||||||
|
allow_placeholder: bool,
|
||||||
|
) -> Result<IconData, IdeviceError> {
|
||||||
|
let bundle_id = bundle_id.into();
|
||||||
|
let res = self
|
||||||
|
.inner
|
||||||
|
.invoke(
|
||||||
|
"com.apple.coredevice.feature.fetchappicons",
|
||||||
|
Some(
|
||||||
|
crate::plist!({
|
||||||
|
"width": width,
|
||||||
|
"height": height,
|
||||||
|
"scale": scale,
|
||||||
|
"allowPlaceholder": allow_placeholder,
|
||||||
|
"bundleIdentifier": bundle_id
|
||||||
|
})
|
||||||
|
.into_dictionary()
|
||||||
|
.unwrap(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let res = match res
|
||||||
|
.as_dictionary()
|
||||||
|
.and_then(|x| x.get("appIconContainer"))
|
||||||
|
.and_then(|x| x.as_dictionary())
|
||||||
|
.and_then(|x| x.get("iconImage"))
|
||||||
|
.and_then(|x| x.as_data())
|
||||||
|
{
|
||||||
|
Some(r) => r.to_vec(),
|
||||||
|
None => {
|
||||||
|
warn!("Did not receive appIconContainer/iconImage data");
|
||||||
|
return Err(IdeviceError::UnexpectedResponse);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let res = ns_keyed_archive::decode::from_bytes(&res)?;
|
||||||
|
println!("{}", pretty_print_plist(&res));
|
||||||
|
match plist::from_value(&res) {
|
||||||
|
Ok(r) => Ok(r),
|
||||||
|
Err(e) => {
|
||||||
|
warn!("Failed to deserialize ns keyed archive: {e:?}");
|
||||||
|
Err(IdeviceError::UnexpectedResponse)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -58,6 +58,35 @@ async fn main() {
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
.subcommand(Command::new("processes").about("List the processes running"))
|
.subcommand(Command::new("processes").about("List the processes running"))
|
||||||
|
.subcommand(
|
||||||
|
Command::new("uninstall").about("Uninstall an app").arg(
|
||||||
|
Arg::new("bundle_id")
|
||||||
|
.required(true)
|
||||||
|
.help("The bundle ID to uninstall"),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.subcommand(
|
||||||
|
Command::new("signal")
|
||||||
|
.about("Send a signal to an app")
|
||||||
|
.arg(Arg::new("pid").required(true).help("PID to send to"))
|
||||||
|
.arg(Arg::new("signal").required(true).help("Signal to send")),
|
||||||
|
)
|
||||||
|
.subcommand(
|
||||||
|
Command::new("icon")
|
||||||
|
.about("Send a signal to an app")
|
||||||
|
.arg(
|
||||||
|
Arg::new("bundle_id")
|
||||||
|
.required(true)
|
||||||
|
.help("The bundle ID to fetch"),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("path")
|
||||||
|
.required(true)
|
||||||
|
.help("The path to save the icon to"),
|
||||||
|
)
|
||||||
|
.arg(Arg::new("hw").required(false).help("The height and width"))
|
||||||
|
.arg(Arg::new("scale").required(false).help("The scale")),
|
||||||
|
)
|
||||||
.get_matches();
|
.get_matches();
|
||||||
|
|
||||||
if matches.get_flag("about") {
|
if matches.get_flag("about") {
|
||||||
@@ -125,6 +154,66 @@ async fn main() {
|
|||||||
} else if matches.subcommand_matches("processes").is_some() {
|
} else if matches.subcommand_matches("processes").is_some() {
|
||||||
let p = asc.list_processes().await.expect("no processes?");
|
let p = asc.list_processes().await.expect("no processes?");
|
||||||
println!("{p:#?}");
|
println!("{p:#?}");
|
||||||
|
} else if let Some(matches) = matches.subcommand_matches("uninstall") {
|
||||||
|
let bundle_id: &String = match matches.get_one("bundle_id") {
|
||||||
|
Some(b) => b,
|
||||||
|
None => {
|
||||||
|
eprintln!("No bundle ID passed");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
asc.uninstall_app(bundle_id).await.expect("no launch")
|
||||||
|
} else if let Some(matches) = matches.subcommand_matches("signal") {
|
||||||
|
let pid: u32 = match matches.get_one::<String>("pid") {
|
||||||
|
Some(b) => b.parse().expect("failed to parse PID as u32"),
|
||||||
|
None => {
|
||||||
|
eprintln!("No bundle PID passed");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let signal: u32 = match matches.get_one::<String>("signal") {
|
||||||
|
Some(b) => b.parse().expect("failed to parse signal as u32"),
|
||||||
|
None => {
|
||||||
|
eprintln!("No bundle signal passed");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let res = asc.send_signal(pid, signal).await.expect("no signal");
|
||||||
|
println!("{res:#?}");
|
||||||
|
} else if let Some(matches) = matches.subcommand_matches("icon") {
|
||||||
|
let bundle_id: &String = match matches.get_one("bundle_id") {
|
||||||
|
Some(b) => b,
|
||||||
|
None => {
|
||||||
|
eprintln!("No bundle ID passed");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let save_path: &String = match matches.get_one("path") {
|
||||||
|
Some(b) => b,
|
||||||
|
None => {
|
||||||
|
eprintln!("No bundle ID passed");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let hw: f32 = match matches.get_one::<String>("hw") {
|
||||||
|
Some(b) => b.parse().expect("failed to parse PID as f32"),
|
||||||
|
None => 1.0,
|
||||||
|
};
|
||||||
|
let scale: f32 = match matches.get_one::<String>("scale") {
|
||||||
|
Some(b) => b.parse().expect("failed to parse signal as f32"),
|
||||||
|
None => 1.0,
|
||||||
|
};
|
||||||
|
|
||||||
|
let res = asc
|
||||||
|
.fetch_app_icon(bundle_id, hw, hw, scale, true)
|
||||||
|
.await
|
||||||
|
.expect("no signal");
|
||||||
|
println!("{res:?}");
|
||||||
|
tokio::fs::write(save_path, res.data)
|
||||||
|
.await
|
||||||
|
.expect("failed to save");
|
||||||
} else {
|
} else {
|
||||||
eprintln!("Invalid usage, pass -h for help");
|
eprintln!("Invalid usage, pass -h for help");
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user