mirror of
				https://github.com/velopack/velopack.git
				synced 2025-10-25 15:19:22 +00:00 
			
		
		
		
	Try using 'mv' instead of fs::rename/copy on linux
This commit is contained in:
		| @@ -22,46 +22,35 @@ pub fn apply_package_impl<'a>(root_path: &PathBuf, _app: &Manifest, pkg: &PathBu | |||||||
|         std::fs::set_permissions(&temp_path, <std::fs::Permissions as std::os::unix::fs::PermissionsExt>::from_mode(0o755))?; |         std::fs::set_permissions(&temp_path, <std::fs::Permissions as std::os::unix::fs::PermissionsExt>::from_mode(0o755))?; | ||||||
|  |  | ||||||
|         info!("Moving temp file to target: {}", &root_path.to_string_lossy()); |         info!("Moving temp file to target: {}", &root_path.to_string_lossy()); | ||||||
|         let mut result: Result<(), std::io::Error> = std::fs::rename(&temp_path, &root_path); |         // we use mv instead of fs::rename / fs::copy because rename fails cross-device | ||||||
|         if result.is_ok() { |         // and copy fails if the process is running (presumably because rust opens the file for writing) | ||||||
|  |         // while mv works in both cases. | ||||||
|  |         let mv_args = vec![&temp_path, &root_path.to_string_lossy().to_string()]; | ||||||
|  |         let mv_output = Command::new("mv").args(mv_args).output()?; | ||||||
|  |  | ||||||
|  |         if mv_output.status.success() { | ||||||
|             info!("AppImage moved successfully to: {}", &root_path.to_string_lossy()); |             info!("AppImage moved successfully to: {}", &root_path.to_string_lossy()); | ||||||
|             return Ok(()); |             return Ok(()); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         let mut result_err = result.unwrap_err(); |         // if the operation failed, let's try again elevated with pkexec | ||||||
|  |         error!("An error occurred ({:?}), will attempt to elevate permissions and try again...", mv_output); | ||||||
|         // if we tried to rename across filesystems, let's try again with a copy. |         dialogs::ask_user_to_elevate(&manifest)?; | ||||||
|         // ideally we check against std::io::ErrorKind::CrossesDevices but that is unstable at the moment |         let script = format!("#!/bin/sh\nmv -f '{}' '{}'", temp_path, &root_path.to_string_lossy()); | ||||||
|         if Some(18) == result_err.raw_os_error() { |         info!("Writing script for elevation: \n{}", script); | ||||||
|             info!("Move failed (cross-device), trying again with a copy."); |         fs::write(&script_path, script)?; | ||||||
|             result = std::fs::copy(&temp_path, &root_path).map(|_| {}); |         std::fs::set_permissions(&script_path, <std::fs::Permissions as std::os::unix::fs::PermissionsExt>::from_mode(0o755))?; | ||||||
|             if result.is_ok() { |         let args = vec![&script_path]; | ||||||
|                 info!("AppImage copied successfully to: {}", &root_path.to_string_lossy()); |         info!("Attempting to elevate: pkexec {:?}", args); | ||||||
|                 return Ok(()); |         let elev_output = Command::new("pkexec").args(args).output()?; | ||||||
|             } |         if elev_output.status.success() { | ||||||
|             result_err = result.unwrap_err(); |             info!("AppImage moved (elevated) to {}", &root_path.to_string_lossy()); | ||||||
|  |             return Ok(()); | ||||||
|  |         } else { | ||||||
|  |             bail!("pkexec failed with status: {:?}", elev_output); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // if the operation failed with permission denied, let's try again elevated with pkexec |         bail!("Failed to move the AppImage to target, bailing..."); | ||||||
|         if result_err.kind() == std::io::ErrorKind::PermissionDenied { |  | ||||||
|             error!("An error occurred {}, will attempt to elevate permissions and try again...", result_err); |  | ||||||
|             dialogs::ask_user_to_elevate(&manifest)?; |  | ||||||
|             let script = format!("#!/bin/sh\nmv -f '{}' '{}'", temp_path, &root_path.to_string_lossy()); |  | ||||||
|             info!("Writing script for elevation: \n{}", script); |  | ||||||
|             fs::write(&script_path, script)?; |  | ||||||
|             std::fs::set_permissions(&script_path, <std::fs::Permissions as std::os::unix::fs::PermissionsExt>::from_mode(0o755))?; |  | ||||||
|             let args = vec![&script_path]; |  | ||||||
|             info!("Attempting to elevate: pkexec {:?}", args); |  | ||||||
|             let status = Command::new("pkexec").args(args).status()?; |  | ||||||
|             if status.success() { |  | ||||||
|                 info!("AppImage moved (elevated) to {}", &root_path.to_string_lossy()); |  | ||||||
|                 return Ok(()); |  | ||||||
|             } else { |  | ||||||
|                 bail!("pkexec exited with status: {}", status); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         bail!("Failed to move the AppImage to target ({})", result_err); |  | ||||||
|     })(); |     })(); | ||||||
|     let _ = fs::remove_file(&script_path); |     let _ = fs::remove_file(&script_path); | ||||||
|     let _ = fs::remove_file(&temp_path); |     let _ = fs::remove_file(&temp_path); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user