mirror of
https://github.com/tauri-apps/plugins-workspace.git
synced 2026-04-21 11:26:15 +02:00
refactor(updater): cleanup install logic on Windows and add unit test (#1496)
This commit is contained in:
@@ -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,
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -580,16 +592,11 @@ 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)
|
||||
@@ -597,7 +604,7 @@ impl 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")))
|
||||
@@ -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(
|
||||
@@ -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);
|
||||
@@ -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);
|
||||
@@ -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)))
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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\"")
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user