refactor(units): implement Formatter trait

This commit is contained in:
Kristofers Solo 2025-07-08 00:55:29 +03:00
parent 9153e27ce5
commit 973b4657fe
Signed by: kristoferssolo
GPG Key ID: 8687F2D3EEE6F0ED
4 changed files with 161 additions and 288 deletions

View File

@ -1,14 +1,14 @@
use super::unit::{Unit, UnitDisplay}; use super::unit::{Unit, UnitDisplay};
use crate::impl_unit_wrapper; use crate::impl_unit_newtype;
use std::fmt::Display; use std::fmt::Display;
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Default)] #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Default)]
pub struct FileSize(Unit); pub struct FileSize(Unit);
impl_unit_wrapper!(FileSize); impl_unit_newtype!(FileSize);
impl FileSize { impl FileSize {
pub fn new(bytes: u64) -> Self { pub fn new(bytes: u64) -> Self {
Self(Unit::new(bytes)) Self(Unit::from_raw(bytes))
} }
} }

View File

@ -5,7 +5,7 @@ pub mod unit;
use filesize::FileSize; use filesize::FileSize;
use netspeed::NetSpeed; use netspeed::NetSpeed;
use transmission_rpc::types::{ use transmission_rpc::types::{
ErrorType, IdleMode, RatioMode, Torrent, TorrentGetField, TorrentStatus, ErrorType, IdleMode, Priority, RatioMode, Torrent, TorrentGetField, TorrentStatus,
}; };
pub trait Wrapper { pub trait Wrapper {
@ -108,130 +108,47 @@ impl Wrapper for TorrentGetField {
fn value(&self, torrent: &Torrent) -> String { fn value(&self, torrent: &Torrent) -> String {
match self { match self {
Self::ActivityDate => torrent Self::ActivityDate => format_option_string(torrent.activity_date),
.activity_date Self::AddedDate => format_option_string(torrent.added_date),
.map(|v| v.to_string())
.unwrap_or_default(),
Self::AddedDate => torrent
.added_date
.map(|v| v.to_string())
.unwrap_or_default(),
Self::Availability => "N/A".to_string(), Self::Availability => "N/A".to_string(),
Self::BandwidthPriority => torrent Self::BandwidthPriority => torrent.bandwidth_priority.format(),
.bandwidth_priority
.map(|v| format!("{:?}", v))
.unwrap_or_default(),
Self::Comment => torrent.comment.clone().unwrap_or_default(), Self::Comment => torrent.comment.clone().unwrap_or_default(),
Self::CorruptEver => FileSize::try_from(torrent.corrupt_ever.unwrap_or(0)) Self::CorruptEver => FileSize::from(torrent.corrupt_ever).to_string(),
.unwrap_or_default()
.to_string(),
Self::Creator => torrent.creator.clone().unwrap_or_default(), Self::Creator => torrent.creator.clone().unwrap_or_default(),
Self::DateCreated => torrent Self::DateCreated => format_option_string(torrent.date_created),
.date_created Self::DesiredAvailable => FileSize::from(torrent.desired_available).to_string(),
.map(|v| v.to_string()) Self::DoneDate => format_option_string(torrent.done_date),
.unwrap_or_default() Self::DownloadDir => torrent.download_dir.clone().unwrap_or_default(),
.to_string(), Self::DownloadLimit => NetSpeed::from(torrent.download_limit).to_string(),
Self::DesiredAvailable => { Self::DownloadLimited => format_option_string(torrent.download_limited),
FileSize::from(torrent.desired_available.unwrap_or(0)).to_string() Self::DownloadedEver => FileSize::from(torrent.downloaded_ever).to_string(),
} Self::EditDate => format_option_string(torrent.edit_date),
Self::DoneDate => torrent.done_date.map(|v| v.to_string()).unwrap_or_default(), Self::Error => torrent.error.format(),
Self::DownloadDir => torrent.download_dir.clone().unwrap_or_default().to_string(),
Self::DownloadLimit => NetSpeed::from(torrent.download_limit.unwrap_or(0)).to_string(),
Self::DownloadLimited => torrent
.download_limited
.map(|v| v.to_string())
.unwrap_or_default()
.to_string(),
Self::DownloadedEver => {
FileSize::from(torrent.downloaded_ever.unwrap_or(0)).to_string()
}
Self::EditDate => torrent.edit_date.map(|v| v.to_string()).unwrap_or_default(),
Self::Error => match torrent.error {
Some(error) => match error {
ErrorType::Ok => "Ok",
ErrorType::LocalError => "LocalError",
ErrorType::TrackerError => "TrackerError",
ErrorType::TrackerWarning => "TrackerWarning",
},
None => "N/A",
}
.to_string(),
Self::ErrorString => torrent.error_string.clone().unwrap_or_default(), Self::ErrorString => torrent.error_string.clone().unwrap_or_default(),
Self::Eta => match torrent.eta { Self::Eta => format_eta(torrent.eta),
Some(eta) => match eta { Self::EtaIdle => format_option_string(torrent.eta_idle),
-1 => "".to_string(), Self::FileCount => format_option_string(torrent.file_count),
-2 => "?".to_string(), Self::FileStats => torrent.file_stats.format(),
_ => format!("{} s", eta), Self::Files => torrent.files.format(),
},
None => "".to_string(),
},
Self::EtaIdle => torrent.eta_idle.map(|v| v.to_string()).unwrap_or_default(),
Self::FileCount => torrent
.file_count
.map(|v| v.to_string())
.unwrap_or_default(),
Self::FileStats => torrent
.file_stats
.as_ref()
.map(|v| format!("{}", v.len()))
.unwrap_or_default(),
Self::Files => torrent
.files
.as_ref()
.map(|v| format!("{}", v.len()))
.unwrap_or_default(),
Self::Group => torrent.group.clone().unwrap_or_default(), Self::Group => torrent.group.clone().unwrap_or_default(),
Self::HashString => torrent.hash_string.clone().unwrap_or_default(), Self::HashString => torrent.hash_string.clone().unwrap_or_default(),
Self::HaveUnchecked => FileSize::from(torrent.have_unchecked.unwrap_or(0)).to_string(), Self::HaveUnchecked => FileSize::from(torrent.have_unchecked).to_string(),
Self::HaveValid => FileSize::from(torrent.have_valid.unwrap_or(0)).to_string(), Self::HaveValid => FileSize::from(torrent.have_valid).to_string(),
Self::HonorsSessionLimits => torrent Self::HonorsSessionLimits => format_option_string(torrent.honors_session_limits),
.honors_session_limits Self::Id => format_option_string(torrent.id),
.map(|v| v.to_string()) Self::IsFinished => format_option_string(torrent.is_finished),
.unwrap_or_default(), Self::IsPrivate => format_option_string(torrent.is_private),
Self::Id => torrent.id.map(|v| v.to_string()).unwrap_or_default(), Self::IsStalled => format_option_string(torrent.is_stalled),
Self::IsFinished => torrent
.is_finished
.map(|v| v.to_string())
.unwrap_or_default(),
Self::IsPrivate => torrent
.is_private
.map(|v| v.to_string())
.unwrap_or_default(),
Self::IsStalled => torrent
.is_stalled
.map(|v| v.to_string())
.unwrap_or_default(),
Self::Labels => torrent.labels.clone().unwrap_or_default().join(", "), Self::Labels => torrent.labels.clone().unwrap_or_default().join(", "),
Self::LeftUntilDone => FileSize::try_from(torrent.left_until_done.unwrap_or(0)) Self::LeftUntilDone => FileSize::from(torrent.left_until_done).to_string(),
.unwrap_or_default()
.to_string(),
Self::MagnetLink => torrent.magnet_link.clone().unwrap_or_default(), Self::MagnetLink => torrent.magnet_link.clone().unwrap_or_default(),
Self::ManualAnnounceTime => torrent Self::ManualAnnounceTime => format_option_string(torrent.manual_announce_time),
.manual_announce_time Self::MaxConnectedPeers => format_option_string(torrent.max_connected_peers),
.map(|v| v.to_string()) Self::MetadataPercentComplete => torrent.metadata_percent_complete.format(),
.unwrap_or_default(),
Self::MaxConnectedPeers => torrent
.max_connected_peers
.map(|v| v.to_string())
.unwrap_or_default(),
Self::MetadataPercentComplete => torrent
.metadata_percent_complete
.map(|v| format!("{:.2}", v))
.unwrap_or_default(),
Self::Name => torrent.name.clone().unwrap_or_default(), Self::Name => torrent.name.clone().unwrap_or_default(),
Self::PeerLimit => torrent Self::PeerLimit => format_option_string(torrent.peer_limit),
.peer_limit Self::Peers => torrent.peers.format(),
.map(|v| v.to_string()) Self::PeersConnected => format_option_string(torrent.peers_connected),
.unwrap_or_default(),
Self::Peers => torrent
.peers
.as_ref()
.map(|v| format!("{}", v.len()))
.unwrap_or_default(),
Self::PeersConnected => torrent
.peers_connected
.map(|v| v.to_string())
.unwrap_or_default(),
Self::PeersFrom => torrent Self::PeersFrom => torrent
.peers_from .peers_from
.as_ref() .as_ref()
@ -242,149 +159,45 @@ impl Wrapper for TorrentGetField {
) )
}) })
.unwrap_or_default(), .unwrap_or_default(),
Self::PeersGettingFromUs => torrent Self::PeersGettingFromUs => format_option_string(torrent.peers_getting_from_us),
.peers_getting_from_us Self::PeersSendingToUs => format_option_string(torrent.peers_sending_to_us),
.map(|v| v.to_string()) Self::PercentComplete => torrent.percent_complete.format(),
.unwrap_or_default(), Self::PercentDone => torrent.percent_done.format(),
Self::PeersSendingToUs => torrent Self::PieceCount => format_option_string(torrent.piece_count),
.peers_sending_to_us Self::PieceSize => FileSize::from(torrent.piece_size).to_string(),
.map(|v| v.to_string())
.unwrap_or_default(),
Self::PercentComplete => torrent
.percent_complete
.map(|v| format!("{:.2}", v))
.unwrap_or_default(),
Self::PercentDone => torrent
.percent_done
.map(|v| format!("{:.2}", v))
.unwrap_or_default(),
Self::PieceCount => torrent
.piece_count
.map(|v| v.to_string())
.unwrap_or_default(),
Self::PieceSize => FileSize::from(torrent.piece_size.unwrap_or(0)).to_string(),
Self::Pieces => torrent Self::Pieces => torrent
.pieces .pieces
.as_ref() .as_ref()
.map(|p| format!("{} bytes", p.len())) .map(|p| format!("{} bytes", p.len()))
.unwrap_or_default(), .unwrap_or_default(),
Self::PrimaryMimeType => torrent.primary_mime_type.clone().unwrap_or_default(), Self::PrimaryMimeType => torrent.primary_mime_type.clone().unwrap_or_default(),
Self::Priorities => torrent Self::Priorities => torrent.priorities.format(),
.priorities Self::QueuePosition => format_option_string(torrent.queue_position),
.as_ref() Self::RateDownload => NetSpeed::from(torrent.rate_download).to_string(),
.map(|v| format!("{}", v.len())) Self::RateUpload => NetSpeed::from(torrent.rate_upload).to_string(),
.unwrap_or_default(), Self::RecheckProgress => torrent.recheck_progress.format(),
Self::QueuePosition => torrent Self::SecondsDownloading => format_option_string(torrent.seconds_downloading),
.queue_position Self::SecondsSeeding => format_option_string(torrent.seconds_seeding),
.map(|v| v.to_string()) Self::SeedIdleLimit => format_option_string(torrent.seed_idle_limit),
.unwrap_or_default(), Self::SeedIdleMode => torrent.seed_idle_mode.format(),
Self::RateDownload => NetSpeed::try_from(torrent.rate_download.unwrap_or(0)) Self::SeedRatioLimit => torrent.seed_ratio_limit.format(),
.unwrap_or_default() Self::SeedRatioMode => torrent.seed_ratio_mode.format(),
.to_string(), Self::SequentialDownload => format_option_string(torrent.sequential_download),
Self::RateUpload => NetSpeed::try_from(torrent.rate_upload.unwrap_or(0)) Self::SizeWhenDone => FileSize::from(torrent.size_when_done).to_string(),
.unwrap_or_default() Self::StartDate => format_option_string(torrent.start_date),
.to_string(), Self::Status => torrent.status.format(),
Self::RecheckProgress => torrent
.recheck_progress
.map(|v| format!("{:.2}", v))
.unwrap_or_default(),
Self::SecondsDownloading => torrent
.seconds_downloading
.map(|v| v.to_string())
.unwrap_or_default(),
Self::SecondsSeeding => torrent
.seconds_seeding
.map(|v| v.to_string())
.unwrap_or_default(),
Self::SeedIdleLimit => torrent
.seed_idle_limit
.map(|v| v.to_string())
.unwrap_or_default(),
Self::SeedIdleMode => torrent
.seed_idle_mode
.map(|v| match v {
IdleMode::Global => "Global",
IdleMode::Single => "Single",
IdleMode::Unlimited => "Unlimited",
})
.unwrap_or("N/A")
.to_string(),
Self::SeedRatioLimit => torrent
.seed_ratio_limit
.map(|v| format!("{:.2}", v))
.unwrap_or_default(),
Self::SeedRatioMode => torrent
.seed_ratio_mode
.map(|v| match v {
RatioMode::Global => "Global",
RatioMode::Single => "Single",
RatioMode::Unlimited => "Unlimited",
})
.unwrap_or_default()
.to_string(),
Self::SequentialDownload => torrent
.sequential_download
.map(|v| v.to_string())
.unwrap_or_default(),
Self::SizeWhenDone => FileSize::try_from(torrent.size_when_done.unwrap_or(0))
.unwrap_or_default()
.to_string(),
Self::StartDate => torrent
.start_date
.map(|v| v.to_string())
.unwrap_or_default(),
Self::Status => match torrent.status {
Some(status) => match status {
TorrentStatus::Stopped => "Stopped",
TorrentStatus::Seeding => "Seeding",
TorrentStatus::Verifying => "Verifying",
TorrentStatus::Downloading => "Downloading",
TorrentStatus::QueuedToSeed => "QueuedToSeed",
TorrentStatus::QueuedToVerify => "QueuedToVerify",
TorrentStatus::QueuedToDownload => "QueuedToDownload",
},
None => "N/A",
}
.to_string(),
Self::TorrentFile => torrent.torrent_file.clone().unwrap_or_default(), Self::TorrentFile => torrent.torrent_file.clone().unwrap_or_default(),
Self::TotalSize => FileSize::try_from(torrent.total_size.unwrap_or(0)) Self::TotalSize => FileSize::from(torrent.total_size).to_string(),
.unwrap_or_default()
.to_string(),
Self::TrackerList => torrent.tracker_list.clone().unwrap_or_default(), Self::TrackerList => torrent.tracker_list.clone().unwrap_or_default(),
Self::TrackerStats => torrent Self::TrackerStats => torrent.tracker_stats.format(),
.tracker_stats Self::Trackers => torrent.trackers.format(),
.as_ref() Self::UploadLimit => NetSpeed::from(torrent.upload_limit).to_string(),
.map(|v| format!("{}", v.len())) Self::UploadLimited => format_option_string(torrent.upload_limited),
.unwrap_or_default(), Self::UploadRatio => torrent.upload_ratio.format(),
Self::Trackers => torrent Self::UploadedEver => FileSize::from(torrent.uploaded_ever).to_string(),
.trackers Self::Wanted => torrent.wanted.format(),
.as_ref()
.map(|v| format!("{}", v.len()))
.unwrap_or_default(),
Self::UploadLimit => NetSpeed::try_from(torrent.upload_limit.unwrap_or(0))
.unwrap_or_default()
.to_string(),
Self::UploadLimited => torrent
.upload_limited
.map(|v| v.to_string())
.unwrap_or_default(),
Self::UploadRatio => torrent
.upload_ratio
.map(|v| format!("{:.2}", v))
.unwrap_or_default(),
Self::UploadedEver => FileSize::try_from(torrent.uploaded_ever.unwrap_or(0))
.unwrap_or_default()
.to_string(),
Self::Wanted => torrent
.wanted
.as_ref()
.map(|v| format!("{}", v.len()))
.unwrap_or_default(),
Self::Webseeds => torrent.webseeds.clone().unwrap_or_default().join(", "), Self::Webseeds => torrent.webseeds.clone().unwrap_or_default().join(", "),
Self::WebseedsSendingToUs => torrent Self::WebseedsSendingToUs => format_option_string(torrent.webseeds_sending_to_us),
.webseeds_sending_to_us
.map(|v| v.to_string())
.unwrap_or_default(),
} }
} }
@ -470,3 +283,80 @@ impl Wrapper for TorrentGetField {
} }
} }
} }
fn format_option_string<T: ToString>(value: Option<T>) -> String {
value.map(|v| v.to_string()).unwrap_or_default()
}
fn format_eta(value: Option<i64>) -> String {
value
.map(|v| match v {
-1 => "".into(),
-2 => "?".into(),
_ => format!("{} s", v),
})
.unwrap_or("".into())
}
trait Formatter {
fn format(&self) -> String;
}
impl Formatter for Option<f32> {
fn format(&self) -> String {
self.map(|v| format!("{:.2}", v)).unwrap_or_default()
}
}
impl<T> Formatter for Option<Vec<T>> {
fn format(&self) -> String {
self.as_ref()
.map(|v| format!("{}", v.len()))
.unwrap_or_default()
}
}
macro_rules! impl_enum_formatter {
($enum_type:ty, { $($variant:pat => $str:expr),* $(,)? }) => {
impl Formatter for Option<$enum_type> {
fn format(&self) -> String {
self.map(|v| match v { $($variant => $str,)* }).unwrap_or("N/A").into()
}
}
};
}
impl_enum_formatter!(Priority, {
Priority::Low => "Low",
Priority::Normal => "Normal",
Priority::High => "High",
});
impl_enum_formatter!(IdleMode, {
IdleMode::Global => "Global",
IdleMode::Single => "Single",
IdleMode::Unlimited => "Unlimited",
});
impl_enum_formatter!(RatioMode, {
RatioMode::Global => "Global",
RatioMode::Single => "Single",
RatioMode::Unlimited => "Unlimited",
});
impl_enum_formatter!(TorrentStatus, {
TorrentStatus::Stopped => "Stopped",
TorrentStatus::Seeding => "Seeding",
TorrentStatus::Verifying => "Verifying",
TorrentStatus::Downloading => "Downloading",
TorrentStatus::QueuedToSeed => "QueuedToSeed",
TorrentStatus::QueuedToVerify => "QueuedToVerify",
TorrentStatus::QueuedToDownload => "QueuedToDownload",
});
impl_enum_formatter!(ErrorType, {
ErrorType::Ok => "OK",
ErrorType::TrackerWarning => "TrackerWarning",
ErrorType::TrackerError => "TrackerError",
ErrorType::LocalError => "LocalError",
});

