summaryrefslogblamecommitdiff
path: root/src/main.rs
blob: ccc0550708f24183d35bd585cfb347e68f0e75ed (plain) (tree)
1
2
3
4
5
6
7
8
9
                                        
 


                                                    
                   
                 

                                                           



                                     
                       
 
        



             


                                                                              




                                                             
              
                                              
                                     
 

                            

                              
                        
                                         




















                                                                          
             



                                                                  
                                            





          
                                                                 
                                   

                                                                             

                                                                                 
     

           
 
 


                                                                                              


                                                            
 
 










                                                                








                                                                               
                                                                 

                                                                  

 








                                                                        
                                                 



         

                             
                  
 
use std::{fmt::Display, time::Duration};

use crate::{cli::print_completions, config::Config};
use clap::{CommandFactory, Parser};
use cli::{Args, Commands, DeviceCmd};
use config::Method;
use error::Error;
use indicatif::{MultiProgress, ProgressBar, ProgressStyle};
use requests::{device, start::start};
use reqwest::{
    header::{HeaderMap, HeaderValue},
    Response,
};
use serde::Deserialize;

mod cli;
mod config;
mod error;
mod requests;

static OVERVIEW_STYLE: &str = "{spinner:.green} ({elapsed_precise}{wide_msg}";
static OVERVIEW_ERROR: &str = "✗ ({elapsed_precise}) {wide_msg}";
static OVERVIEW_DONE: &str = "✓ ({elapsed_precise}) {wide_msg}";
static DEFAULT_STYLE: &str = "  {spinner:.green} {wide_msg}";
static DONE_STYLE: &str = "  ✓ {wide_msg}";
static ERROR_STYLE: &str = "  ✗ {wide_msg}";
static TICK_SPEED: u64 = 1000 / 16;

#[tokio::main]
async fn main() -> Result<(), anyhow::Error> {
    let mut config = Config::load()?;

    let cli = Args::parse();

    config.cli_override(&cli);

    match cli.commands {
        Commands::Start { id, ping } => {
            start(&config, id, ping.unwrap_or(true)).await?;
        }
        Commands::Device { devicecmd } => match devicecmd {
            DeviceCmd::Add {
                id,
                mac,
                broadcast_addr,
                ip,
            } => {
                device::put(&config, id, mac, broadcast_addr, ip).await?;
            }
            DeviceCmd::Get { id } => {
                device::get(&config, id).await?;
            }
            DeviceCmd::Edit {
                id,
                mac,
                broadcast_addr,
                ip,
            } => {
                device::post(&config, id, mac, broadcast_addr, ip).await?;
            }
        },
        Commands::CliGen { id } => {
            eprintln!("Generating completion file for {id:?}...");
            let mut cmd = Args::command();
            print_completions(id, &mut cmd);
        }
    }

    Ok(())
}

fn default_headers(config: &Config) -> Result<HeaderMap, Error> {
    let mut map = HeaderMap::new();
    map.append("Accept-Content", HeaderValue::from_str("application/json")?);
    map.append("Content-Type", HeaderValue::from_str("application/json")?);
    if config.auth.method != Method::None {
        map.append("Authorization", HeaderValue::from_str(&config.auth.secret)?);
    }

    Ok(map)
}

fn format_url(config: &Config, path: &str, protocol: &Protocols, id: Option<&str>) -> String {
    if let Some(id) = id {
        format!("{}://{}/{}/{}", protocol, config.server, path, id)
    } else {
        format!("{}://{}/{}", protocol, config.server, path)
    }
}

async fn check_success(res: Response) -> Result<String, Error> {
    let status = res.status();
    if status.is_success() {
        Ok(res.text().await?)
    } else if status.as_u16() == 401 {
        Err(Error::Authorization)
    } else {
        Err(Error::HttpStatus(status.as_u16()))
    }
}

fn add_pb(mp: &MultiProgress, template: &str, message: String) -> ProgressBar {
    let pb = mp.add(ProgressBar::new(1));
    pb.set_style(ProgressStyle::with_template(template).unwrap());
    pb.enable_steady_tick(Duration::from_millis(TICK_SPEED));
    pb.set_message(message);

    pb
}

fn finish_pb(pb: &ProgressBar, message: String, template: &str) {
    pb.set_style(ProgressStyle::with_template(template).unwrap());
    pb.finish_with_message(message);
}

enum Protocols {
    Http,
    Websocket,
}

impl Display for Protocols {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            Self::Http => f.write_str("http"),
            Self::Websocket => f.write_str("ws"),
        }
    }
}

#[derive(Debug, Deserialize)]
struct ErrorResponse {
    error: String,
}