extern crate chrono;
use std::fs;
use std::fs::{File};
use std::fs::Metadata;
use std::collections::HashMap;
use std::io::Error;
use std::os::unix::fs::PermissionsExt;
use std::{mem, ptr};
use std::fmt::{Display,Formatter};
use chrono::offset::Utc;
use chrono::DateTime;
use std::os::unix::fs::MetadataExt;
use users::{Users,Groups,UsersCache};
static PERMS : [char;5] = ['d', '-', 'r', 'w', 'x'];
struct FileStats {
is_dir : bool,
permissions : String,
user : String,
group : String,
timestamp : String,
size : u64,
}
impl Display for FileStats {
fn fmt(&self, f: &mut Formatter) -> std::fmt::Result {
write!(f, "{}\t{}\t{}\t{}\t{}", &self.permissions, &self.user, &self.group, bytes_to_string(self.size), &self.timestamp)
}
}
impl FileStats {
fn new(is_dir : bool, permissions : String, user : String, group: String, timestamp : String, size : u64) -> FileStats {
FileStats {
is_dir, permissions, user, group, timestamp, size
}
}
}
fn bytes_to_string(size: u64) -> String {
let mut size_str = String::new();
if size >= 1099511627776 {
size_str.push_str(&bytes_to_string_with_factor(size, 1099511627776));
size_str.push_str(" TB");
} else if size >= 1073741824 {
size_str.push_str(&bytes_to_string_with_factor(size, 1073741824));
size_str.push_str(" GB");
} else if size >= 1048576 {
size_str.push_str(&bytes_to_string_with_factor(size, 1048576));
size_str.push_str(" MB");
} else if size >= 1024 {
size_str.push_str(&bytes_to_string_with_factor(size, 1024));
size_str.push_str(" KB");
} else {
size_str.push_str(&format!("{}", size));
}
if size_str.len() <= 7 {
size_str.push('\t');
}
size_str
}
fn bytes_to_string_with_factor(size: u64, factor: u64) -> String {
format!("{:.2}", (size as f64 / factor as f64))
}
fn main() {
let args : Vec<String> = std::env::args().collect();
let default_file = String::from(".");
let path = args.get(1).unwrap_or_else(|| {
&default_file
});
let mut map : HashMap<String, FileStats> = HashMap::new();
let cache = UsersCache::new();
read_file(&mut map, path, &cache, false);
print_results(&map);
}
fn read_dir(map : &mut HashMap<String, FileStats>, path : &str, cache: &UsersCache) {
for p in fs::read_dir(path).expect(&format!("Failed to read path '{}'", path)) {
let sub_path = p.expect("Failed to read file").path().display().to_string();
read_file(map, &sub_path, cache, true);
}
}
fn read_file(map : &mut HashMap<String, FileStats>, path : &str, cache : &UsersCache, no_recursion : bool) {
let result = File::open(path);
match result {
Ok(file) => {
let meta = file.metadata().expect(&format!("Failed to read '{}' metadata", path));
let stats = create_file_stats(&meta, cache);
let stats = stats.expect(&format!("Failed to create file stats for '{}'", path));
let mut relative_path = String::new();
if !path.starts_with(".") && !path.starts_with("/") {
relative_path.push_str("./");
};
let is_dir = meta.is_dir();
relative_path.push_str(path);
if is_dir && !path.ends_with("/") {
relative_path.push('/');
}
map.insert(relative_path, stats);
if is_dir && !no_recursion {
read_dir(map, path, cache);
}
},
Err(_) => println!("Failed to stat '{}'", path),
}
}
fn print_results(map: &HashMap<String, FileStats>) {
let mut keys : Vec<String> = Vec::new();
for key in map.keys() {
keys.push(key.to_string());
}
keys.sort_unstable();
for path in keys.iter() {
let file_stats = map.get(path).unwrap();
if file_stats.is_dir {
println!("{}\t\x1B[36m{}\x1B[0m", file_stats, path);
} else {
println!("{}\t{}", file_stats, path);
}
}
println!("{} files/directories", keys.len());
}
fn create_file_stats(meta : &Metadata, cache: &UsersCache) -> Result<FileStats, Error> {
let is_dir = meta.is_dir();
let size = meta.len();
let created = meta.created().expect("Failed to reads files creation date");
let created: DateTime<Utc> = created.into();
let created = format!("{}", created.format("%d/%m/%Y %T"));
let permissions = meta.permissions();
let mode = format!("{:#o}", permissions.mode());
let mode = &mode[mode.len()-3..];
let permissions = build_permissions(is_dir, &mode);
let user = cache.get_user_by_uid(meta.uid()).unwrap();
let group = cache.get_group_by_gid(meta.gid()).unwrap();
let user = user.name().to_os_string().into_string().unwrap();
let group = group.name().to_os_string().into_string().unwrap();
Ok(FileStats::new(
is_dir,
permissions,
user,
group,
created,
size)
)
}
fn build_permissions(is_dir : bool, mode : &str) -> String {
let groups : Vec<char> = mode.chars().collect();
let mut permissions : [char; 10] = unsafe {
let mut array : [char; 10] = mem::uninitialized();
for (_i,e) in array.iter_mut().enumerate() {
ptr::write(e, '-');
}
array
};
match is_dir {
true => permissions[0] = PERMS[0],
_ => permissions[0] = PERMS[1],
};
let mut i : usize = 1;
let mut j : usize = 0;
while i < 8 {
build_permission_group(&mut permissions, groups.get(j).unwrap(), i);
i = i+3;
j = j+1;
}
let result : String = permissions.iter().collect();
result
}
fn build_permission_group(permissions : &mut [char], mode : &char, index : usize) {
if index > 9 {
return;
}
if *mode == '0' || *mode == '1' || *mode == '2' {
permissions[index] = '-';
} else {
permissions[index] = 'r';
}
if *mode == '0' || *mode == '1' || *mode == '4' || *mode == '5' {
permissions[index+1] = '-';
} else {
permissions[index+1] = 'w';
}
if *mode == '0' || *mode == '2' || *mode == '4' || *mode == '6' {
permissions[index+2] = '-';
} else {
permissions[index+2] = 'x';
}
}
extern crate chrono;

