You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
distant/src/subcommand/launch.rs

93 lines
2.9 KiB
Rust

use crate::{opt::LaunchSubcommand, PROJECT_DIRS, SESSION_PATH};
use derive_more::{Display, Error, From};
use hex::FromHexError;
use orion::{aead::SecretKey, errors::UnknownCryptoError};
use std::string::FromUtf8Error;
use tokio::{io, process::Command};
#[derive(Debug, Display, Error, From)]
pub enum Error {
#[display(fmt = "Missing data for session")]
MissingSessionData,
BadKey(UnknownCryptoError),
HexError(FromHexError),
IoError(io::Error),
Utf8Error(FromUtf8Error),
}
pub fn run(cmd: LaunchSubcommand) -> Result<(), Error> {
let rt = tokio::runtime::Runtime::new()?;
rt.block_on(async { run_async(cmd).await })
}
async fn run_async(cmd: LaunchSubcommand) -> Result<(), Error> {
let remote_command = format!(
"{} listen --daemon --host {}",
cmd.remote_program, cmd.bind_server
);
let ssh_command = format!(
"{} -o StrictHostKeyChecking=no ssh://{}@{}:{} {} {}",
cmd.ssh_program,
cmd.username,
cmd.host,
cmd.port,
cmd.identity_file
.map(|f| format!("-i {}", f.as_path().display()))
.unwrap_or_default(),
remote_command,
);
let out = Command::new("sh")
.arg("-c")
.arg(ssh_command)
.output()
.await?;
// If our attempt to run the program via ssh failed, report it
if !out.status.success() {
return Err(Error::from(io::Error::new(
io::ErrorKind::Other,
String::from_utf8(out.stderr)?.trim().to_string(),
)));
}
// Parse our output for the specific info line
let out = String::from_utf8(out.stdout)?.trim().to_string();
let result = out
.lines()
.find_map(|line| {
let tokens: Vec<&str> = line.split(' ').take(4).collect();
let is_data_line = tokens.len() == 4 && tokens[0] == "DISTANT" && tokens[1] == "DATA";
match tokens[2].parse::<u16>() {
Ok(port) if is_data_line => {
let key = hex::decode(tokens[3])
.map_err(Error::from)
.and_then(|bytes| SecretKey::from_slice(&bytes).map_err(Error::from));
match key {
Ok(key) => Some(Ok((port, key))),
Err(x) => Some(Err(x)),
}
}
_ => None,
}
})
.unwrap_or(Err(Error::MissingSessionData));
// Write a session file containing our data for use in subsequent calls
let (port, key) = result?;
let key_hex_str = hex::encode(key.unprotected_as_bytes());
if cmd.print_startup_data {
println!("DISTANT DATA {} {}", port, key_hex_str);
}
// Ensure our cache directory exists
let cache_dir = PROJECT_DIRS.cache_dir();
tokio::fs::create_dir_all(cache_dir).await?;
// Write our session file
tokio::fs::write(SESSION_PATH.as_path(), format!("{} {}", port, key_hex_str)).await?;
Ok(())
}