mirror of
https://github.com/kristoferssolo/traxor.git
synced 2025-10-21 20:10:35 +00:00
feat: use filecaster crate
This commit is contained in:
parent
eff0fe20e4
commit
46486a3409
32
Cargo.lock
generated
32
Cargo.lock
generated
@ -314,15 +314,6 @@ dependencies = [
|
|||||||
"powerfmt",
|
"powerfmt",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "derive_macro"
|
|
||||||
version = "0.1.0"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "derive_more"
|
name = "derive_more"
|
||||||
version = "2.0.1"
|
version = "2.0.1"
|
||||||
@ -438,6 +429,27 @@ dependencies = [
|
|||||||
"once_cell",
|
"once_cell",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "filecaster"
|
||||||
|
version = "0.2.3"
|
||||||
|
dependencies = [
|
||||||
|
"filecaster-derive",
|
||||||
|
"merge",
|
||||||
|
"serde",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "filecaster-derive"
|
||||||
|
version = "0.2.3"
|
||||||
|
dependencies = [
|
||||||
|
"merge",
|
||||||
|
"proc-macro-error2",
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"serde",
|
||||||
|
"syn",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fnv"
|
name = "fnv"
|
||||||
version = "1.0.7"
|
version = "1.0.7"
|
||||||
@ -2045,9 +2057,9 @@ version = "0.1.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"color-eyre",
|
"color-eyre",
|
||||||
"crossterm 0.29.0",
|
"crossterm 0.29.0",
|
||||||
"derive_macro",
|
|
||||||
"derive_more",
|
"derive_more",
|
||||||
"dirs",
|
"dirs",
|
||||||
|
"filecaster",
|
||||||
"merge",
|
"merge",
|
||||||
"ratatui",
|
"ratatui",
|
||||||
"serde",
|
"serde",
|
||||||
|
|||||||
@ -6,9 +6,11 @@ license = "GPLv3"
|
|||||||
edition = "2024"
|
edition = "2024"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
filecaster = { version = "0.2", features = [
|
||||||
|
"merge",
|
||||||
|
], path = "../filecaster/filecaster/" }
|
||||||
color-eyre = "0.6"
|
color-eyre = "0.6"
|
||||||
crossterm = "0.29"
|
crossterm = "0.29"
|
||||||
derive_macro = { path = "derive_macro" }
|
|
||||||
derive_more = { version = "2.0", features = ["display"] }
|
derive_more = { version = "2.0", features = ["display"] }
|
||||||
dirs = "6.0"
|
dirs = "6.0"
|
||||||
merge = "0.2"
|
merge = "0.2"
|
||||||
|
|||||||
47
derive_macro/Cargo.lock
generated
47
derive_macro/Cargo.lock
generated
@ -1,47 +0,0 @@
|
|||||||
# This file is automatically @generated by Cargo.
|
|
||||||
# It is not intended for manual editing.
|
|
||||||
version = 4
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "derive_macro"
|
|
||||||
version = "0.1.0"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "proc-macro2"
|
|
||||||
version = "1.0.95"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "02b3e5e68a3a1a02aad3ec490a98007cbc13c37cbe84a3cd7b8e406d76e7f778"
|
|
||||||
dependencies = [
|
|
||||||
"unicode-ident",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "quote"
|
|
||||||
version = "1.0.40"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "1885c039570dc00dcb4ff087a89e185fd56bae234ddc7f056a945bf36467248d"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "syn"
|
|
||||||
version = "2.0.104"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "17b6f705963418cdb9927482fa304bc562ece2fdd4f616084c50b7023b435a40"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"unicode-ident",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "unicode-ident"
|
|
||||||
version = "1.0.18"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5a5f39404a5da50712a4c1eecf25e90dd62b613502b7e925fd4e4d19b5c96512"
|
|
||||||
@ -1,12 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "derive_macro"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2024"
|
|
||||||
|
|
||||||
[lib]
|
|
||||||
crate-type = ["proc-macro"]
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
proc-macro2 = "1.0"
|
|
||||||
quote = "1.0"
|
|
||||||
syn = { version = "2.0", features = ["full", "extra-traits"] }
|
|
||||||
@ -1,42 +0,0 @@
|
|||||||
use proc_macro::TokenStream;
|
|
||||||
use quote::quote;
|
|
||||||
use syn::{Data, DeriveInput, Fields, Ident};
|
|
||||||
|
|
||||||
pub fn impl_from_file(input: DeriveInput) -> TokenStream {
|
|
||||||
let name = &input.ident;
|
|
||||||
let file_name = format!("{name}File");
|
|
||||||
let file_ident = Ident::new(&file_name, name.span());
|
|
||||||
|
|
||||||
let fields = match &input.data {
|
|
||||||
Data::Struct(data) => match &data.fields {
|
|
||||||
Fields::Named(fields) => &fields.named,
|
|
||||||
_ => panic!("Only named fields are supported"),
|
|
||||||
},
|
|
||||||
_ => panic!("Only structs are supported"),
|
|
||||||
};
|
|
||||||
|
|
||||||
let field_idents = fields.iter().map(|f| &f.ident);
|
|
||||||
let field_idents2 = field_idents.clone();
|
|
||||||
|
|
||||||
quote! {
|
|
||||||
impl #name {
|
|
||||||
fn from_file(file: Option<#file_ident>) -> Self
|
|
||||||
where
|
|
||||||
#file_ident: Default + Clone
|
|
||||||
{
|
|
||||||
let default = #file_ident::default();
|
|
||||||
let file = file.unwrap_or_else(|| default.clone());
|
|
||||||
|
|
||||||
Self {
|
|
||||||
#(#field_idents: file.#field_idents2.unwrap_or_else(|| default.#field_idents.clone().unwrap())),*
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<Option<#file_ident>> for #name {
|
|
||||||
fn from(value: Option<#file_ident>) -> Self {
|
|
||||||
Self::from_file(value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}.into()
|
|
||||||
}
|
|
||||||
@ -1,10 +0,0 @@
|
|||||||
mod from_file;
|
|
||||||
|
|
||||||
use proc_macro::TokenStream;
|
|
||||||
use syn::{DeriveInput, parse_macro_input};
|
|
||||||
|
|
||||||
#[proc_macro_derive(FromFile)]
|
|
||||||
pub fn derive_from_file(input: TokenStream) -> TokenStream {
|
|
||||||
let input = parse_macro_input!(input as DeriveInput);
|
|
||||||
from_file::impl_from_file(input)
|
|
||||||
}
|
|
||||||
@ -1,34 +1,13 @@
|
|||||||
use derive_macro::FromFile;
|
use filecaster::FromFile;
|
||||||
use merge::{Merge, option::overwrite_none};
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize, Serialize, Merge)]
|
|
||||||
pub struct ColorConfigFile {
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub highlight_background: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub highlight_foreground: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub header_foreground: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub info_foreground: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, FromFile)]
|
#[derive(Debug, Clone, FromFile)]
|
||||||
pub struct ColorConfig {
|
pub struct ColorConfig {
|
||||||
|
#[from_file(default = "magenta")]
|
||||||
pub highlight_background: String,
|
pub highlight_background: String,
|
||||||
|
#[from_file(default = "black")]
|
||||||
pub highlight_foreground: String,
|
pub highlight_foreground: String,
|
||||||
|
#[from_file(default = "yellow")]
|
||||||
pub header_foreground: String,
|
pub header_foreground: String,
|
||||||
|
#[from_file(default = "blue")]
|
||||||
pub info_foreground: String,
|
pub info_foreground: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for ColorConfigFile {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self {
|
|
||||||
highlight_background: Some("magenta".to_string()),
|
|
||||||
highlight_foreground: Some("black".to_string()),
|
|
||||||
header_foreground: Some("yellow".to_string()),
|
|
||||||
info_foreground: Some("blue".to_string()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@ -1,78 +1,35 @@
|
|||||||
use derive_macro::FromFile;
|
use filecaster::FromFile;
|
||||||
use merge::{Merge, option::overwrite_none};
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize, Serialize, Merge)]
|
|
||||||
pub struct KeybindsConfigFile {
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub quit: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub next_tab: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub prev_tab: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub next_torrent: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub prev_torrent: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub switch_tab_1: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub switch_tab_2: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub switch_tab_3: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub toggle_torrent: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub toggle_all: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub delete: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub delete_force: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub select: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub toggle_help: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub move_torrent: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, FromFile)]
|
#[derive(Debug, Clone, FromFile)]
|
||||||
pub struct KeybindsConfig {
|
pub struct KeybindsConfig {
|
||||||
|
#[from_file(default = "q")]
|
||||||
pub quit: String,
|
pub quit: String,
|
||||||
|
#[from_file(default = "l")]
|
||||||
pub next_tab: String,
|
pub next_tab: String,
|
||||||
|
#[from_file(default = "h")]
|
||||||
pub prev_tab: String,
|
pub prev_tab: String,
|
||||||
|
#[from_file(default = "j")]
|
||||||
pub next_torrent: String,
|
pub next_torrent: String,
|
||||||
|
#[from_file(default = "k")]
|
||||||
pub prev_torrent: String,
|
pub prev_torrent: String,
|
||||||
|
#[from_file(default = "1")]
|
||||||
pub switch_tab_1: String,
|
pub switch_tab_1: String,
|
||||||
|
#[from_file(default = "2")]
|
||||||
pub switch_tab_2: String,
|
pub switch_tab_2: String,
|
||||||
|
#[from_file(default = "3")]
|
||||||
pub switch_tab_3: String,
|
pub switch_tab_3: String,
|
||||||
|
#[from_file(default = "enter")]
|
||||||
pub toggle_torrent: String,
|
pub toggle_torrent: String,
|
||||||
|
#[from_file(default = "a")]
|
||||||
pub toggle_all: String,
|
pub toggle_all: String,
|
||||||
|
#[from_file(default = "d")]
|
||||||
pub delete: String,
|
pub delete: String,
|
||||||
|
#[from_file(default = "D")]
|
||||||
pub delete_force: String,
|
pub delete_force: String,
|
||||||
|
#[from_file(default = " ")]
|
||||||
pub select: String,
|
pub select: String,
|
||||||
|
#[from_file(default = "?")]
|
||||||
pub toggle_help: String,
|
pub toggle_help: String,
|
||||||
|
#[from_file(default = "m")]
|
||||||
pub move_torrent: String,
|
pub move_torrent: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for KeybindsConfigFile {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self {
|
|
||||||
quit: Some("q".to_string()),
|
|
||||||
next_tab: Some("l".to_string()),
|
|
||||||
prev_tab: Some("h".to_string()),
|
|
||||||
next_torrent: Some("j".to_string()),
|
|
||||||
prev_torrent: Some("k".to_string()),
|
|
||||||
switch_tab_1: Some("1".to_string()),
|
|
||||||
switch_tab_2: Some("2".to_string()),
|
|
||||||
switch_tab_3: Some("3".to_string()),
|
|
||||||
toggle_torrent: Some("enter".to_string()),
|
|
||||||
toggle_all: Some("a".to_string()),
|
|
||||||
delete: Some("d".to_string()),
|
|
||||||
delete_force: Some("D".to_string()),
|
|
||||||
select: Some(" ".to_string()),
|
|
||||||
toggle_help: Some("?".to_string()),
|
|
||||||
move_torrent: Some("m".to_string()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@ -1,30 +1,11 @@
|
|||||||
use derive_macro::FromFile;
|
use filecaster::FromFile;
|
||||||
use merge::{Merge, option::overwrite_none};
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Deserialize, Serialize, Merge)]
|
|
||||||
pub struct LogConfigFile {
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub traxor: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub ratatui: Option<String>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub transmission_rpc: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, FromFile)]
|
#[derive(Debug, Clone, FromFile)]
|
||||||
pub struct LogConfig {
|
pub struct LogConfig {
|
||||||
|
#[from_file(default = "warn")]
|
||||||
pub traxor: String,
|
pub traxor: String,
|
||||||
|
#[from_file(default = "warn")]
|
||||||
pub ratatui: String,
|
pub ratatui: String,
|
||||||
|
#[from_file(default = "warn")]
|
||||||
pub transmission_rpc: String,
|
pub transmission_rpc: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for LogConfigFile {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self {
|
|
||||||
traxor: Some("warn".to_string()),
|
|
||||||
ratatui: Some("warn".to_string()),
|
|
||||||
transmission_rpc: Some("warn".to_string()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@ -2,32 +2,22 @@ pub mod color;
|
|||||||
pub mod keybinds;
|
pub mod keybinds;
|
||||||
pub mod log;
|
pub mod log;
|
||||||
|
|
||||||
use color::{ColorConfig, ColorConfigFile};
|
use color::ColorConfig;
|
||||||
use color_eyre::{
|
use color_eyre::{
|
||||||
Result,
|
Result,
|
||||||
eyre::{Context, ContextCompat, Ok},
|
eyre::{Context, ContextCompat, Ok},
|
||||||
};
|
};
|
||||||
use keybinds::{KeybindsConfig, KeybindsConfigFile};
|
use filecaster::FromFile;
|
||||||
use log::{LogConfig, LogConfigFile};
|
use keybinds::KeybindsConfig;
|
||||||
use merge::{Merge, option::overwrite_none};
|
use log::LogConfig;
|
||||||
use serde::{Deserialize, Serialize};
|
use merge::Merge;
|
||||||
use std::{
|
use std::{
|
||||||
fs::read_to_string,
|
fs::read_to_string,
|
||||||
path::{Path, PathBuf},
|
path::{Path, PathBuf},
|
||||||
};
|
};
|
||||||
use tracing::{debug, info, warn};
|
use tracing::{debug, info, warn};
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, Deserialize, Serialize, Merge)]
|
#[derive(Debug, Clone, FromFile)]
|
||||||
pub struct ConfigFile {
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub keybinds: Option<KeybindsConfigFile>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub colors: Option<ColorConfigFile>,
|
|
||||||
#[merge(strategy = overwrite_none)]
|
|
||||||
pub log: Option<LogConfigFile>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
pub keybinds: KeybindsConfig,
|
pub keybinds: KeybindsConfig,
|
||||||
pub colors: ColorConfig,
|
pub colors: ColorConfig,
|
||||||
@ -56,16 +46,6 @@ impl Config {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<ConfigFile> for Config {
|
|
||||||
fn from(value: ConfigFile) -> Self {
|
|
||||||
Self {
|
|
||||||
keybinds: value.keybinds.into(),
|
|
||||||
colors: value.colors.into(),
|
|
||||||
log: value.log.into(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tracing::instrument(name = "Getting config path")]
|
#[tracing::instrument(name = "Getting config path")]
|
||||||
fn get_config_path() -> Result<PathBuf> {
|
fn get_config_path() -> Result<PathBuf> {
|
||||||
let config_dir =
|
let config_dir =
|
||||||
|
|||||||
@ -120,3 +120,34 @@ async fn test_get_action_unhandled() {
|
|||||||
None
|
None
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_get_action_toggle_help() {
|
||||||
|
let config = Config::load().unwrap();
|
||||||
|
let mut app = App::new(config).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
get_action(KeyEvent::from(KeyCode::Char('?')), &mut app)
|
||||||
|
.await
|
||||||
|
.unwrap(),
|
||||||
|
Some(Action::ToggleHelp)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_get_action_input_mode() {
|
||||||
|
let config = Config::load().unwrap();
|
||||||
|
let mut app = App::new(config).unwrap();
|
||||||
|
app.input_mode = true;
|
||||||
|
assert_eq!(
|
||||||
|
get_action(KeyEvent::from(KeyCode::Enter), &mut app)
|
||||||
|
.await
|
||||||
|
.unwrap(),
|
||||||
|
Some(Action::Submit)
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
get_action(KeyEvent::from(KeyCode::Esc), &mut app)
|
||||||
|
.await
|
||||||
|
.unwrap(),
|
||||||
|
Some(Action::Cancel)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user