Skip to content

Commit

Permalink
refactor(updater): cleanup install logic on Windows and add unit test (
Browse files Browse the repository at this point in the history
  • Loading branch information
amrbashir authored Jun 26, 2024
1 parent 0cb1baf commit 757ab74
Showing 1 changed file with 79 additions and 43 deletions.
122 changes: 79 additions & 43 deletions plugins/updater/src/updater.rs
Original file line number Diff line number Diff line change
Expand Up @@ -538,16 +538,28 @@ impl Update {

#[cfg(windows)]
enum WindowsUpdaterType {
Nsis,
Msi,
Nsis {
path: PathBuf,
#[allow(unused)]
temp: Option<tempfile::TempPath>,
},
Msi {
path: PathBuf,
#[allow(unused)]
temp: Option<tempfile::TempPath>,
},
}

#[cfg(windows)]
impl WindowsUpdaterType {
fn extension(&self) -> &str {
match self {
WindowsUpdaterType::Nsis => ".exe",
WindowsUpdaterType::Msi => ".msi",
fn nsis(path: PathBuf, temp: Option<tempfile::TempPath>) -> Self {
Self::Nsis { path, temp }
}

fn msi(path: PathBuf, temp: Option<tempfile::TempPath>) -> Self {
Self::Msi {
path: path.wrap_in_quotes(),
temp,
}
}
}
Expand Down Expand Up @@ -580,24 +592,19 @@ impl Update {
Win32::UI::{Shell::ShellExecuteW, WindowsAndMessaging::SW_SHOW},
};

let (updater_type, path, _temp) = Self::extract(bytes)?;

let mut msi_path = std::ffi::OsString::new();
msi_path.push("\"");
msi_path.push(&path);
msi_path.push("\"");
let updater_type = Self::extract(bytes)?;

let install_mode = self.config.install_mode();
let installer_args: Vec<&OsStr> = match updater_type {
WindowsUpdaterType::Nsis => install_mode
let installer_args: Vec<&OsStr> = match &updater_type {
WindowsUpdaterType::Nsis { .. } => install_mode
.nsis_args()
.iter()
.map(OsStr::new)
.chain(once(OsStr::new("/UPDATE")))
.chain(self.nsis_installer_args())
.chain(self.installer_args())
.collect(),
WindowsUpdaterType::Msi => [OsStr::new("/i"), msi_path.as_os_str()]
WindowsUpdaterType::Msi { path, .. } => [OsStr::new("/i"), path.as_os_str()]
.into_iter()
.chain(install_mode.msiexec_args().iter().map(OsStr::new))
.chain(once(OsStr::new("/promptrestart")))
Expand All @@ -609,17 +616,17 @@ impl Update {
on_before_exit();
}

let parameters = installer_args.join(OsStr::new(" "));
let parameters = encode_wide(parameters);

let path = match updater_type {
WindowsUpdaterType::Msi => std::env::var("SYSTEMROOT").as_ref().map_or_else(
let file = match &updater_type {
WindowsUpdaterType::Nsis { path, .. } => path.as_os_str().to_os_string(),
WindowsUpdaterType::Msi { .. } => std::env::var("SYSTEMROOT").as_ref().map_or_else(
|_| OsString::from("msiexec.exe"),
|p| OsString::from(format!("{p}\\System32\\msiexec.exe")),
),
WindowsUpdaterType::Nsis => path.as_os_str().to_os_string(),
};
let file = encode_wide(path);
let file = encode_wide(file);

let parameters = installer_args.join(OsStr::new(" "));
let parameters = encode_wide(parameters);

unsafe {
ShellExecuteW(
Expand Down Expand Up @@ -649,7 +656,7 @@ impl Update {
.collect::<Vec<_>>()
}

fn extract(bytes: &[u8]) -> Result<(WindowsUpdaterType, PathBuf, Option<tempfile::TempPath>)> {
fn extract(bytes: &[u8]) -> Result<WindowsUpdaterType> {
#[cfg(feature = "zip")]
if infer::archive::is_zip(bytes) {
return Self::extract_zip(bytes);
Expand All @@ -659,9 +666,7 @@ impl Update {
}

#[cfg(feature = "zip")]
fn extract_zip(
bytes: &[u8],
) -> Result<(WindowsUpdaterType, PathBuf, Option<tempfile::TempPath>)> {
fn extract_zip(bytes: &[u8]) -> Result<WindowsUpdaterType> {
let tmp_dir = tempfile::Builder::new().tempdir()?.into_path();

let archive = Cursor::new(bytes);
Expand All @@ -670,38 +675,38 @@ impl Update {

let paths = std::fs::read_dir(&tmp_dir)?;
for path in paths {
let found_path = path?.path();
let ext = found_path.extension();
let path = path?.path();
let ext = path.extension();
if ext == Some(OsStr::new("exe")) {
return Ok((WindowsUpdaterType::Nsis, found_path, None));
return Ok(WindowsUpdaterType::nsis(path, None));
} else if ext == Some(OsStr::new("msi")) {
return Ok((WindowsUpdaterType::Msi, found_path, None));
return Ok(WindowsUpdaterType::msi(path, None));
}
}

Err(crate::Error::BinaryNotFoundInArchive)
}

fn extract_exe(
bytes: &[u8],
) -> Result<(WindowsUpdaterType, PathBuf, Option<tempfile::TempPath>)> {
use std::io::Write;

let updater_type = if infer::app::is_exe(bytes) {
WindowsUpdaterType::Nsis
fn extract_exe(bytes: &[u8]) -> Result<WindowsUpdaterType> {
if infer::app::is_exe(bytes) {
let (path, temp) = Self::write_to_temp(bytes, ".exe")?;
Ok(WindowsUpdaterType::nsis(path, temp))
} else if infer::archive::is_msi(bytes) {
WindowsUpdaterType::Msi
let (path, temp) = Self::write_to_temp(bytes, ".msi")?;
Ok(WindowsUpdaterType::msi(path, temp))
} else {
return Err(crate::Error::InvalidUpdaterFormat);
};
Err(crate::Error::InvalidUpdaterFormat)
}
}

let ext = updater_type.extension();
fn write_to_temp(bytes: &[u8], ext: &str) -> Result<(PathBuf, Option<tempfile::TempPath>)> {
use std::io::Write;

let mut temp_file = tempfile::Builder::new().suffix(ext).tempfile()?;
temp_file.write_all(bytes)?;
let temp_path = temp_file.into_temp_path();

Ok((updater_type, temp_path.to_path_buf(), Some(temp_path)))
let temp = temp_file.into_temp_path();
Ok((temp.to_path_buf(), Some(temp)))
}
}

Expand Down Expand Up @@ -1005,3 +1010,34 @@ fn encode_wide(string: impl AsRef<OsStr>) -> Vec<u16> {
.chain(std::iter::once(0))
.collect()
}

#[cfg(windows)]
trait PathExt {
fn wrap_in_quotes(&self) -> Self;
}

#[cfg(windows)]
impl PathExt for PathBuf {
fn wrap_in_quotes(&self) -> Self {
let mut msi_path = OsString::from("\"");
msi_path.push(self.as_os_str());
msi_path.push("\"");
PathBuf::from(msi_path)
}
}

#[cfg(test)]
mod tests {

#[test]
#[cfg(windows)]
fn it_wraps_correctly() {
use super::PathExt;
use std::path::PathBuf;

assert_eq!(
PathBuf::from("C:\\Users\\Some User\\AppData\\tauri-example.exe").wrap_in_quotes(),
PathBuf::from("\"C:\\Users\\Some User\\AppData\\tauri-example.exe\"")
)
}
}

0 comments on commit 757ab74

Please sign in to comment.