mirror of
https://github.com/Llewellynvdm/exa.git
synced 2024-11-22 12:05:11 +00:00
b5a1b24bda
I checked strace and it was reading /etc/passwd every time it needed to look up a user or group. Now it only does it once per.
79 lines
2.0 KiB
Rust
79 lines
2.0 KiB
Rust
use std::str::raw::from_c_str;
|
|
use std::ptr::read;
|
|
use std::collections::hashmap::HashMap;
|
|
|
|
mod c {
|
|
#![allow(non_camel_case_types)]
|
|
extern crate libc;
|
|
use self::libc::{
|
|
c_char,
|
|
c_int,
|
|
uid_t,
|
|
time_t
|
|
};
|
|
|
|
pub struct c_passwd {
|
|
pub pw_name: *c_char, // login name
|
|
pub pw_passwd: *c_char,
|
|
pub pw_uid: c_int, // user ID
|
|
pub pw_gid: c_int, // group ID
|
|
pub pw_change: time_t,
|
|
pub pw_class: *c_char,
|
|
pub pw_gecos: *c_char, // full name
|
|
pub pw_dir: *c_char, // login dir
|
|
pub pw_shell: *c_char, // login shell
|
|
pub pw_expire: time_t // password expiry time
|
|
}
|
|
|
|
pub struct c_group {
|
|
pub gr_name: *c_char // group name
|
|
}
|
|
|
|
extern {
|
|
pub fn getpwuid(uid: c_int) -> *c_passwd;
|
|
pub fn getgrgid(gid: uid_t) -> *c_group;
|
|
pub fn getuid() -> libc::c_int;
|
|
}
|
|
}
|
|
pub struct Unix {
|
|
user_names: HashMap<i32, Option<String>>,
|
|
group_names: HashMap<u32, Option<String>>,
|
|
}
|
|
|
|
impl Unix {
|
|
pub fn empty_cache() -> Unix {
|
|
Unix {
|
|
user_names: HashMap::new(),
|
|
group_names: HashMap::new(),
|
|
}
|
|
}
|
|
|
|
pub fn get_user_name<'a> (&'a mut self, uid: i32) -> Option<String> {
|
|
self.user_names.find_or_insert_with(uid, |&u| {
|
|
let pw = unsafe { c::getpwuid(u) };
|
|
if pw.is_not_null() {
|
|
return unsafe { Some(from_c_str(read(pw).pw_name)) };
|
|
}
|
|
else {
|
|
return None;
|
|
}
|
|
}).clone()
|
|
}
|
|
|
|
pub fn get_group_name<'a>(&'a mut self, gid: u32) -> Option<String> {
|
|
self.group_names.find_or_insert_with(gid, |&gid| {
|
|
let gr = unsafe { c::getgrgid(gid) };
|
|
if gr.is_not_null() {
|
|
return unsafe { Some(from_c_str(read(gr).gr_name)) };
|
|
}
|
|
else {
|
|
return None;
|
|
}
|
|
}).clone()
|
|
}
|
|
}
|
|
|
|
pub fn get_current_user_id() -> u64 {
|
|
unsafe { c::getuid() as u64 }
|
|
}
|