295 lines
7.5 KiB
Rust
295 lines
7.5 KiB
Rust
use std::process;
|
|
|
|
use anyhow::{anyhow, Error};
|
|
use colored::Colorize;
|
|
use log::{debug, error};
|
|
use serde::{Deserialize, Serialize};
|
|
use tokio::sync::mpsc::Sender;
|
|
use tokio_util::sync::CancellationToken;
|
|
|
|
use crate::app::context_clues::{build_context_clue_string, SERVARR_CONTEXT_CLUES};
|
|
use crate::cli::Command;
|
|
use crate::models::servarr_data::radarr::radarr_data::{ActiveRadarrBlock, RadarrData};
|
|
use crate::models::servarr_data::sonarr::sonarr_data::{ActiveSonarrBlock, SonarrData};
|
|
use crate::models::{HorizontallyScrollableText, Route, TabRoute, TabState};
|
|
use crate::network::NetworkEvent;
|
|
|
|
#[cfg(test)]
|
|
#[path = "app_tests.rs"]
|
|
mod app_tests;
|
|
pub mod context_clues;
|
|
pub mod key_binding;
|
|
mod key_binding_tests;
|
|
pub mod radarr;
|
|
pub mod sonarr;
|
|
|
|
pub struct App<'a> {
|
|
navigation_stack: Vec<Route>,
|
|
network_tx: Option<Sender<NetworkEvent>>,
|
|
pub cancellation_token: CancellationToken,
|
|
pub is_first_render: bool,
|
|
pub server_tabs: TabState,
|
|
pub error: HorizontallyScrollableText,
|
|
pub tick_until_poll: u64,
|
|
pub ticks_until_scroll: u64,
|
|
pub tick_count: u64,
|
|
pub is_routing: bool,
|
|
pub is_loading: bool,
|
|
pub should_refresh: bool,
|
|
pub should_ignore_quit_key: bool,
|
|
pub cli_mode: bool,
|
|
pub config: AppConfig,
|
|
pub data: Data<'a>,
|
|
}
|
|
|
|
impl<'a> App<'a> {
|
|
pub fn new(
|
|
network_tx: Sender<NetworkEvent>,
|
|
config: AppConfig,
|
|
cancellation_token: CancellationToken,
|
|
) -> Self {
|
|
let mut server_tabs = Vec::new();
|
|
|
|
if config.radarr.is_some() {
|
|
server_tabs.push(TabRoute {
|
|
title: "Radarr",
|
|
route: ActiveRadarrBlock::Movies.into(),
|
|
help: format!(
|
|
"<↑↓> scroll | ←→ change tab | {} ",
|
|
build_context_clue_string(&SERVARR_CONTEXT_CLUES)
|
|
),
|
|
contextual_help: None,
|
|
});
|
|
}
|
|
|
|
if config.sonarr.is_some() {
|
|
server_tabs.push(TabRoute {
|
|
title: "Sonarr",
|
|
route: ActiveSonarrBlock::Series.into(),
|
|
help: format!(
|
|
"<↑↓> scroll | ←→ change tab | {} ",
|
|
build_context_clue_string(&SERVARR_CONTEXT_CLUES)
|
|
),
|
|
contextual_help: None,
|
|
});
|
|
}
|
|
|
|
App {
|
|
network_tx: Some(network_tx),
|
|
config,
|
|
cancellation_token,
|
|
server_tabs: TabState::new(server_tabs),
|
|
..App::default()
|
|
}
|
|
}
|
|
|
|
pub async fn dispatch_network_event(&mut self, action: NetworkEvent) {
|
|
debug!("Dispatching network event: {action:?}");
|
|
|
|
if !self.should_refresh {
|
|
self.is_loading = true;
|
|
}
|
|
|
|
if let Some(network_tx) = &self.network_tx {
|
|
if let Err(e) = network_tx.send(action).await {
|
|
self.is_loading = false;
|
|
error!("Failed to send event. {e:?}");
|
|
self.handle_error(anyhow!(e));
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn reset_tick_count(&mut self) {
|
|
self.tick_count = 0;
|
|
}
|
|
|
|
#[allow(dead_code)]
|
|
pub fn reset(&mut self) {
|
|
self.reset_tick_count();
|
|
self.error = HorizontallyScrollableText::default();
|
|
self.is_first_render = true;
|
|
self.data = Data::default();
|
|
}
|
|
|
|
pub fn handle_error(&mut self, error: Error) {
|
|
if self.error.text.is_empty() {
|
|
self.error = error.to_string().into();
|
|
}
|
|
}
|
|
|
|
pub async fn on_tick(&mut self) {
|
|
if self.tick_count % self.tick_until_poll == 0 || self.is_routing || self.should_refresh {
|
|
match self.get_current_route() {
|
|
Route::Radarr(active_radarr_block, _) => self.radarr_on_tick(active_radarr_block).await,
|
|
Route::Sonarr(active_sonarr_block, _) => self.sonarr_on_tick(active_sonarr_block).await,
|
|
_ => (),
|
|
}
|
|
|
|
self.is_routing = false;
|
|
self.should_refresh = false;
|
|
}
|
|
|
|
self.tick_count += 1;
|
|
}
|
|
|
|
pub fn push_navigation_stack(&mut self, route: Route) {
|
|
self.navigation_stack.push(route);
|
|
self.is_routing = true;
|
|
}
|
|
|
|
pub fn pop_navigation_stack(&mut self) {
|
|
self.is_routing = true;
|
|
if !self.navigation_stack.is_empty() {
|
|
self.navigation_stack.pop();
|
|
}
|
|
}
|
|
|
|
pub fn reset_cancellation_token(&mut self) -> CancellationToken {
|
|
self.cancellation_token = CancellationToken::new();
|
|
self.should_refresh = true;
|
|
self.is_loading = false;
|
|
|
|
self.cancellation_token.clone()
|
|
}
|
|
|
|
pub fn pop_and_push_navigation_stack(&mut self, route: Route) {
|
|
self.pop_navigation_stack();
|
|
self.push_navigation_stack(route);
|
|
}
|
|
|
|
pub fn get_current_route(&self) -> Route {
|
|
*self
|
|
.navigation_stack
|
|
.last()
|
|
.unwrap_or(&self.server_tabs.tabs.first().unwrap().route)
|
|
}
|
|
}
|
|
|
|
impl<'a> Default for App<'a> {
|
|
fn default() -> Self {
|
|
App {
|
|
navigation_stack: Vec::new(),
|
|
network_tx: None,
|
|
cancellation_token: CancellationToken::new(),
|
|
error: HorizontallyScrollableText::default(),
|
|
is_first_render: true,
|
|
server_tabs: TabState::new(vec![
|
|
TabRoute {
|
|
title: "Radarr",
|
|
route: ActiveRadarrBlock::Movies.into(),
|
|
help: format!(
|
|
"<↑↓> scroll | ←→ change tab | {} ",
|
|
build_context_clue_string(&SERVARR_CONTEXT_CLUES)
|
|
),
|
|
contextual_help: None,
|
|
},
|
|
TabRoute {
|
|
title: "Sonarr",
|
|
route: ActiveSonarrBlock::Series.into(),
|
|
help: format!(
|
|
"<↑↓> scroll | ←→ change tab | {} ",
|
|
build_context_clue_string(&SERVARR_CONTEXT_CLUES)
|
|
),
|
|
contextual_help: None,
|
|
},
|
|
]),
|
|
tick_until_poll: 400,
|
|
ticks_until_scroll: 4,
|
|
tick_count: 0,
|
|
is_loading: false,
|
|
is_routing: false,
|
|
should_refresh: false,
|
|
should_ignore_quit_key: false,
|
|
cli_mode: false,
|
|
config: AppConfig::default(),
|
|
data: Data::default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Default)]
|
|
pub struct Data<'a> {
|
|
pub radarr_data: RadarrData<'a>,
|
|
pub sonarr_data: SonarrData<'a>,
|
|
}
|
|
|
|
#[derive(Debug, Deserialize, Serialize, Default, Clone)]
|
|
pub struct AppConfig {
|
|
pub radarr: Option<ServarrConfig>,
|
|
pub sonarr: Option<ServarrConfig>,
|
|
}
|
|
|
|
impl AppConfig {
|
|
pub fn validate(&self) {
|
|
if self.radarr.is_none() && self.sonarr.is_none() {
|
|
log_and_print_error(
|
|
"No Servarr configuration provided in the specified configuration file".to_owned(),
|
|
);
|
|
process::exit(1);
|
|
}
|
|
|
|
if let Some(radarr_config) = &self.radarr {
|
|
radarr_config.validate();
|
|
}
|
|
|
|
if let Some(sonarr_config) = &self.sonarr {
|
|
sonarr_config.validate();
|
|
}
|
|
}
|
|
|
|
pub fn verify_config_present_for_cli(&self, command: &Command) {
|
|
let msg = |servarr: &str| {
|
|
log_and_print_error(format!(
|
|
"{} configuration missing; Unable to run any {} commands.",
|
|
servarr, servarr
|
|
))
|
|
};
|
|
match command {
|
|
Command::Radarr(_) if self.radarr.is_none() => {
|
|
msg("Radarr");
|
|
process::exit(1);
|
|
}
|
|
Command::Sonarr(_) if self.sonarr.is_none() => {
|
|
msg("Sonarr");
|
|
process::exit(1);
|
|
}
|
|
_ => (),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Deserialize, Serialize, Clone)]
|
|
pub struct ServarrConfig {
|
|
pub host: Option<String>,
|
|
pub port: Option<u16>,
|
|
pub uri: Option<String>,
|
|
pub api_token: String,
|
|
pub ssl_cert_path: Option<String>,
|
|
}
|
|
|
|
impl ServarrConfig {
|
|
fn validate(&self) {
|
|
if self.host.is_none() && self.uri.is_none() {
|
|
log_and_print_error("'host' or 'uri' is required for configuration".to_owned());
|
|
process::exit(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Default for ServarrConfig {
|
|
fn default() -> Self {
|
|
ServarrConfig {
|
|
host: Some("localhost".to_string()),
|
|
port: None,
|
|
uri: None,
|
|
api_token: "".to_string(),
|
|
ssl_cert_path: None,
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn log_and_print_error(error: String) {
|
|
error!("{}", error);
|
|
eprintln!("error: {}", error.red());
|
|
}
|