mirror of
https://github.com/NotAShelf/microfetch.git
synced 2025-12-18 09:51:26 +00:00
* perf: break early after parsing required meminfo Also a match statement for compiler magic. * perf: pre-allocate strings when reading files * refactor: remove duplicate .to_string() * perf: try to print everything in one syscall println! sends a syscall for each line. * perf: get rid of duplicate uname syscall * perf: simplify first letter capitalization * refactor: directly use key in match statement
84 lines
2.7 KiB
Rust
84 lines
2.7 KiB
Rust
use color_eyre::Result;
|
|
use nix::sys::{statvfs::statvfs, utsname::UtsName};
|
|
|
|
use std::{
|
|
env,
|
|
fs::File,
|
|
io::{self, Read},
|
|
};
|
|
|
|
use crate::colors::{CYAN, GREEN, RED, RESET, YELLOW};
|
|
|
|
pub fn get_username_and_hostname(utsname: &UtsName) -> String {
|
|
let username = env::var("USER").unwrap_or("unknown_user".to_string());
|
|
let hostname = utsname
|
|
.nodename()
|
|
.to_str()
|
|
.unwrap_or("unknown_host")
|
|
.to_string();
|
|
|
|
format!("{YELLOW}{username}{RED}@{GREEN}{hostname}")
|
|
}
|
|
|
|
pub fn get_shell() -> String {
|
|
let shell_path = env::var("SHELL").unwrap_or("unknown_shell".to_string());
|
|
let shell_name = shell_path.rsplit('/').next().unwrap_or("unknown_shell");
|
|
|
|
shell_name.to_string()
|
|
}
|
|
|
|
pub fn get_root_disk_usage() -> Result<String, io::Error> {
|
|
let vfs = statvfs("/")?;
|
|
let block_size = vfs.block_size() as u64;
|
|
let total_blocks = vfs.blocks();
|
|
let available_blocks = vfs.blocks_available();
|
|
|
|
let total_size = block_size * total_blocks;
|
|
let used_size = total_size - (block_size * available_blocks);
|
|
|
|
let total_size = total_size as f64 / (1024.0 * 1024.0 * 1024.0);
|
|
let used_size = used_size as f64 / (1024.0 * 1024.0 * 1024.0);
|
|
let usage = (used_size as f64 / total_size as f64) * 100.0;
|
|
|
|
Ok(format!(
|
|
"{used_size:.2} GiB / {total_size:.2} GiB ({CYAN}{usage:.0}%{RESET})"
|
|
))
|
|
}
|
|
|
|
pub fn get_memory_usage() -> Result<String, io::Error> {
|
|
#[inline(always)]
|
|
fn parse_memory_info() -> Result<(f64, f64), io::Error> {
|
|
let mut total_memory_kb = 0.0;
|
|
let mut available_memory_kb = 0.0;
|
|
|
|
let mut meminfo = String::with_capacity(2048);
|
|
File::open("/proc/meminfo")?.read_to_string(&mut meminfo)?;
|
|
|
|
for line in meminfo.lines() {
|
|
let mut split = line.split_whitespace();
|
|
match split.next().unwrap_or_default() {
|
|
"MemTotal:" => total_memory_kb = split.next().unwrap_or("0").parse().unwrap_or(0.0),
|
|
"MemAvailable:" => {
|
|
available_memory_kb = split.next().unwrap_or("0").parse().unwrap_or(0.0);
|
|
|
|
// MemTotal comes before MemAvailable, stop parsing
|
|
break;
|
|
}
|
|
_ => (),
|
|
}
|
|
}
|
|
|
|
let total_memory_gb = total_memory_kb / 1024.0 / 1024.0;
|
|
let available_memory_gb = available_memory_kb / 1024.0 / 1024.0;
|
|
let used_memory_gb = total_memory_gb - available_memory_gb;
|
|
|
|
Ok((used_memory_gb, total_memory_gb))
|
|
}
|
|
|
|
let (used_memory, total_memory) = parse_memory_info()?;
|
|
let percentage_used = (used_memory / total_memory * 100.0).round() as u64;
|
|
|
|
Ok(format!(
|
|
"{used_memory:.2} GiB / {total_memory:.2} GiB ({CYAN}{percentage_used}%{RESET})"
|
|
))
|
|
}
|