View File

@ -1,14 +1,14 @@
use super::unit::{Unit, UnitDisplay}; use super::unit::{Unit, UnitDisplay};
use crate::impl_unit_wrapper; use crate::impl_unit_newtype;
use std::fmt::Display; use std::fmt::Display;
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Default)] #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Default)]
pub struct NetSpeed(Unit); pub struct NetSpeed(Unit);
impl_unit_wrapper!(NetSpeed); impl_unit_newtype!(NetSpeed);
impl NetSpeed { impl NetSpeed {
pub fn new(bytes_per_second: u64) -> Self { pub fn new(bytes_per_second: u64) -> Self {
Self(Unit::new(bytes_per_second)) Self(Unit::from_raw(bytes_per_second))
} }
} }

View File

@ -5,7 +5,10 @@ mod sealed {
macro_rules! impl_sealed { macro_rules! impl_sealed {
($($t:ty),*) => { ($($t:ty),*) => {
$(impl Sealed for $t {})* $(
impl Sealed for $t {}
impl Sealed for Option<$t> {}
)*
}; };
} }
@ -23,49 +26,29 @@ macro_rules! impl_into_u64 {
self.try_into().unwrap_or(0) self.try_into().unwrap_or(0)
} }
} }
impl IntoU64 for Option<$t> {
fn into_u64(self) -> u64 {
self.unwrap_or(0).try_into().unwrap_or(0)
}
}
)*}; )*};
} }
impl_into_u64!(u8, u16, u32, u64, usize, i8, i16, i32, i64, isize); impl_into_u64!(u8, u16, u32, u64, usize, i8, i16, i32, i64, isize);
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord)] #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Default)]
pub struct UnitWrapper<T = u64>(T); pub struct Unit(u64);
impl<T> Default for UnitWrapper<T> impl<U> From<U> for Unit
where
T: From<u64>,
{
fn default() -> Self {
Self(T::from(0))
}
}
impl<T, U> From<U> for UnitWrapper<T>
where where
U: IntoU64, U: IntoU64,
T: From<u64>,
{ {
fn from(value: U) -> Self { fn from(value: U) -> Self {
Self(T::from(value.into_u64())) Self(value.into_u64())
} }
} }
impl<T> UnitWrapper<T> {
pub fn new(inner: T) -> Self {
Self(inner)
}
pub fn inner(&self) -> &T {
&self.0
}
pub fn into_inner(self) -> T {
self.0
}
}
pub type Unit = UnitWrapper<u64>;
impl Unit { impl Unit {
pub const fn from_raw(value: u64) -> Self { pub const fn from_raw(value: u64) -> Self {
Self(value) Self(value)
@ -110,7 +93,7 @@ impl<'a> Display for UnitDisplay<'a> {
} }
#[macro_export] #[macro_export]
macro_rules! impl_unit_wrapper { macro_rules! impl_unit_newtype {
($wrapper:ident) => { ($wrapper:ident) => {
impl From<Unit> for $wrapper { impl From<Unit> for $wrapper {
fn from(unit: $crate::app::utils::unit::Unit) -> Self { fn from(unit: $crate::app::utils::unit::Unit) -> Self {