2014-07-06 08:13:36 +00:00
|
|
|
#![crate_name = "uptime"]
|
2014-03-18 12:17:32 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This file is part of the uutils coreutils package.
|
|
|
|
*
|
|
|
|
* (c) Jordi Boggiano <j.boggiano@seld.be>
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* last synced with: cat (GNU coreutils) 8.13 */
|
|
|
|
|
2014-03-31 16:40:21 +00:00
|
|
|
#![allow(non_camel_case_types)]
|
|
|
|
#![feature(macro_rules, globs)]
|
2014-03-18 12:17:32 +00:00
|
|
|
|
|
|
|
extern crate getopts;
|
2014-04-26 05:03:08 +00:00
|
|
|
extern crate libc;
|
2014-03-18 12:17:32 +00:00
|
|
|
|
2014-05-16 09:03:19 +00:00
|
|
|
use std::mem::transmute;
|
2014-04-26 05:03:08 +00:00
|
|
|
use std::io::{print, File};
|
2014-06-30 11:13:38 +00:00
|
|
|
use std::ptr::{mut_null, null};
|
2014-03-18 12:17:32 +00:00
|
|
|
use std::from_str::from_str;
|
2014-04-26 05:03:08 +00:00
|
|
|
use libc::{time_t, c_double, c_int, c_char};
|
2014-03-18 12:17:32 +00:00
|
|
|
use c_types::c_tm;
|
|
|
|
use utmpx::*;
|
|
|
|
|
|
|
|
#[path = "../common/util.rs"] mod util;
|
|
|
|
|
|
|
|
#[path = "../common/c_types.rs"] mod c_types;
|
|
|
|
|
|
|
|
#[path = "../common/utmpx.rs"] mod utmpx;
|
|
|
|
|
|
|
|
static NAME: &'static str = "uptime";
|
|
|
|
|
|
|
|
extern {
|
2014-06-30 11:13:38 +00:00
|
|
|
fn time(timep: *mut time_t) -> time_t;
|
|
|
|
fn localtime(timep: *const time_t) -> *const c_tm;
|
2014-03-18 12:17:32 +00:00
|
|
|
|
2014-06-30 11:13:38 +00:00
|
|
|
fn getloadavg(loadavg: *mut c_double, nelem: c_int) -> c_int;
|
2014-03-18 12:17:32 +00:00
|
|
|
|
2014-06-30 11:13:38 +00:00
|
|
|
fn getutxent() -> *const c_utmp;
|
2014-03-18 12:17:32 +00:00
|
|
|
fn setutxent();
|
|
|
|
fn endutxent();
|
|
|
|
|
2014-06-30 11:13:38 +00:00
|
|
|
fn utmpxname(file: *const c_char) -> c_int;
|
2014-03-18 12:17:32 +00:00
|
|
|
}
|
|
|
|
|
2014-06-08 07:56:37 +00:00
|
|
|
pub fn uumain(args: Vec<String>) -> int {
|
2014-05-16 08:32:58 +00:00
|
|
|
let program = args.get(0).clone();
|
2014-05-30 08:35:54 +00:00
|
|
|
let opts = [
|
2014-03-18 12:17:32 +00:00
|
|
|
getopts::optflag("v", "version", "output version information and exit"),
|
|
|
|
getopts::optflag("h", "help", "display this help and exit"),
|
|
|
|
];
|
|
|
|
let matches = match getopts::getopts(args.tail(), opts) {
|
|
|
|
Ok(m) => m,
|
2014-06-15 10:50:40 +00:00
|
|
|
Err(f) => crash!(1, "Invalid options\n{}", f)
|
2014-03-18 12:17:32 +00:00
|
|
|
};
|
|
|
|
if matches.opt_present("version") {
|
|
|
|
println!("uptime 1.0.0");
|
2014-06-08 07:56:37 +00:00
|
|
|
return 0;
|
2014-03-18 12:17:32 +00:00
|
|
|
}
|
|
|
|
if matches.opt_present("help") || matches.free.len() > 0 {
|
|
|
|
println!("Usage:");
|
|
|
|
println!(" {0:s} [OPTION]", program);
|
|
|
|
println!("");
|
|
|
|
print(getopts::usage("Print the current time, the length of time the system has been up,\n\
|
|
|
|
the number of users on the system, and the average number of jobs\n\
|
2014-05-17 10:32:14 +00:00
|
|
|
in the run queue over the last 1, 5 and 15 minutes.", opts).as_slice());
|
2014-06-08 07:56:37 +00:00
|
|
|
return 0;
|
2014-03-18 12:17:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
print_time();
|
2014-03-25 17:28:31 +00:00
|
|
|
let (boot_time, user_count) = process_utmpx();
|
|
|
|
let upsecs = get_uptime(boot_time) / 100;
|
|
|
|
print_uptime(upsecs);
|
|
|
|
print_nusers(user_count);
|
2014-03-18 12:17:32 +00:00
|
|
|
print_loadavg();
|
2014-06-08 07:56:37 +00:00
|
|
|
|
2014-06-12 04:41:53 +00:00
|
|
|
0
|
2014-03-18 12:17:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn print_loadavg() {
|
2014-06-30 11:13:38 +00:00
|
|
|
let mut avg: [c_double, ..3] = [0.0, ..3];
|
|
|
|
let loads: i32 = unsafe { transmute(getloadavg(avg.as_mut_ptr(), 3)) };
|
2014-03-18 12:17:32 +00:00
|
|
|
|
|
|
|
if loads == -1 {
|
|
|
|
print!("\n");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
print!("load average: ")
|
|
|
|
for n in range(0, loads) {
|
2014-04-26 05:03:08 +00:00
|
|
|
print!("{:.2f}{}", avg[n as uint], if n == loads - 1 { "\n" }
|
2014-03-18 12:17:32 +00:00
|
|
|
else { ", " } );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-25 17:28:31 +00:00
|
|
|
fn process_utmpx() -> (Option<time_t>, uint) {
|
2014-03-18 12:17:32 +00:00
|
|
|
DEFAULT_FILE.with_c_str(|filename| {
|
|
|
|
unsafe {
|
|
|
|
utmpxname(filename);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
let mut nusers = 0;
|
2014-03-25 17:28:31 +00:00
|
|
|
let mut boot_time = None;
|
2014-03-18 12:17:32 +00:00
|
|
|
|
|
|
|
unsafe {
|
|
|
|
setutxent();
|
|
|
|
|
|
|
|
loop {
|
|
|
|
let line = getutxent();
|
|
|
|
|
|
|
|
if line == null() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-03-25 17:28:31 +00:00
|
|
|
match (*line).ut_type {
|
|
|
|
USER_PROCESS => nusers += 1,
|
|
|
|
BOOT_TIME => {
|
|
|
|
let t = (*line).ut_tv;
|
|
|
|
if t.tv_sec > 0 {
|
|
|
|
boot_time = Some(t.tv_sec);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => continue
|
2014-03-18 12:17:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
endutxent();
|
|
|
|
}
|
|
|
|
|
2014-03-25 17:28:31 +00:00
|
|
|
(boot_time, nusers)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_nusers(nusers: uint) {
|
2014-03-18 12:17:32 +00:00
|
|
|
if nusers == 1 {
|
|
|
|
print!("1 user, ");
|
|
|
|
} else if nusers > 1 {
|
|
|
|
print!("{} users, ", nusers);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_time() {
|
2014-06-30 11:13:38 +00:00
|
|
|
let local_time = unsafe { *localtime(&time(mut_null())) };
|
2014-03-18 12:17:32 +00:00
|
|
|
|
|
|
|
if local_time.tm_hour >= 0 && local_time.tm_min >= 0 &&
|
|
|
|
local_time.tm_sec >= 0 {
|
|
|
|
print!(" {:02d}:{:02d}:{:02d} ", local_time.tm_hour,
|
|
|
|
local_time.tm_min, local_time.tm_sec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-25 17:28:31 +00:00
|
|
|
fn get_uptime(boot_time: Option<time_t>) -> i64 {
|
2014-03-25 15:59:29 +00:00
|
|
|
let proc_uptime = File::open(&Path::new("/proc/uptime"))
|
2014-07-09 08:29:50 +00:00
|
|
|
.read_to_string();
|
2014-03-25 15:59:29 +00:00
|
|
|
|
|
|
|
let uptime_text = match proc_uptime {
|
|
|
|
Ok(s) => s,
|
2014-03-25 17:28:31 +00:00
|
|
|
_ => return match boot_time {
|
|
|
|
Some(t) => {
|
2014-06-30 11:13:38 +00:00
|
|
|
let now = unsafe { time(mut_null()) };
|
2014-06-18 12:42:34 +00:00
|
|
|
((now - t) * 100) as i64 // Return in ms
|
2014-03-25 17:28:31 +00:00
|
|
|
},
|
|
|
|
_ => -1
|
|
|
|
}
|
2014-03-25 15:59:29 +00:00
|
|
|
};
|
2014-03-18 12:17:32 +00:00
|
|
|
|
2014-05-23 12:28:40 +00:00
|
|
|
match uptime_text.as_slice().words().next() {
|
|
|
|
Some(s) => match from_str(s.replace(".","").as_slice()) {
|
2014-03-18 12:17:32 +00:00
|
|
|
Some(n) => n,
|
|
|
|
None => -1
|
|
|
|
},
|
|
|
|
None => -1
|
2014-03-25 15:59:29 +00:00
|
|
|
}
|
2014-03-18 12:17:32 +00:00
|
|
|
}
|
|
|
|
|
2014-03-25 17:28:31 +00:00
|
|
|
fn print_uptime(upsecs: i64) {
|
|
|
|
let updays = upsecs / 86400;
|
|
|
|
let uphours = (upsecs - (updays * 86400)) / 3600;
|
|
|
|
let upmins = (upsecs - (updays * 86400) - (uphours * 3600)) / 60;
|
2014-03-25 15:59:29 +00:00
|
|
|
if updays == 1 {
|
2014-03-18 12:17:32 +00:00
|
|
|
print!("up {:1d} day, {:2d}:{:02d}, ", updays, uphours, upmins);
|
|
|
|
}
|
|
|
|
else if updays > 1 {
|
|
|
|
print!("up {:1d} days, {:2d}:{:02d}, ", updays, uphours, upmins);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
print!("up {:2d}:{:02d}, ", uphours, upmins);
|
|
|
|
}
|
|
|
|
}
|