use std::fs;
use std::fs::{File};
use std::fs::Metadata;
use std::collections::HashMap;
use std::io::Error;
use std::os::unix::fs::PermissionsExt;
use std::{mem, ptr};
use std::fmt::{Display,Formatter};
use chrono::offset::Utc;
use chrono::DateTime;
use std::os::unix::fs::MetadataExt;
use users::{Users,Groups,UsersCache};

static PERMS : [char;5] = ['d', '-', 'r', 'w', 'x'];

struct FileStats {
    is_dir : bool,
    permissions : String,
    user : String,
    group : String,
    timestamp : String,
    size : u64,
}

impl Display for FileStats {
    fn fmt(&self, f: &mut Formatter) -> std::fmt::Result {
        write!(f, "{}\t{}\t{}\t{}\t{}", &self.permissions, &self.user, &self.group, bytes_to_string(self.size), &self.timestamp)
    }
}

impl FileStats {
    fn new(is_dir : bool, permissions : String, user : String, group: String, timestamp : String, size : u64) -> FileStats {
        FileStats {
            is_dir, permissions, user, group, timestamp, size
        }
    }
}

fn bytes_to_string(size: u64) -> String {
    let mut size_str = String::new();
    if size >= 1099511627776 {
        size_str.push_str(&bytes_to_string_with_factor(size, 1099511627776));
        size_str.push_str(" TB");
    } else if size >= 1073741824 {
        size_str.push_str(&bytes_to_string_with_factor(size, 1073741824));
        size_str.push_str(" GB");
    } else if size >= 1048576 {
        size_str.push_str(&bytes_to_string_with_factor(size, 1048576));
        size_str.push_str(" MB");
    } else if size >= 1024 {
        size_str.push_str(&bytes_to_string_with_factor(size, 1024));
        size_str.push_str(" KB");
    } else {
        size_str.push_str(&format!("{}", size));
    }
    if size_str.len() <= 7 {
        size_str.push('\t');
    }
    size_str
}

fn bytes_to_string_with_factor(size: u64, factor: u64) -> String {
    format!("{:.2}", (size as f64 / factor as f64))
}

