improv-setup/src/main.rs
2024-12-23 17:28:56 +01:00

174 lines
4.9 KiB
Rust

use anyhow::{
Context,
Result,
};
use clap::{
Parser,
Subcommand,
};
use env_logger;
use hex;
use improv_setup::improv::{
IMPROV_HEADER,
IMPROV_VERSION,
RPCCommand,
CurrentState,
calculate_checksum,
ImprovDataToPacket,
ImprovDataFromPacket,
RawPacket,
ImprovPacket,
RequestCurrentStateCommand,
RequestDeviceInformationPacket,
};
use improv_setup::serial;
use log::{
debug,
log_enabled,
info,
Level,
};
use tokio_serial;
#[derive(Subcommand, Clone)]
enum DeviceCommands {
/// Request current state
State,
/// Set wifi credentials
SetWifi {
/// SSID of the network
ssid: String,
/// Password for the SSID
password: String,
},
/// Request device info
Info,
}
impl Default for DeviceCommands {
fn default() -> Self {
return Self::State;
}
}
#[derive(Subcommand, Clone)]
enum Commands {
/// List available serial devices
ListDevices,
/// Device
Device {
/// Path to the serial device
path: String,
/// Baud rate used for connecting to the serial device
#[arg(long, default_value_t = 115200)]
baud_rate: u32,
#[command(subcommand)]
device_command: Option<DeviceCommands>,
},
}
#[derive(Parser)]
#[command(version, about, long_about = None)]
struct Cli {
#[command(subcommand)]
command: Option<Commands>,
}
fn to_ascii_debug(bytes: &Vec<u8>) -> String {
let mut out = String::new();
for b in bytes {
if b.is_ascii_graphic() {
out += &b.escape_ascii().to_string();
}
else {
out.push_str(".");
}
}
return out;
}
fn to_bytewise_debug(bytes: &Vec<u8>) -> String {
let mut out = String::new();
for b in bytes {
out += &hex::encode(&[b.clone()]);
out += " ";
if b.is_ascii_graphic() {
out += &b.escape_ascii().to_string();
}
out += "\n";
}
return out;
}
#[tokio::main]
async fn main() -> Result<()>{
env_logger::init();
let cli = Cli::parse();
let command: Commands = match &cli.command {
Some(command) => command.clone(),
None => Commands::ListDevices,
};
match &command {
Commands::ListDevices => {
println!("{}", tokio_serial::available_ports()
.context("Failed to list available ports")?
.iter()
.map(|serialport| serialport.port_name.clone())
.fold(String::new(), |a, b| a + &b + &String::from("\n")));
},
Commands::Device {path, baud_rate, device_command} => {
match &device_command.clone().unwrap_or_default() {
DeviceCommands::State => {
let request_current_state_packet = RequestCurrentStateCommand {};
let mut serial_interface = serial::SerialInterface::new(path, *baud_rate).context("Couldn't init serial interface")?;
serial_interface.send(&request_current_state_packet).context("Failed to send improv packet")?;
let result_bytes = serial_interface.recv_bytes().context("Couldn't receive any improv packet")?;
let raw_packet = RawPacket::try_from_bytes(&result_bytes).context("Failed to deserialize packet")?;
if let ImprovPacket::CurrentStateResponse(current_state_response) = ImprovPacket::try_from_raw_packet(&raw_packet).context("Failed to read packet")? {
println!("Current state: {}", &current_state_response.current_state);
}
if let ImprovPacket::ErrorState(error_state) = ImprovPacket::try_from_raw_packet(&raw_packet).context("Failed to read packet")? {
println!("Error state: {}", &error_state.error_state);
}
},
DeviceCommands::SetWifi {ssid, password} => {
println!("Not implemented");
},
DeviceCommands::Info => {
let request_device_information_packet = RequestDeviceInformationPacket {};
let mut serial_interface = serial::SerialInterface::new(path, *baud_rate).context("Couldn't init serial interface")?;
serial_interface.send(&request_device_information_packet).context("Failed to send improv packet")?;
let result_bytes = serial_interface.recv_bytes().context("Couldn't receive any improv packet")?;
let raw_packet = RawPacket::try_from_bytes(&result_bytes).context("Failed to deserialize packet")?;
if let ImprovPacket::RPCResult(rpc_result) = ImprovPacket::try_from_raw_packet(&raw_packet).context("Failed to read packet")? {
for r in rpc_result.results {
println!("{}", &r);
}
}
},
};
},
};
Ok(())
}