-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathupdate.rs
143 lines (129 loc) · 4.95 KB
/
update.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
#![cfg(feature = "self_update")]
use self_update::self_replace;
use self_update::update::Release;
use semver::Version;
use std::path::Path;
use std::{env, fs, io};
const REPO_OWNER: &str = "hacknus";
const REPO_NAME: &str = "serial-monitor-rust";
const MACOS_APP_NAME: &str = "Serial Monitor.app";
/// method to copy the complete directory `src` to `dest` but skipping the binary `binary_name`
/// since we have to use `self-replace` for that.
fn copy_dir(src: &Path, dest: &Path, binary_name: &str) -> io::Result<()> {
// Ensure the destination directory exists
if !dest.exists() {
fs::create_dir_all(dest)?;
}
// Iterate through entries in the source directory
for entry in fs::read_dir(src)? {
let entry = entry?;
let path = entry.path();
let dest_path = dest.join(entry.file_name());
if path.is_dir() {
// Recursively copy subdirectories
copy_dir(&path, &dest_path, binary_name)?;
} else if let Some(file_name) = path.file_name() {
if file_name != binary_name {
// Copy files except for the binary
fs::copy(&path, &dest_path)?;
}
}
}
Ok(())
}
/// Function to check for updates and return the latest one, if it is more recent than the current version
pub fn check_update() -> Option<Release> {
if let Ok(builder) = self_update::backends::github::ReleaseList::configure()
.repo_owner(REPO_OWNER)
.repo_name(REPO_NAME)
.build()
{
if let Ok(releases) = builder.fetch() {
let current_version = Version::parse(env!("CARGO_PKG_VERSION")).unwrap();
return releases
.iter()
.filter_map(|release| {
let release_version_str = release
.version
.strip_prefix("v")
.unwrap_or(&release.version);
Version::parse(release_version_str)
.ok()
.map(|parsed_version| (parsed_version, release))
})
.filter(|(parsed_version, _)| parsed_version > ¤t_version) // Compare versions
.max_by(|(a, _), (b, _)| a.cmp(b)) // Find the max version
.map(|(_, release)| release.clone()); // Return the release
}
}
None
}
/// custom update function for use with bundles
pub fn update(release: Release) -> Result<(), Box<dyn std::error::Error>> {
let target_asset = if cfg!(target_os = "windows") {
release.asset_for(self_update::get_target(), Some("exe"))
} else if cfg!(target_os = "linux") {
release.asset_for(self_update::get_target(), Some("bin"))
} else {
release.asset_for(self_update::get_target(), None)
}
.ok_or("No asset found")?;
let tmp_archive_dir = tempfile::TempDir::new()?;
let tmp_archive_path = tmp_archive_dir.path().join(&target_asset.name);
let tmp_archive = fs::File::create(&tmp_archive_path)?;
self_update::Download::from_url(&target_asset.download_url)
.set_header(reqwest::header::ACCEPT, "application/octet-stream".parse()?)
.download_to(&tmp_archive)?;
self_update::Extract::from_source(&tmp_archive_path).extract_into(tmp_archive_dir.path())?;
let new_exe = if cfg!(target_os = "windows") {
let binary = env::current_exe()?
.file_name()
.unwrap()
.to_str()
.unwrap()
.to_string();
tmp_archive_dir.path().join(binary)
} else if cfg!(target_os = "macos") {
let binary = env::current_exe()?
.file_name()
.unwrap()
.to_str()
.unwrap()
.to_string();
let app_dir = env::current_exe()?
.parent()
.unwrap()
.parent()
.unwrap()
.parent()
.unwrap()
.to_path_buf();
let app_name = app_dir
.clone()
.file_name()
.unwrap()
.to_str()
.unwrap()
.to_string();
let _ = copy_dir(&tmp_archive_dir.path().join(&app_name), &app_dir, &binary);
// MACOS_APP_NAME either needs to be hardcoded or extracted from the downloaded and
// extracted archive, but we cannot just assume that the parent directory of the
// currently running executable is equal to the app name - this is especially not
// the case if we run the code with `cargo run`.
tmp_archive_dir
.path()
.join(format!("{}/Contents/MacOS/{}", MACOS_APP_NAME, binary))
} else if cfg!(target_os = "linux") {
let binary = env::current_exe()?
.file_name()
.unwrap()
.to_str()
.unwrap()
.to_string();
tmp_archive_dir.path().join(binary)
} else {
return Err("Running on unsupported OS".into());
};
self_replace::self_replace(new_exe)?;
Ok(())
}