mirror of
https://github.com/lights0123/n-link.git
synced 2025-01-21 23:11:53 +00:00
make the build actually happen
This commit is contained in:
parent
0472908ef4
commit
c20c4d9492
3636
desktop/src-tauri/Cargo.lock
generated
3636
desktop/src-tauri/Cargo.lock
generated
File diff suppressed because it is too large
Load diff
|
@ -16,14 +16,14 @@ libnspire = "0.2.2"
|
|||
lazy_static = "1.4.0"
|
||||
rusb = "0.6.4"
|
||||
serde = { version = "1.0", features = [ "derive" ] }
|
||||
tauri = { version = "1.0.0-beta.8", features = ["dialog-open", "dialog-save", "notification-all", "shell-open"] }
|
||||
tauri = { version = "1.5.4", features = ["dialog-open", "shell-open"] }
|
||||
clap = "3.0.0-beta.2"
|
||||
indicatif = "0.15"
|
||||
libusb1-sys = { version = "0.4.2", features = [ "vendored" ] }
|
||||
hashbrown = "0.11"
|
||||
|
||||
[build-dependencies]
|
||||
tauri-build = { version = "1.0.0-beta.4" }
|
||||
tauri-build = { version = "1.5.1", features = [] }
|
||||
|
||||
[features]
|
||||
custom-protocol = [ "tauri/custom-protocol" ]
|
||||
|
|
|
@ -1,394 +0,0 @@
|
|||
use std::ffi::OsStr;
|
||||
use std::io::{Read, Write};
|
||||
use std::path::PathBuf;
|
||||
use std::{fs::File, path::Path};
|
||||
|
||||
use clap::Clap;
|
||||
use indicatif::{ProgressBar, ProgressStyle};
|
||||
use libnspire::{dir::EntryType, PID, PID_CX2, VID};
|
||||
|
||||
#[derive(Clap, Debug)]
|
||||
#[clap(author, about, version)]
|
||||
struct Opt {
|
||||
#[clap(subcommand)]
|
||||
cmd: Option<SubCommand>,
|
||||
}
|
||||
|
||||
#[derive(Clap, Debug)]
|
||||
enum SubCommand {
|
||||
Upload(Upload),
|
||||
Download(Download),
|
||||
UploadOS(UploadOS),
|
||||
Copy(Copy),
|
||||
Move(Move),
|
||||
Mkdir(Mkdir),
|
||||
Rmdir(Rmdir),
|
||||
Ls(Ls),
|
||||
/// View license information
|
||||
License,
|
||||
}
|
||||
|
||||
/// Upload files to the calculator
|
||||
#[derive(Clap, Debug)]
|
||||
struct Upload {
|
||||
/// Files to upload
|
||||
#[clap(required = true, parse(from_os_str))]
|
||||
files: Vec<PathBuf>,
|
||||
/// Destination path
|
||||
dest: String,
|
||||
}
|
||||
|
||||
/// Download files from the calculator
|
||||
#[derive(Clap, Debug)]
|
||||
struct Download {
|
||||
/// Files to download
|
||||
#[clap(required = true)]
|
||||
files: Vec<String>,
|
||||
/// Destination path
|
||||
#[clap(required = true, parse(from_os_str))]
|
||||
dest: PathBuf,
|
||||
}
|
||||
|
||||
/// Upload and install a .tcc/.tco/.tcc2/.tco2/.tct2 OS file
|
||||
#[derive(Clap, Debug)]
|
||||
struct UploadOS {
|
||||
/// Path to the OS file
|
||||
#[clap(required = true, parse(from_os_str))]
|
||||
file: PathBuf,
|
||||
|
||||
/// Disables the file extension check
|
||||
#[clap(long)]
|
||||
no_check_os: bool,
|
||||
}
|
||||
|
||||
/// Copy a file to a different location
|
||||
#[derive(Clap, Debug)]
|
||||
struct Copy {
|
||||
/// Path to file
|
||||
#[clap(required = true)]
|
||||
from_path: String,
|
||||
|
||||
/// Path to new location
|
||||
#[clap(required = true)]
|
||||
dist_path: String,
|
||||
}
|
||||
|
||||
/// Move a file or directory to a new location
|
||||
#[derive(Clap, Debug)]
|
||||
struct Move {
|
||||
/// Path to file
|
||||
#[clap(required = true)]
|
||||
from_path: String,
|
||||
|
||||
/// Path to new location
|
||||
#[clap(required = true)]
|
||||
dist_path: String,
|
||||
}
|
||||
|
||||
/// Create a directory
|
||||
#[derive(Clap, Debug)]
|
||||
struct Mkdir {
|
||||
/// Path to directory
|
||||
#[clap(required = true)]
|
||||
path: String,
|
||||
}
|
||||
|
||||
/// Delete a directory
|
||||
#[derive(Clap, Debug)]
|
||||
struct Rmdir {
|
||||
/// Path to directory
|
||||
#[clap(required = true)]
|
||||
path: String,
|
||||
}
|
||||
|
||||
/// List the contents of a directory
|
||||
#[derive(Clap, Debug)]
|
||||
struct Ls {
|
||||
/// Path to directory
|
||||
#[clap(required = true)]
|
||||
path: String,
|
||||
}
|
||||
|
||||
fn get_dev() -> Option<libnspire::Handle<rusb::GlobalContext>> {
|
||||
rusb::devices()
|
||||
.unwrap()
|
||||
.iter()
|
||||
.find(|dev| {
|
||||
let descriptor = match dev.device_descriptor() {
|
||||
Ok(d) => d,
|
||||
Err(_) => return false,
|
||||
};
|
||||
descriptor.vendor_id() == VID && matches!(descriptor.product_id(), PID | PID_CX2)
|
||||
})
|
||||
.map(|dev| libnspire::Handle::new(dev.open().unwrap()).unwrap())
|
||||
}
|
||||
|
||||
pub fn cwd() -> PathBuf {
|
||||
#[cfg(target_os = "linux")]
|
||||
if std::env::var_os("APPIMAGE").is_some() && std::env::var_os("APPDIR").is_some() {
|
||||
if let Some(cwd) = std::env::var_os("OWD") {
|
||||
return cwd.into();
|
||||
}
|
||||
};
|
||||
std::env::current_dir().expect("Couldn't get current directory")
|
||||
}
|
||||
|
||||
pub fn run() -> bool {
|
||||
let opt: Opt = Opt::parse();
|
||||
if let Some(cmd) = opt.cmd {
|
||||
match cmd {
|
||||
SubCommand::Upload(Upload { files, mut dest }) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
for file in files {
|
||||
let mut buf = vec![];
|
||||
File::open(cwd().join(&file))
|
||||
.unwrap()
|
||||
.read_to_end(&mut buf)
|
||||
.unwrap();
|
||||
let name = file
|
||||
.file_name()
|
||||
.expect("Failed to get file name")
|
||||
.to_string_lossy()
|
||||
.to_string();
|
||||
let bar = ProgressBar::new(buf.len() as u64);
|
||||
bar.set_style(ProgressStyle::default_bar().template("{spinner:.green} {msg} [{elapsed_precise}] [{bar:40.cyan/blue}] {bytes}/{total_bytes} ({bytes_per_sec}, {eta})"));
|
||||
bar.set_message(&format!("Upload {}", name));
|
||||
bar.enable_steady_tick(100);
|
||||
if dest.ends_with('/') {
|
||||
dest.remove(dest.len() - 1);
|
||||
}
|
||||
let res = handle.write_file(&format!("{}/{}", dest, name), &buf, &mut |remaining| {
|
||||
bar.set_position((buf.len() - remaining) as u64)
|
||||
});
|
||||
|
||||
match res {
|
||||
Ok(_) => {
|
||||
bar.finish_with_message(&format!("Upload {}: Ok", dest));
|
||||
}
|
||||
Err(error) => {
|
||||
bar.abandon_with_message(&format!("Failed: {}", error));
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::Download(Download { dest, files }) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
for file in files {
|
||||
let attr = handle.file_attr(&file);
|
||||
match attr {
|
||||
Ok(attr) => {
|
||||
let path = Path::new(&file);
|
||||
let dest_path = Path::join(&dest, path.file_name().unwrap().to_str().unwrap());
|
||||
match File::create(dest_path) {
|
||||
Ok(mut dest_file) => {
|
||||
let mut buf = vec![0u8; attr.size() as usize];
|
||||
|
||||
let bar = ProgressBar::new(buf.len() as u64);
|
||||
bar.set_style(ProgressStyle::default_bar().template("{spinner:.green} {msg} [{elapsed_precise}] [{bar:40.cyan/blue}] {bytes}/{total_bytes} ({bytes_per_sec}, {eta})"));
|
||||
bar.set_message(&format!(
|
||||
"Download {}",
|
||||
path.file_name().unwrap().to_str().unwrap()
|
||||
));
|
||||
bar.enable_steady_tick(100);
|
||||
|
||||
let len = buf.len();
|
||||
|
||||
let res = handle.read_file(&file, &mut buf, &mut |remaining| {
|
||||
bar.set_position((len - remaining) as u64);
|
||||
});
|
||||
|
||||
match res {
|
||||
Ok(_) => {
|
||||
bar.set_message("Writing file to disk");
|
||||
|
||||
match dest_file.write_all(&buf) {
|
||||
Ok(_) => {
|
||||
bar.finish_with_message("Transfer completed");
|
||||
}
|
||||
Err(error) => {
|
||||
bar.abandon_with_message(&format!(
|
||||
"Failed to write file to disk: {}",
|
||||
error
|
||||
));
|
||||
}
|
||||
}
|
||||
}
|
||||
Err(error) => {
|
||||
bar.abandon_with_message(&format!("Failed to transfer file: {}", error))
|
||||
}
|
||||
}
|
||||
}
|
||||
Err(error) => {
|
||||
eprintln!("Failed to open destination file: {}", error);
|
||||
}
|
||||
}
|
||||
}
|
||||
Err(error) => {
|
||||
eprintln!("Failed to read file info: {}", error);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::UploadOS(UploadOS { file, no_check_os }) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
let calc_info = handle.info().expect("Failed to obtain device info");
|
||||
|
||||
let file_ext = file
|
||||
.extension()
|
||||
.unwrap_or(OsStr::new(""))
|
||||
.to_string_lossy()
|
||||
.to_string();
|
||||
|
||||
let mut buf = vec![];
|
||||
let mut f = File::open(cwd().join(&file)).unwrap_or_else(|err| {
|
||||
eprintln!("Failed to open file: {}", err);
|
||||
std::process::exit(1);
|
||||
});
|
||||
|
||||
if format!(".{}", file_ext) != calc_info.os_extension {
|
||||
if no_check_os {
|
||||
eprintln!(
|
||||
"Warning: {} expects file of type {}",
|
||||
calc_info.name, calc_info.os_extension
|
||||
);
|
||||
} else {
|
||||
eprintln!(
|
||||
"Error: {} expects file of type {}",
|
||||
calc_info.name, calc_info.os_extension
|
||||
);
|
||||
eprintln!("Provide --no-check-os to bypass this check.");
|
||||
std::process::exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
f.read_to_end(&mut buf).unwrap();
|
||||
|
||||
let name = file
|
||||
.file_name()
|
||||
.expect("Failed to get file name")
|
||||
.to_string_lossy()
|
||||
.to_string();
|
||||
|
||||
let bar = ProgressBar::new(buf.len() as u64);
|
||||
bar.set_style(ProgressStyle::default_bar().template("{spinner:.green} {msg} [{elapsed_precise}] [{bar:40.cyan/blue}] {bytes}/{total_bytes} ({bytes_per_sec}, {eta})"));
|
||||
bar.set_message(&format!("Upload OS {}", name));
|
||||
bar.enable_steady_tick(100);
|
||||
|
||||
let res = handle.send_os(&buf, &mut |remaining| {
|
||||
bar.set_position((buf.len() - remaining) as u64);
|
||||
});
|
||||
|
||||
match res {
|
||||
Ok(_) => {
|
||||
bar.finish();
|
||||
}
|
||||
Err(error) => {
|
||||
bar.abandon_with_message(&format!("OS Upload failed: {}", error));
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::Copy(Copy {
|
||||
from_path,
|
||||
dist_path,
|
||||
}) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
match handle.copy_file(&from_path, &dist_path) {
|
||||
Ok(_) => {
|
||||
println!("Copy {} => {}: Ok", from_path, dist_path);
|
||||
}
|
||||
Err(error) => {
|
||||
eprintln!("Failed to copy file or directory: {}", error);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::Move(Move {
|
||||
from_path,
|
||||
dist_path,
|
||||
}) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
match handle.move_file(&from_path, &dist_path) {
|
||||
Ok(_) => {
|
||||
println!("Move {} => {}: Ok", from_path, dist_path);
|
||||
}
|
||||
Err(error) => {
|
||||
eprintln!("Failed to move file or directory: {}", error);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::Mkdir(Mkdir { path }) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
match handle.create_dir(&path) {
|
||||
Ok(_) => {
|
||||
println!("Create {}: Ok", path);
|
||||
}
|
||||
Err(error) => {
|
||||
eprintln!("Failed to create directory: {}", error);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::Rmdir(Rmdir { path }) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
match handle.delete_dir(&path) {
|
||||
Ok(_) => {
|
||||
println!("Remove {}: Ok", path);
|
||||
}
|
||||
Err(error) => {
|
||||
eprintln!("Failed to delete directory: {}", error);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::Ls(Ls { path }) => {
|
||||
if let Some(handle) = get_dev() {
|
||||
match handle.list_dir(&path) {
|
||||
Ok(dir_list) => {
|
||||
for item in dir_list.iter() {
|
||||
println!(
|
||||
"{}{}",
|
||||
item.name().to_str().unwrap(),
|
||||
if item.entry_type() == EntryType::Directory {
|
||||
"/"
|
||||
} else {
|
||||
""
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
||||
Err(error) => {
|
||||
eprintln!("Failed to list directory: {}", error);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
eprintln!("Couldn't find any device");
|
||||
}
|
||||
}
|
||||
SubCommand::License => {
|
||||
println!("{}", include_str!("../../LICENSE"));
|
||||
println!(include_str!("NOTICE.txt"), env!("CARGO_PKG_REPOSITORY"));
|
||||
}
|
||||
}
|
||||
true
|
||||
} else {
|
||||
false
|
||||
}
|
||||
}
|
|
@ -104,7 +104,7 @@ pub fn enumerate<R: Runtime>(handle: Window<R>) -> Result<Vec<AddDevice>, Serial
|
|||
)
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize)]
|
||||
#[derive(Clone, Debug, Serialize)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct AddDevice {
|
||||
#[serde(flatten)]
|
||||
|
@ -114,7 +114,7 @@ pub struct AddDevice {
|
|||
pub needs_drivers: bool,
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize)]
|
||||
#[derive(Clone, Debug, Serialize)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct ProgressUpdate {
|
||||
#[serde(flatten)]
|
||||
|
|
|
@ -15,7 +15,6 @@ use tauri::{Runtime, Window};
|
|||
|
||||
use crate::cmd::{add_device, AddDevice, DevId, ProgressUpdate};
|
||||
|
||||
mod cli;
|
||||
mod cmd;
|
||||
|
||||
pub enum DeviceState {
|
||||
|
@ -434,9 +433,6 @@ mod invoked {
|
|||
}
|
||||
|
||||
fn main() {
|
||||
if cli::run() {
|
||||
return;
|
||||
}
|
||||
let has_registered_callback = AtomicBool::new(false);
|
||||
tauri::Builder::default()
|
||||
.on_page_load(move |window, _p| {
|
||||
|
|
|
@ -20,7 +20,6 @@
|
|||
"macOS": {
|
||||
"frameworks": [],
|
||||
"minimumSystemVersion": "",
|
||||
"useBootstrapper": false,
|
||||
"exceptionDomain": "",
|
||||
"signingIdentity": null,
|
||||
"entitlements": null
|
||||
|
|
Loading…
Reference in a new issue