225 lines
7.8 KiB
Rust
Raw Normal View History

2021-12-08 03:09:45 +00:00
#![deny(clippy::all)]
2023-09-17 19:37:02 -04:00
#![allow(clippy::comparison_chain, clippy::upper_case_acronyms)]
2021-12-09 16:11:52 -05:00
#![deny(unused_must_use)]
#![recursion_limit = "256"]
2021-11-27 21:34:08 -05:00
2023-12-14 17:23:43 -05:00
use crate::{settings::NamedSocketAddrs, tools::*};
use clap::{Parser, ValueEnum};
2021-11-22 11:28:30 -05:00
use flexi_logger::*;
2023-12-14 17:23:43 -05:00
use std::path::PathBuf;
2021-11-22 11:28:30 -05:00
2023-12-09 12:27:44 -05:00
mod cached_text_view;
2021-11-22 11:28:30 -05:00
mod client_api_connection;
mod command_processor;
2022-09-06 16:49:43 -04:00
mod peers_table_view;
2021-11-22 11:28:30 -05:00
mod settings;
2022-06-27 23:46:29 -04:00
mod tools;
2021-11-22 11:28:30 -05:00
mod ui;
#[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, ValueEnum, Debug)]
enum LogLevel {
/// Turn on debug logging
Debug,
/// Turn on trace logging
Trace,
}
#[derive(Parser, Debug)]
#[command(author, version, about = "Veilid Console Client")]
struct CmdlineArgs {
2023-12-14 17:23:43 -05:00
/// IPC socket to connect to
#[arg(long, short = 'p')]
ipc_path: Option<PathBuf>,
2023-12-14 21:28:02 -05:00
/// Subnode index to use when connecting
2023-12-14 17:23:43 -05:00
#[arg(long, short = 'i', default_value = "0")]
subnode_index: usize,
/// Address to connect to
2023-12-14 17:23:43 -05:00
#[arg(long, short = 'a')]
address: Option<String>,
/// Wait for debugger to attach
#[arg(long)]
wait_for_debug: bool,
/// Specify a configuration file to use
2023-12-14 17:23:43 -05:00
#[arg(short = 'c', long, value_name = "FILE")]
config_file: Option<PathBuf>,
/// log level
#[arg(value_enum)]
log_level: Option<LogLevel>,
2021-11-22 11:28:30 -05:00
}
2022-06-27 23:46:29 -04:00
fn main() -> Result<(), String> {
2021-11-22 11:28:30 -05:00
// Get command line options
let default_config_path = settings::Settings::get_default_config_path();
let args = CmdlineArgs::parse();
if args.wait_for_debug {
2021-11-22 11:28:30 -05:00
use bugsalot::debugger;
debugger::wait_until_attached(None).expect("state() not implemented on this platform");
}
// Attempt to load configuration
let settings_path = args.config_file.unwrap_or(default_config_path);
let settings_path = if settings_path.exists() {
Some(settings_path.into_os_string())
2022-06-26 17:00:05 -04:00
} else {
None
};
2023-09-17 14:14:39 -04:00
let mut settings = settings::Settings::new(settings_path.as_deref())
2022-06-26 17:00:05 -04:00
.map_err(|e| format!("configuration is invalid: {}", e))?;
2021-11-22 11:28:30 -05:00
// Set config from command line
if let Some(LogLevel::Debug) = args.log_level {
2021-11-22 11:28:30 -05:00
settings.logging.level = settings::LogLevel::Debug;
settings.logging.terminal.enabled = true;
}
if let Some(LogLevel::Trace) = args.log_level {
2021-11-22 11:28:30 -05:00
settings.logging.level = settings::LogLevel::Trace;
settings.logging.terminal.enabled = true;
}
// Create UI object
2023-06-08 14:07:09 -04:00
let (mut sivui, uisender) = ui::UI::new(settings.interface.node_log.scrollback, &settings);
2021-11-22 11:28:30 -05:00
// Set up loggers
{
let mut specbuilder = LogSpecBuilder::new();
specbuilder.default(settings::convert_loglevel(settings.logging.level));
2022-07-01 16:20:43 -04:00
specbuilder.module("cursive", LevelFilter::Off);
2021-11-22 11:28:30 -05:00
specbuilder.module("cursive_core", LevelFilter::Off);
specbuilder.module("cursive_buffered_backend", LevelFilter::Off);
2022-07-02 11:41:25 -04:00
specbuilder.module("tokio_util", LevelFilter::Off);
2021-11-22 11:28:30 -05:00
specbuilder.module("mio", LevelFilter::Off);
specbuilder.module("async_std", LevelFilter::Off);
specbuilder.module("async_io", LevelFilter::Off);
specbuilder.module("polling", LevelFilter::Off);
let logger = Logger::with(specbuilder.build());
if settings.logging.terminal.enabled {
if settings.logging.file.enabled {
std::fs::create_dir_all(settings.logging.file.directory.clone())
.map_err(map_to_string)?;
2021-11-22 11:28:30 -05:00
logger
2022-06-26 17:00:05 -04:00
.log_to_file_and_writer(
FileSpec::default()
.directory(settings.logging.file.directory.clone())
.suppress_timestamp(),
2023-12-09 12:27:44 -05:00
Box::new(uisender.clone()),
2022-06-26 17:00:05 -04:00
)
2021-11-22 11:28:30 -05:00
.start()
.expect("failed to initialize logger!");
} else {
logger
2023-12-09 12:27:44 -05:00
.log_to_writer(Box::new(uisender.clone()))
2021-11-22 11:28:30 -05:00
.start()
.expect("failed to initialize logger!");
}
} else if settings.logging.file.enabled {
std::fs::create_dir_all(settings.logging.file.directory.clone())
.map_err(map_to_string)?;
2021-11-22 11:28:30 -05:00
logger
2022-06-26 17:00:05 -04:00
.log_to_file(
FileSpec::default()
.directory(settings.logging.file.directory.clone())
.suppress_timestamp(),
)
2021-11-22 11:28:30 -05:00
.start()
.expect("failed to initialize logger!");
}
}
2023-12-14 17:23:43 -05:00
2021-11-22 11:28:30 -05:00
// Get client address
2023-12-14 21:42:58 -05:00
let enable_ipc = settings.enable_ipc && args.address.is_none();
let mut enable_network = settings.enable_network && args.ipc_path.is_none();
2023-12-14 17:23:43 -05:00
// Determine IPC path to try
let mut client_api_ipc_path = None;
if enable_ipc {
if let Some(ipc_path) = args.ipc_path.or(settings.ipc_path.clone()) {
if ipc_path.exists() && !ipc_path.is_dir() {
// try direct path
enable_network = false;
client_api_ipc_path = Some(ipc_path);
} else if ipc_path.exists() && ipc_path.is_dir() {
// try subnode index inside path
let ipc_path = ipc_path.join(args.subnode_index.to_string());
if ipc_path.exists() && !ipc_path.is_dir() {
// subnode indexed path exists
enable_network = false;
client_api_ipc_path = Some(ipc_path);
}
}
}
}
let mut client_api_network_addresses = None;
if enable_network {
let args_address = if let Some(args_address) = args.address {
match NamedSocketAddrs::try_from(args_address) {
Ok(v) => Some(v),
Err(e) => {
return Err(format!("Invalid server address: {}", e));
}
}
} else {
None
};
if let Some(address_arg) = args_address.or(settings.address.clone()) {
client_api_network_addresses = Some(address_arg.addrs);
} else if let Some(address) = settings.address.clone() {
client_api_network_addresses = Some(address.addrs.clone());
}
}
2021-11-22 11:28:30 -05:00
// Create command processor
debug!("Creating Command Processor ");
2023-06-08 14:07:09 -04:00
let comproc = command_processor::CommandProcessor::new(uisender, &settings);
2021-11-22 11:28:30 -05:00
sivui.set_command_processor(comproc.clone());
// Create client api client side
info!("Starting API connection");
2023-06-08 14:07:09 -04:00
let capi = client_api_connection::ClientApiConnection::new(comproc.clone());
2021-11-22 11:28:30 -05:00
// Save client api in command processor
comproc.set_client_api_connection(capi.clone());
// Keep a connection to the server
2023-12-14 17:23:43 -05:00
if let Some(client_api_ipc_path) = client_api_ipc_path {
comproc.set_ipc_path(Some(client_api_ipc_path));
} else if let Some(client_api_network_address) = client_api_network_addresses {
let network_addr = client_api_network_address.first().cloned();
comproc.set_network_address(network_addr);
}
2023-06-08 14:07:09 -04:00
let comproc2 = comproc.clone();
2021-11-22 11:28:30 -05:00
let connection_future = comproc.connection_manager();
2022-06-27 23:46:29 -04:00
// Start async
block_on(async move {
// Start UI
let ui_future = async move {
sivui.run_async().await;
// When UI quits, close connection and command processor cleanly
comproc2.quit();
capi.disconnect().await;
};
cfg_if! {
if #[cfg(feature="rt-async-std")] {
use async_std::prelude::*;
// Wait for ui and connection to complete
let _ = ui_future.join(connection_future).await;
} else if #[cfg(feature="rt-tokio")] {
// Wait for ui and connection to complete
let _ = tokio::join!(ui_future, connection_future);
2023-08-29 15:15:47 -05:00
} else {
compile_error!("needs executor implementation")
2022-06-27 23:46:29 -04:00
}
}
2021-11-22 11:28:30 -05:00
});
Ok(())
}