fn main() {
    let args : Vec<String> = std::env::args().collect();
    let default_file = String::from(".");
    let path = args.get(1).unwrap_or_else(|| {
        &default_file
    });
    let mut map : HashMap<String, FileStats> = HashMap::new();
    let cache = UsersCache::new();
    read_file(&mut map, path, &cache, false);
    print_results(&map);
}

fn read_dir(map : &mut HashMap<String, FileStats>, path : &str, cache: &UsersCache) {
    for p in fs::read_dir(path).expect(&format!("Failed to read path '{}'", path)) {
        let sub_path = p.expect("Failed to read file").path().display().to_string();
        read_file(map, &sub_path, cache, true);
    }
}

fn read_file(map : &mut HashMap<String, FileStats>, path : &str, cache : &UsersCache, no_recursion : bool) {
    let result = File::open(path);
    match result {
        Ok(file) => {
            let meta = file.metadata().expect(&format!("Failed to read '{}' metadata", path));
            let stats = create_file_stats(&meta, cache);
            let stats = stats.expect(&format!("Failed to create file stats for '{}'", path));
            let mut relative_path = String::new();
            if !path.starts_with(".") && !path.starts_with("/") {
                relative_path.push_str("./");
            };
            let is_dir = meta.is_dir();
            relative_path.push_str(path);
            if is_dir && !path.ends_with("/") {
                relative_path.push('/');
            }
            map.insert(relative_path, stats);
            if is_dir && !no_recursion {
                read_dir(map, path, cache);
            } 
        },
        Err(_) => println!("Failed to stat '{}'", path),
    }
}

fn print_results(map: &HashMap<String, FileStats>) {
    let mut keys : Vec<String> = Vec::new();
    for key in map.keys() {
        keys.push(key.to_string());
    }
    keys.sort_unstable();
    for path in keys.iter() {
        let file_stats = map.get(path).unwrap();
        if file_stats.is_dir {
            println!("{}\t\x1B[36m{}\x1B[0m", file_stats, path);
        } else {
            println!("{}\t{}", file_stats, path);
        }
    }
    println!("{} files/directories", keys.len());
}

fn create_file_stats(meta : &Metadata, cache: &UsersCache) -> Result<FileStats, Error> {
    let is_dir = meta.is_dir();
    let size = meta.len();
    let created = meta.created().expect("Failed to reads files creation date");
    let created: DateTime<Utc> = created.into();
    let created = format!("{}", created.format("%d/%m/%Y %T"));
    let permissions = meta.permissions();
    let mode = format!("{:#o}", permissions.mode());
    let mode = &mode[mode.len()-3..];
    let permissions = build_permissions(is_dir, &mode);
    let user = cache.get_user_by_uid(meta.uid()).unwrap();
    let group = cache.get_group_by_gid(meta.gid()).unwrap();
    let user = user.name().to_os_string().into_string().unwrap();
    let group = group.name().to_os_string().into_string().unwrap();

    Ok(FileStats::new(
            is_dir,
            permissions,
            user, 
            group,
            created, 
            size)
        )
}

fn build_permissions(is_dir : bool, mode : &str) -> String {
    let groups : Vec<char> = mode.chars().collect();
    let mut permissions : [char; 10] = unsafe {
        let mut array : [char; 10] = mem::uninitialized();
        for (_i,e) in array.iter_mut().enumerate() {
            ptr::write(e, '-');
        }
        array
    };
    match is_dir {
        true => permissions[0] = PERMS[0],
        _ => permissions[0] = PERMS[1],
    };
    let mut i : usize = 1;
    let mut j : usize = 0;
    while i < 8 {
        build_permission_group(&mut permissions, groups.get(j).unwrap(), i);
        i = i+3;
        j = j+1;
    }
    let result : String = permissions.iter().collect();
    result
}

fn build_permission_group(permissions : &mut [char], mode : &char, index : usize) {
    if index > 9 {
        return;
    }
    if *mode == '0' || *mode == '1' || *mode == '2' {
        permissions[index] = '-';
    } else {
        permissions[index] = 'r';
    }
    if *mode == '0' || *mode == '1' || *mode == '4' || *mode == '5' {
        permissions[index+1] = '-';
    } else {
        permissions[index+1] = 'w';
    }
    if *mode == '0' || *mode == '2' || *mode == '4' || *mode == '6' {
        permissions[index+2] = '-';
    } else {
        permissions[index+2] = 'x';
    }
}