mirror of
https://github.com/uutils/coreutils
synced 2024-12-15 15:52:42 +00:00
146 lines
3.4 KiB
Rust
146 lines
3.4 KiB
Rust
#[crate_id(name="users", vers="1.0.0", author="KokaKiwi")];
|
|
|
|
/*
|
|
* This file is part of the uutils coreutils package.
|
|
*
|
|
* (c) KokaKiwi <kokakiwi@kokakiwi.net>
|
|
*
|
|
* For the full copyright and license information, please view the LICENSE
|
|
* file that was distributed with this source code.
|
|
*/
|
|
|
|
/* last synced with: whoami (GNU coreutils) 8.22 */
|
|
|
|
// Allow dead code here in order to keep all fields, constants here, for consistency.
|
|
#[allow(dead_code)];
|
|
|
|
extern mod extra;
|
|
|
|
use std::io::print;
|
|
use std::cast;
|
|
use std::libc;
|
|
use std::os;
|
|
use std::ptr;
|
|
use std::str;
|
|
use extra::getopts::groups;
|
|
|
|
static DEFAULT_FILE: &'static str = "/var/run/utmp";
|
|
|
|
static UT_LINESIZE: uint = 32;
|
|
static UT_NAMESIZE: uint = 32;
|
|
static UT_HOSTSIZE: uint = 256;
|
|
|
|
static EMPTY: libc::c_short = 0;
|
|
static RUN_LVL: libc::c_short = 1;
|
|
static BOOT_TIME: libc::c_short = 2;
|
|
static NEW_TIME: libc::c_short = 3;
|
|
static OLD_TIME: libc::c_short = 4;
|
|
static INIT_PROCESS: libc::c_short = 5;
|
|
static LOGIN_PROCESS: libc::c_short = 6;
|
|
static USER_PROCESS: libc::c_short = 7;
|
|
static DEAD_PROCESS: libc::c_short = 8;
|
|
static ACCOUNTING: libc::c_short = 9;
|
|
|
|
struct c_exit_status {
|
|
e_termination: libc::c_short,
|
|
e_exit: libc::c_short,
|
|
}
|
|
|
|
struct c_utmp {
|
|
ut_type: libc::c_short,
|
|
ut_pid: libc::pid_t,
|
|
ut_line: [libc::c_char, ..UT_LINESIZE],
|
|
ut_id: [libc::c_char, ..4],
|
|
|
|
ut_user: [libc::c_char, ..UT_NAMESIZE],
|
|
ut_host: [libc::c_char, ..UT_HOSTSIZE],
|
|
ut_exit: c_exit_status,
|
|
ut_session: libc::c_long,
|
|
ut_tv: libc::timeval,
|
|
|
|
ut_addr_v6: [libc::int32_t, ..4],
|
|
__unused: [libc::c_char, ..20],
|
|
}
|
|
|
|
extern {
|
|
fn getutent() -> *c_utmp;
|
|
fn getutid(ut: *c_utmp) -> *c_utmp;
|
|
fn getutline(ut: *c_utmp) -> *c_utmp;
|
|
|
|
fn pututline(ut: *c_utmp) -> *c_utmp;
|
|
|
|
fn setutent();
|
|
fn endutent();
|
|
|
|
fn utmpname(file: *libc::c_char) -> libc::c_int;
|
|
}
|
|
|
|
fn main() {
|
|
let args = os::args();
|
|
let program = args[0].as_slice();
|
|
let opts = ~[
|
|
groups::optflag("h", "help", "display this help and exit"),
|
|
groups::optflag("V", "version", "output version information and exit"),
|
|
];
|
|
|
|
let matches = match groups::getopts(args.tail(), opts) {
|
|
Ok(m) => m,
|
|
Err(f) => fail!(f.to_err_msg()),
|
|
};
|
|
|
|
if matches.opt_present("help") {
|
|
println!("users 1.0.0");
|
|
println!("");
|
|
println!("Usage:");
|
|
println!(" {:s} [OPTION]... [FILE]", program);
|
|
println!("");
|
|
print(groups::usage("Output who is currently logged in according to FILE.", opts));
|
|
return;
|
|
}
|
|
|
|
if matches.opt_present("version") {
|
|
println!("users 1.0.0");
|
|
return;
|
|
}
|
|
|
|
let mut filename = DEFAULT_FILE;
|
|
if matches.free.len() > 0 {
|
|
filename = matches.free[0].as_slice();
|
|
}
|
|
|
|
exec(filename);
|
|
}
|
|
|
|
fn exec(filename: &str) {
|
|
filename.with_c_str(|filename| {
|
|
unsafe {
|
|
utmpname(filename);
|
|
}
|
|
});
|
|
|
|
let mut users: ~[~str] = ~[];
|
|
|
|
unsafe {
|
|
setutent();
|
|
|
|
loop {
|
|
let line = getutent();
|
|
|
|
if line == ptr::null() {
|
|
break;
|
|
}
|
|
|
|
if (*line).ut_type == USER_PROCESS {
|
|
let user = str::raw::from_c_str(cast::transmute(&(*line).ut_user));
|
|
users.push(user);
|
|
}
|
|
}
|
|
|
|
endutent();
|
|
}
|
|
|
|
if users.len() > 0 {
|
|
users.sort();
|
|
println!("{}", users.connect(" "));
|
|
}
|
|
}
|