2018-12-05 19:12:33 +00:00
|
|
|
use color::{ColoredString, Colors, Elem};
|
2018-12-12 10:17:32 +00:00
|
|
|
use icon::Icons;
|
2018-12-01 17:59:53 +00:00
|
|
|
use meta::filetype::FileType;
|
|
|
|
use std::cmp::{Ordering, PartialOrd};
|
|
|
|
use std::path::Path;
|
|
|
|
|
|
|
|
#[derive(Debug, Eq)]
|
|
|
|
pub struct Name {
|
|
|
|
name: String,
|
|
|
|
extension: Option<String>,
|
|
|
|
file_type: FileType,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Name {
|
|
|
|
pub fn new(path: &Path, file_type: FileType) -> Self {
|
2018-12-10 17:26:34 +00:00
|
|
|
let name = path
|
2018-12-01 17:59:53 +00:00
|
|
|
.file_name()
|
|
|
|
.expect("failed to retrieve file name")
|
2018-12-05 20:13:36 +00:00
|
|
|
.to_string_lossy()
|
2018-12-01 17:59:53 +00:00
|
|
|
.to_string();
|
|
|
|
|
|
|
|
let mut extension = None;
|
|
|
|
if let Some(res) = path.extension() {
|
|
|
|
extension = Some(
|
|
|
|
res.to_str()
|
|
|
|
.expect("failed to encode file name")
|
|
|
|
.to_string(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-12-13 15:50:47 +00:00
|
|
|
Self {
|
2018-12-01 17:59:53 +00:00
|
|
|
name,
|
|
|
|
extension,
|
|
|
|
file_type,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-11 09:11:13 +00:00
|
|
|
pub fn render(&self, colors: &Colors, icons: &Icons) -> ColoredString {
|
2018-12-12 07:34:59 +00:00
|
|
|
let icon = icons.get(self);
|
2018-12-13 13:51:31 +00:00
|
|
|
let mut content = String::with_capacity(icon.len() + self.name.len() + 3 /* spaces */);
|
2018-12-11 09:11:13 +00:00
|
|
|
|
2018-12-13 13:51:31 +00:00
|
|
|
content += icon.as_str();
|
2018-12-01 17:59:53 +00:00
|
|
|
|
2018-12-08 22:03:18 +00:00
|
|
|
let elem = match self.file_type {
|
2018-12-12 10:21:45 +00:00
|
|
|
FileType::CharDevice => &Elem::CharDevice,
|
2018-12-08 22:03:18 +00:00
|
|
|
FileType::Directory => &Elem::Dir,
|
|
|
|
FileType::SymLink => &Elem::SymLink,
|
2018-12-12 07:16:43 +00:00
|
|
|
FileType::ExecutableFile => &Elem::ExecutableFile,
|
2018-12-08 22:03:18 +00:00
|
|
|
_ => &Elem::File,
|
2018-12-01 17:59:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
content += &self.name;
|
|
|
|
|
2018-12-05 19:12:33 +00:00
|
|
|
colors.colorize(content, elem)
|
2018-12-01 17:59:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(&self) -> String {
|
|
|
|
self.name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn extension(&self) -> Option<String> {
|
|
|
|
self.extension.clone()
|
|
|
|
}
|
|
|
|
|
2018-12-08 18:52:56 +00:00
|
|
|
pub fn file_type(&self) -> FileType {
|
|
|
|
self.file_type
|
2018-12-01 17:59:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_hidden(&self) -> bool {
|
|
|
|
self.name.starts_with('.')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Ord for Name {
|
2018-12-13 15:50:47 +00:00
|
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
2018-12-17 09:56:24 +00:00
|
|
|
self.name.to_lowercase().cmp(&other.name.to_lowercase())
|
2018-12-01 17:59:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialOrd for Name {
|
2018-12-13 15:50:47 +00:00
|
|
|
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
|
2018-12-10 19:42:51 +00:00
|
|
|
Some(self.name.to_lowercase().cmp(&other.name.to_lowercase()))
|
2018-12-01 17:59:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq for Name {
|
2018-12-13 15:50:47 +00:00
|
|
|
fn eq(&self, other: &Self) -> bool {
|
2018-12-10 19:42:51 +00:00
|
|
|
self.name.eq_ignore_ascii_case(&other.name)
|
2018-12-01 17:59:53 +00:00
|
|
|
}
|
|
|
|
}
|
2018-12-09 15:53:20 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::Name;
|
|
|
|
use ansi_term::Colour;
|
2018-12-13 13:51:31 +00:00
|
|
|
use color::{self, Colors};
|
|
|
|
use icon::{self, Icons};
|
2018-12-09 15:53:20 +00:00
|
|
|
use meta::FileType;
|
|
|
|
use meta::Permissions;
|
2018-12-17 09:56:24 +00:00
|
|
|
use std::cmp::Ordering;
|
2018-12-09 15:53:20 +00:00
|
|
|
use std::fs::{self, File};
|
|
|
|
use std::os::unix::fs::symlink;
|
|
|
|
use std::path::Path;
|
|
|
|
use std::process::Command;
|
|
|
|
use tempdir::TempDir;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_print_file_name() {
|
|
|
|
let tmp_dir = TempDir::new("test_print_file_name").expect("failed to create temp dir");
|
2018-12-13 13:51:31 +00:00
|
|
|
let icons = Icons::new(icon::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
|
|
|
|
// Create the file;
|
|
|
|
let file_path = tmp_dir.path().join("file.txt");
|
|
|
|
File::create(&file_path).expect("failed to create file");
|
|
|
|
let meta = file_path.metadata().expect("failed to get metas");
|
|
|
|
|
2018-12-13 13:51:31 +00:00
|
|
|
let colors = Colors::new(color::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
let file_type = FileType::new(&meta, &Permissions::from(&meta));
|
|
|
|
let name = Name::new(&file_path, file_type);
|
|
|
|
|
2018-12-12 10:17:32 +00:00
|
|
|
assert_eq!(
|
|
|
|
Colour::Fixed(184).paint(" file.txt"),
|
|
|
|
name.render(&colors, &icons)
|
|
|
|
);
|
2018-12-09 15:53:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_print_dir_name() {
|
|
|
|
let tmp_dir = TempDir::new("test_print_dir_name").expect("failed to create temp dir");
|
2018-12-13 13:51:31 +00:00
|
|
|
let icons = Icons::new(icon::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
|
|
|
|
// Chreate the directory
|
|
|
|
let dir_path = tmp_dir.path().join("directory");
|
|
|
|
fs::create_dir(&dir_path).expect("failed to create the dir");
|
|
|
|
let meta = dir_path.metadata().expect("failed to get metas");
|
|
|
|
|
2018-12-13 13:51:31 +00:00
|
|
|
let colors = Colors::new(color::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
let file_type = FileType::new(&meta, &Permissions::from(&meta));
|
|
|
|
let name = Name::new(&dir_path, file_type);
|
|
|
|
|
2018-12-12 10:17:32 +00:00
|
|
|
assert_eq!(
|
|
|
|
Colour::Fixed(33).paint(" directory"),
|
|
|
|
name.render(&colors, &icons)
|
|
|
|
);
|
2018-12-09 15:53:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_print_symlink_name() {
|
|
|
|
let tmp_dir = TempDir::new("test_symlink_name").expect("failed to create temp dir");
|
2018-12-13 13:51:31 +00:00
|
|
|
let icons = Icons::new(icon::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
|
|
|
|
// Create the file;
|
|
|
|
let file_path = tmp_dir.path().join("file.tmp");
|
|
|
|
File::create(&file_path).expect("failed to create file");
|
|
|
|
|
|
|
|
// Create the symlink
|
|
|
|
let symlink_path = tmp_dir.path().join("target.tmp");
|
|
|
|
symlink(&file_path, &symlink_path).expect("failed to create symlink");
|
|
|
|
let meta = symlink_path
|
|
|
|
.symlink_metadata()
|
|
|
|
.expect("failed to get metas");
|
|
|
|
|
2018-12-13 13:51:31 +00:00
|
|
|
let colors = Colors::new(color::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
let file_type = FileType::new(&meta, &Permissions::from(&meta));
|
|
|
|
let name = Name::new(&symlink_path, file_type);
|
|
|
|
|
2018-12-12 10:17:32 +00:00
|
|
|
assert_eq!(
|
2018-12-12 13:23:49 +00:00
|
|
|
Colour::Fixed(44).paint(" target.tmp"),
|
2018-12-12 10:17:32 +00:00
|
|
|
name.render(&colors, &icons)
|
|
|
|
);
|
2018-12-09 15:53:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_print_other_type_name() {
|
|
|
|
let tmp_dir = TempDir::new("test_other_type_name").expect("failed to create temp dir");
|
2018-12-13 13:51:31 +00:00
|
|
|
let icons = Icons::new(icon::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
|
|
|
|
// Create the pipe;
|
|
|
|
let pipe_path = tmp_dir.path().join("pipe.tmp");
|
|
|
|
let success = Command::new("mkfifo")
|
|
|
|
.arg(&pipe_path)
|
|
|
|
.status()
|
|
|
|
.expect("failed to exec mkfifo")
|
|
|
|
.success();
|
|
|
|
assert_eq!(true, success, "failed to exec mkfifo");
|
|
|
|
let meta = pipe_path.metadata().expect("failed to get metas");
|
|
|
|
|
2018-12-13 13:51:31 +00:00
|
|
|
let colors = Colors::new(color::Theme::Default);
|
2018-12-09 15:53:20 +00:00
|
|
|
let file_type = FileType::new(&meta, &Permissions::from(&meta));
|
|
|
|
let name = Name::new(&pipe_path, file_type);
|
|
|
|
|
2018-12-12 10:17:32 +00:00
|
|
|
assert_eq!(
|
2018-12-12 13:23:49 +00:00
|
|
|
Colour::Fixed(184).paint(" pipe.tmp"),
|
2018-12-12 10:17:32 +00:00
|
|
|
name.render(&colors, &icons)
|
|
|
|
);
|
2018-12-13 13:51:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_print_without_icon_or_color() {
|
|
|
|
let tmp_dir =
|
|
|
|
TempDir::new("test_print_without_icon_or_color").expect("failed to create temp dir");
|
|
|
|
let icons = Icons::new(icon::Theme::NoIcon);
|
|
|
|
|
|
|
|
// Create the file;
|
|
|
|
let file_path = tmp_dir.path().join("file.txt");
|
|
|
|
File::create(&file_path).expect("failed to create file");
|
|
|
|
let meta = file_path.metadata().expect("failed to get metas");
|
|
|
|
|
|
|
|
let colors = Colors::new(color::Theme::NoColor);
|
|
|
|
let file_type = FileType::new(&meta, &Permissions::from(&meta));
|
|
|
|
let name = Name::new(&file_path, file_type);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
"file.txt",
|
|
|
|
name.render(&colors, &icons).to_string().as_str()
|
|
|
|
);
|
2018-12-09 15:53:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_extensions_with_valid_file() {
|
|
|
|
let path = Path::new("some-file.txt");
|
|
|
|
|
|
|
|
let name = Name::new(&path, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(Some(String::from("txt")), name.extension());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_extensions_with_file_without_extension() {
|
|
|
|
let path = Path::new(".gitignore");
|
|
|
|
|
|
|
|
let name = Name::new(&path, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(None, name.extension());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_is_hidder_with_hidden_file() {
|
|
|
|
let path = Path::new(".gitignore");
|
|
|
|
|
|
|
|
let name = Name::new(&path, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(true, name.is_hidden());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_is_hidder_with_visible_file() {
|
|
|
|
let path = Path::new("some-file.txt");
|
|
|
|
|
|
|
|
let name = Name::new(&path, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(false, name.is_hidden());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2018-12-17 09:56:24 +00:00
|
|
|
fn test_order_impl_is_case_insensitive() {
|
|
|
|
let path_1 = Path::new("AAAA");
|
|
|
|
let name_1 = Name::new(&path_1, FileType::File);
|
|
|
|
|
|
|
|
let path_2 = Path::new("aaaa");
|
|
|
|
let name_2 = Name::new(&path_2, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(Ordering::Equal, name_1.cmp(&name_2));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_partial_order_impl() {
|
2018-12-09 15:53:20 +00:00
|
|
|
let path_a = Path::new("aaaa");
|
|
|
|
let name_a = Name::new(&path_a, FileType::File);
|
|
|
|
|
|
|
|
let path_z = Path::new("zzzz");
|
|
|
|
let name_z = Name::new(&path_z, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(true, name_a < name_z);
|
|
|
|
}
|
|
|
|
|
2018-12-10 19:42:51 +00:00
|
|
|
#[test]
|
2018-12-17 09:56:24 +00:00
|
|
|
fn test_partial_order_impl_is_case_insensitive() {
|
2018-12-10 19:42:51 +00:00
|
|
|
let path_a = Path::new("aaaa");
|
|
|
|
let name_a = Name::new(&path_a, FileType::File);
|
|
|
|
|
|
|
|
let path_z = Path::new("ZZZZ");
|
|
|
|
let name_z = Name::new(&path_z, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(true, name_a < name_z);
|
|
|
|
}
|
|
|
|
|
2018-12-09 15:53:20 +00:00
|
|
|
#[test]
|
2018-12-17 09:56:24 +00:00
|
|
|
fn test_partial_eq_impl() {
|
2018-12-09 15:53:20 +00:00
|
|
|
let path_1 = Path::new("aaaa");
|
|
|
|
let name_1 = Name::new(&path_1, FileType::File);
|
|
|
|
|
|
|
|
let path_2 = Path::new("aaaa");
|
|
|
|
let name_2 = Name::new(&path_2, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(true, name_1 == name_2);
|
|
|
|
}
|
2018-12-10 19:42:51 +00:00
|
|
|
|
|
|
|
#[test]
|
2018-12-17 09:56:24 +00:00
|
|
|
fn test_partial_eq_impl_is_case_insensitive() {
|
2018-12-10 19:42:51 +00:00
|
|
|
let path_1 = Path::new("AAAA");
|
|
|
|
let name_1 = Name::new(&path_1, FileType::File);
|
|
|
|
|
|
|
|
let path_2 = Path::new("aaaa");
|
|
|
|
let name_2 = Name::new(&path_2, FileType::File);
|
|
|
|
|
|
|
|
assert_eq!(true, name_1 == name_2);
|
|
|
|
}
|
2018-12-09 15:53:20 +00:00
|
|
|
}
|