Merge pull request #361 from akiradeveloper/feature/add-tsort-v2

Add tsort
This commit is contained in:
Arcterus 2014-07-19 20:32:54 -07:00
commit 84c5ccb11b
3 changed files with 194 additions and 0 deletions

View file

@ -203,6 +203,10 @@ path = "true/true.rs"
name = "truncate" name = "truncate"
path = "truncate/truncate.rs" path = "truncate/truncate.rs"
[[bin]]
name = "tsort"
path = "tsort/tsort.rs"
[[bin]] [[bin]]
name = "tty" name = "tty"
path = "tty/tty.rs" path = "tty/tty.rs"

View file

@ -53,6 +53,7 @@ PROGS := \
tr \ tr \
true \ true \
truncate \ truncate \
tsort \
unlink \ unlink \
uniq \ uniq \
wc \ wc \

189
tsort/tsort.rs Normal file
View file

@ -0,0 +1,189 @@
#![crate_name = "tsort"]
/*
* This file is part of the uutils coreutils package.
*
* (c) Ben Eggers <ben.eggers36@gmail.com>
* (c) Akira Hayakawa <ruby.wktk@gmail.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
#![feature(macro_rules)]
extern crate getopts;
extern crate libc;
use std::io;
use std::collections::{HashSet, HashMap};
#[path = "../common/util.rs"]
mod util;
static NAME: &'static str = "tsort";
static VERSION: &'static str = "1.0.0";
pub fn uumain(args: Vec<String>) -> int {
let opts = [
getopts::optflag("h", "help", "display this help and exit"),
getopts::optflag("V", "version", "output version information and exit"),
];
let matches = match getopts::getopts(args.tail(), opts) {
Ok(m) => m,
Err(f) => crash!(1, "{}", f)
};
if matches.opt_present("h") {
println!("{} v{}", NAME, VERSION);
println!("");
println!("Usage:");
println!(" {} [OPTIONS] FILE", NAME);
println!("");
io::print(getopts::usage("Topological sort the strings in FILE. Strings are defined as any sequence of tokens separated by whitespace (tab, space, or newline). If FILE is not passed in, stdin is used instead.", opts).as_slice());
return 0;
}
if matches.opt_present("V") {
println!("{} v{}", NAME, VERSION);
return 0;
}
let files = matches.free.clone();
let input = if files.len() > 1 {
crash!(1, "{}, extra operand '{}'", NAME, matches.free[1]);
} else if files.is_empty() {
"-".to_string()
} else {
files[0].to_string()
};
let mut stdin_buf;
let mut file_buf;
let mut reader = io::BufferedReader::new(
if input.as_slice() == "-" {
stdin_buf = io::stdio::stdin_raw();
&mut stdin_buf as &mut Reader
} else {
file_buf = match io::File::open(&Path::new(input.as_slice())) {
Ok(a) => a,
_ => {
show_error!("{}: No such file or directory", input);
return 1;
}
};
&mut file_buf as &mut Reader
}
);
let mut g = Graph::new();
loop {
match reader.read_line() {
Ok(line) => {
let ab: Vec<&str> = line.as_slice().trim_right_chars('\n').split(' ').collect();
if ab.len() > 2 {
crash!(1, "{}: input contains an odd number of tokens", input);
}
g.add_edge(&ab[0].to_string(), &ab[1].to_string());
},
_ => break
}
}
g.run_tsort();
if !g.is_acyclic() {
crash!(1, "{}, input contains a loop:", input);
}
for x in g.result.iter() {
println!("{}", x);
}
return 0
}
// We use String as a representation of node here
// but using integer may improve performance.
struct Graph {
in_edges: HashMap<String, HashSet<String>>,
out_edges: HashMap<String, Vec<String>>,
result: Vec<String>
}
impl Graph {
fn new() -> Graph {
Graph {
in_edges: HashMap::new(),
out_edges: HashMap::new(),
result: vec!(),
}
}
fn has_node(&self, n: &String) -> bool {
self.in_edges.contains_key(n)
}
fn has_edge(&self, from: &String, to: &String) -> bool {
self.in_edges.find(to).unwrap().contains(from)
}
fn init_node(&mut self, n: &String) {
self.in_edges.insert(n.clone(), HashSet::new());
self.out_edges.insert(n.clone(), vec!());
}
fn add_edge(&mut self, from: &String, to: &String) {
if !self.has_node(to) {
self.init_node(to);
}
if !self.has_node(from) {
self.init_node(from);
}
if !self.has_edge(from, to) {
self.in_edges.find_mut(to).unwrap().insert(from.clone());
self.out_edges.find_mut(from).unwrap().push(to.clone());
}
}
// Kahn's algorithm
// O(|V|+|E|)
fn run_tsort(&mut self) {
let mut start_nodes = vec!();
for (n, edges) in self.in_edges.iter() {
if edges.is_empty() {
start_nodes.push(n.clone());
}
}
while !start_nodes.is_empty() {
let n = start_nodes.shift().unwrap();
self.result.push(n.clone());
let n_out_edges = self.out_edges.find_mut(&n).unwrap();
for m in n_out_edges.iter() {
let m_in_edges = self.in_edges.find_mut(m).unwrap();
m_in_edges.remove(&n);
// If m doesn't have other in-coming edges add it to start_nodes
if m_in_edges.is_empty() {
start_nodes.push(m.clone());
}
}
n_out_edges.clear();
}
}
fn is_acyclic(&self) -> bool {
for (_, edges) in self.out_edges.iter() {
if !edges.is_empty() {
return false
}
}
true
}
}