Compare commits

...

5 commits

Author SHA1 Message Date
fc62d72cab
add demo to README
Some checks are pending
Rust / build (push) Waiting to run
2024-08-04 16:35:16 +03:00
c2deeb38a4
increment version 2024-08-04 13:08:48 +03:00
a44db5e5f3
shell name & colors in output 2024-08-04 13:07:27 +03:00
81cdc0a281
speeding it up 2024-08-04 12:44:19 +03:00
mewoocat
ef2f6dc56e
Better logo (#1) 2024-08-04 09:42:53 +00:00
9 changed files with 92 additions and 91 deletions

BIN
.github/assets/demo.png vendored Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 32 KiB

2
Cargo.lock generated
View file

@ -105,7 +105,7 @@ checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3"
[[package]] [[package]]
name = "microfetch" name = "microfetch"
version = "0.2.0" version = "0.3.0"
dependencies = [ dependencies = [
"color-eyre", "color-eyre",
"nix", "nix",

View file

@ -1,10 +1,10 @@
[package] [package]
name = "microfetch" name = "microfetch"
version = "0.2.0" version = "0.3.0"
edition = "2021" edition = "2021"
[dependencies] [dependencies]
nix = {version = "0.29", features = ["fs"]} nix = {version = "0.29", features = ["fs", "hostname"]}
color-eyre = { version = "0.6", default-features = false } color-eyre = { version = "0.6", default-features = false }
[profile.release] [profile.release]
@ -13,6 +13,3 @@ opt-level = "z"
lto = true lto = true
codegen-units = 1 codegen-units = 1
panic = "abort" panic = "abort"

View file

@ -1,16 +1,35 @@
# Microfetch # Microfetch
A stupidly simple fetch program written in Rust. A stupidly simple fetch tool, written in Rust. Runs in a fraction of a second,
displays most nonsense people on r/unixporn care about. Aims to replace
fastfetch on my system, but probably not on yours. Though you are more than
welcome to use it on your system: it's fast.
![Demo](.github/assets/demo.png)
## Features ## Features
Microfetch can reliably detect and display the following: - Fast
- Really fast
- Hostname/Username - Minimal dependencies
- Kernel - Actually very fast
- Cool NixOS logo (other, inferior, distros are not supported)
- Reliable detection of following info:
- Hostname/Username
- Kernel
- Name - Name
- Version - Version
- Architecture - Architecture
- WM/Compositor and display backend - Current shell (from $SHELL)
- Memory Usage/Total Memory - WM/Compositor and display backend
- Storage Usage/Total Storage (for `/` only) - Memory Usage/Total Memory
- Storage Usage/Total Storage (for `/` only)
- Shell Colors
## Customizing
You can't.
## License
Microfetch is licensed under [GPL3](LICENSE). See the license file for details.

View file

@ -4,3 +4,10 @@ pub const CYAN: &str = "\x1b[36m";
pub const GREEN: &str = "\x1b[32m"; pub const GREEN: &str = "\x1b[32m";
pub const YELLOW: &str = "\x1b[33m"; pub const YELLOW: &str = "\x1b[33m";
pub const RED: &str = "\x1b[31m"; pub const RED: &str = "\x1b[31m";
pub const MAGENTA: &str = "\x1b[35m";
pub fn print_dots() -> Result<String, std::io::Error> {
let colors = format!("{BLUE}{CYAN}{GREEN}{YELLOW}{RED}{MAGENTA}{RESET}");
Ok(colors)
}

View file

@ -4,13 +4,13 @@ mod release;
mod system; mod system;
mod uptime; mod uptime;
use color_eyre::{Report, Result}; use crate::colors::{print_dots, BLUE, CYAN, RESET};
use crate::colors::{BLUE, CYAN, RESET};
use crate::desktop::get_desktop_info; use crate::desktop::get_desktop_info;
use crate::release::{get_os_pretty_name, get_system_info}; use crate::release::{get_os_pretty_name, get_system_info};
use crate::system::{get_memory_usage, get_root_disk_usage, get_username_and_hostname}; use crate::system::{get_memory_usage, get_root_disk_usage, get_shell, get_username_and_hostname};
use crate::uptime::get_current; use crate::uptime::get_current;
use color_eyre::Report;
use nix::sys::sysinfo::sysinfo;
fn main() -> Result<(), Report> { fn main() -> Result<(), Report> {
color_eyre::install()?; color_eyre::install()?;
@ -18,19 +18,24 @@ fn main() -> Result<(), Report> {
let user_info = get_username_and_hostname()?; let user_info = get_username_and_hostname()?;
let os_name = get_os_pretty_name()?; let os_name = get_os_pretty_name()?;
let kernel_version = get_system_info()?; let kernel_version = get_system_info()?;
let shell = get_shell()?;
let uptime = get_current()?; let uptime = get_current()?;
let window_manager = get_desktop_info()?; let window_manager = get_desktop_info()?;
let memory_usage = get_memory_usage()?; let sys_info = sysinfo()?;
let memory_usage = get_memory_usage(sys_info);
let storage = get_root_disk_usage()?; let storage = get_root_disk_usage()?;
let colors = print_dots()?;
print_system_info( print_system_info(
&user_info, &user_info,
&os_name, &os_name,
&kernel_version, &kernel_version,
&shell,
&uptime, &uptime,
&window_manager, &window_manager,
&memory_usage, &memory_usage,
&storage, &storage,
&colors,
); );
Ok(()) Ok(())
@ -40,20 +45,24 @@ fn print_system_info(
user_info: &str, user_info: &str,
os_name: &str, os_name: &str,
kernel_version: &str, kernel_version: &str,
shell: &str,
uptime: &str, uptime: &str,
window_manager: &str, window_manager: &str,
memory_usage: &str, memory_usage: &str,
storage: &str, storage: &str,
colors: &str,
) { ) {
println!( println!(
" "
{CYAN} {BLUE} {user_info} ~{RESET} {CYAN} {BLUE} {user_info} ~{RESET}
{CYAN} 🬸{BLUE}🬸 {CYAN} {CYAN} {BLUE}System{RESET} {os_name} {CYAN} {BLUE} {CYAN} {CYAN} {BLUE}System{RESET} {os_name}
{BLUE} {CYAN}🬕 {CYAN} {BLUE}Kernel{RESET} {kernel_version} {CYAN} {BLUE} {CYAN} {CYAN} {BLUE}Kernel{RESET} {kernel_version}
{BLUE}🬋🬋🬫 {CYAN}🬛🬋🬋 {CYAN} {BLUE}Uptime{RESET} {uptime} {BLUE} {BLUE}{CYAN} {CYAN} {BLUE}Shell{RESET} {shell}
{BLUE} 🬷🮃{CYAN} {CYAN} {BLUE}WM{RESET} {window_manager} {BLUE} {CYAN} {CYAN} {BLUE}Uptime{RESET} {uptime}
{BLUE} 🮃 {CYAN}🬴{BLUE}🬴 {CYAN}󰍛 {BLUE}Memory{RESET} {memory_usage} {BLUE} {CYAN} {CYAN} {CYAN} {BLUE}WM{RESET} {window_manager}
{CYAN} {BLUE} {CYAN}󱥎 {BLUE}Storage (/){RESET} {storage} {BLUE} {CYAN}{BLUE} {CYAN}󰍛 {BLUE}Memory{RESET} {memory_usage}
{BLUE} {CYAN}{BLUE} {CYAN}󱥎 {BLUE}Storage (/){RESET} {storage}
{CYAN} {BLUE} {CYAN} {BLUE}Colors{RESET} {colors}
" "
); );
} }

View file

@ -1,38 +1,27 @@
use color_eyre::Result; use color_eyre::Result;
use std::fs::{self, read_to_string}; use std::fs::{self, read_to_string};
use std::io::{self, Read}; use std::io;
// Try to detect OS type as accurately as possible and without depending on uname. // Try to detect OS type as accurately as possible and without depending on uname.
// /etc/os-release should generally imply Linux, and /etc/bsd-release would imply BSD system. // /etc/os-release should generally imply Linux, and /etc/bsd-release would imply BSD system.
fn detect_os() -> Result<String, io::Error> { fn detect_os() -> Result<&'static str, io::Error> {
if fs::metadata("/etc/os-release").is_ok() || fs::metadata("/usr/lib/os-release").is_ok() { if fs::metadata("/etc/os-release").is_ok() || fs::metadata("/usr/lib/os-release").is_ok() {
Ok("Linux".to_string()) Ok("Linux")
} else if fs::metadata("/etc/rc.conf").is_ok() || fs::metadata("/etc/bsd-release").is_ok() { } else if fs::metadata("/etc/rc.conf").is_ok() || fs::metadata("/etc/bsd-release").is_ok() {
Ok("BSD".to_string()) Ok("BSD")
} else { } else {
Ok("Unknown".to_string()) Ok("Unknown")
} }
} }
fn get_architecture() -> Result<String, io::Error> {
// Read architecture from /proc/sys/kernel/arch
let mut arch = String::new();
fs::File::open("/proc/sys/kernel/arch")?.read_to_string(&mut arch)?;
let arch = arch.trim().to_string();
Ok(arch)
}
pub fn get_system_info() -> Result<String, io::Error> { pub fn get_system_info() -> Result<String, io::Error> {
let system = detect_os()?; let system = detect_os()?;
let mut kernel_release = String::new(); let kernel_release = read_to_string("/proc/sys/kernel/osrelease")?;
fs::File::open("/proc/sys/kernel/osrelease")?.read_to_string(&mut kernel_release)?; let kernel_release = kernel_release.trim();
let kernel_release = kernel_release.trim().to_string();
let mut cpuinfo = String::new(); let architecture = read_to_string("/proc/sys/kernel/arch")?;
fs::File::open("/proc/cpuinfo")?.read_to_string(&mut cpuinfo)?; let architecture = architecture.trim();
let architecture = get_architecture()?;
let result = format!("{system} {kernel_release} ({architecture})"); let result = format!("{system} {kernel_release} ({architecture})");
Ok(result) Ok(result)

View file

@ -1,21 +1,29 @@
use nix::sys::statvfs::statvfs; use nix::sys::statvfs::statvfs;
use nix::sys::sysinfo::SysInfo;
use std::env; use std::env;
use std::fs::File; use std::io::{self};
use std::io::{self, BufRead};
use std::path::Path;
use std::process::Command;
use crate::colors::{CYAN, GREEN, RED, RESET, YELLOW}; use crate::colors::{CYAN, GREEN, RED, RESET, YELLOW};
pub fn get_username_and_hostname() -> Result<String, io::Error> { pub fn get_username_and_hostname() -> Result<String, io::Error> {
let username = env::var("USER").unwrap_or_else(|_| "unknown_user".to_string()); let username = env::var("USER").unwrap_or_else(|_| "unknown_user".to_string());
let output = Command::new("hostname").output()?; let hostname = nix::unistd::gethostname()?;
let hostname = String::from_utf8_lossy(&output.stdout).trim().to_string(); let hostname = hostname.to_string_lossy();
Ok(format!("{YELLOW}{username}{RED}@{GREEN}{hostname}")) Ok(format!("{YELLOW}{username}{RED}@{GREEN}{hostname}"))
} }
pub fn get_shell() -> Result<String, io::Error> {
// In some setups, $SHELL is set to the store path
// of the actual shell program. While we can consider
// trimming by name, I will leave it to the user to handle
// what their SHELL variable is really set to.
let shell = env::var("SHELL").unwrap_or_else(|_| "unknown_shell".to_string());
Ok(shell)
}
pub fn get_root_disk_usage() -> Result<String, io::Error> { pub fn get_root_disk_usage() -> Result<String, io::Error> {
let vfs = statvfs("/")?; let vfs = statvfs("/")?;
let block_size = vfs.block_size() as u64; let block_size = vfs.block_size() as u64;
@ -34,49 +42,21 @@ pub fn get_root_disk_usage() -> Result<String, io::Error> {
)) ))
} }
pub fn get_memory_usage() -> Result<String, io::Error> { pub fn get_memory_usage(info: SysInfo) -> String {
fn parse_memory_info() -> Result<(f64, f64), io::Error> { #[inline(always)]
let path = Path::new("/proc/meminfo"); fn parse_memory_info(info: SysInfo) -> (f64, f64) {
let file = File::open(path)?; let total_memory_kb = (info.ram_total() / 1024) as f64;
let reader = io::BufReader::new(file); let available_memory_kb = (info.ram_unused() / 1024) as f64;
let mut total_memory_kb = 0.0;
let mut available_memory_kb = 0.0;
for line in reader.lines() {
let line = line?;
if line.starts_with("MemTotal:") {
total_memory_kb = line
.split_whitespace()
.nth(1)
.ok_or_else(|| {
io::Error::new(io::ErrorKind::InvalidData, "Failed to parse MemTotal")
})?
.parse::<f64>()
.unwrap_or(0.0);
} else if line.starts_with("MemAvailable:") {
available_memory_kb = line
.split_whitespace()
.nth(1)
.ok_or_else(|| {
io::Error::new(io::ErrorKind::InvalidData, "Failed to parse MemAvailable")
})?
.parse::<f64>()
.unwrap_or(0.0);
}
}
let total_memory_gb = total_memory_kb / (1024.0 * 1024.0); let total_memory_gb = total_memory_kb / (1024.0 * 1024.0);
let available_memory_gb = available_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; let used_memory_gb = total_memory_gb - available_memory_gb;
Ok((used_memory_gb, total_memory_gb)) (used_memory_gb, total_memory_gb)
} }
let (used_memory, total_memory) = parse_memory_info()?; let (used_memory, total_memory) = parse_memory_info(info);
let percentage_used = (used_memory / total_memory * 100.0).round() as u64; let percentage_used = (used_memory / total_memory * 100.0).round() as u64;
Ok(format!( format!("{used_memory:.2} GiB / {total_memory:.2} GiB ({CYAN}{percentage_used}%{RESET})")
"{used_memory:.2} GiB / {total_memory:.2} GiB ({CYAN}{percentage_used}%{RESET})"
))
} }

View file

@ -25,7 +25,7 @@ pub fn get_current() -> Result<String, io::Error> {
let hours = (total_minutes % (60 * 24)) / 60; let hours = (total_minutes % (60 * 24)) / 60;
let minutes = total_minutes % 60; let minutes = total_minutes % 60;
let mut parts = Vec::new(); let mut parts = Vec::with_capacity(3);
if days > 0 { if days > 0 {
parts.push(format!("{days} days")); parts.push(format!("{days} days"));
} }