1#![warn(bad_style)]
11#![warn(missing_copy_implementations)]
12#![warn(missing_debug_implementations)]
13#![warn(missing_docs)]
14#![warn(trivial_casts)]
15#![warn(trivial_numeric_casts)]
16#![warn(unused)]
17#![warn(unused_extern_crates)]
18#![warn(unused_import_braces)]
19#![warn(unused_qualifications)]
20#![warn(unused_results)]
21#![warn(clippy::if_not_else)]
22#![warn(clippy::invalid_upcast_comparisons)]
23#![warn(clippy::items_after_statements)]
24#![warn(clippy::mut_mut)]
25#![warn(clippy::never_loop)]
26#![warn(clippy::nonminimal_bool)]
27#![warn(clippy::used_underscore_binding)]
28
29use std::cmp::Ordering;
30use std::collections::hash_map::Entry;
31use std::collections::{HashMap, HashSet};
32use std::fmt;
33use std::hash::Hash;
34use std::iter::FromIterator;
35
36#[derive(Clone)]
37struct Dependency<T> {
38 num_prec: usize,
39 succ: HashSet<T>,
40}
41
42impl<T: Hash + Eq> Dependency<T> {
43 fn new() -> Dependency<T> {
44 Dependency {
45 num_prec: 0,
46 succ: HashSet::new(),
47 }
48 }
49}
50
51#[derive(Clone)]
53pub struct TopologicalSort<T> {
54 top: HashMap<T, Dependency<T>>,
55}
56
57impl<T> Default for TopologicalSort<T> {
58 fn default() -> TopologicalSort<T> {
59 TopologicalSort {
60 top: HashMap::new(),
61 }
62 }
63}
64
65impl<T: Hash + Eq + Clone> TopologicalSort<T> {
66 #[inline]
87 pub fn new() -> TopologicalSort<T> {
88 Default::default()
89 }
90
91 #[inline]
93 pub fn len(&self) -> usize {
94 self.top.len()
95 }
96
97 #[inline]
99 pub fn is_empty(&self) -> bool {
100 self.top.is_empty()
101 }
102
103 pub fn add_dependency<P, S>(&mut self, prec: P, succ: S)
110 where
111 P: Into<T>,
112 S: Into<T>,
113 {
114 self.add_dependency_impl(prec.into(), succ.into())
115 }
116
117 fn add_dependency_impl(&mut self, prec: T, succ: T) {
118 match self.top.entry(prec) {
119 Entry::Vacant(e) => {
120 let mut dep = Dependency::new();
121 let _ = dep.succ.insert(succ.clone());
122 let _ = e.insert(dep);
123 }
124 Entry::Occupied(e) => {
125 if !e.into_mut().succ.insert(succ.clone()) {
126 return;
128 }
129 }
130 }
131
132 match self.top.entry(succ) {
133 Entry::Vacant(e) => {
134 let mut dep = Dependency::new();
135 dep.num_prec += 1;
136 let _ = e.insert(dep);
137 }
138 Entry::Occupied(e) => {
139 e.into_mut().num_prec += 1;
140 }
141 }
142 }
143
144 pub fn add_link(&mut self, link: DependencyLink<T>) {
146 self.add_dependency(link.prec, link.succ)
147 }
148
149 pub fn insert<U>(&mut self, elt: U) -> bool
155 where
156 U: Into<T>,
157 {
158 match self.top.entry(elt.into()) {
159 Entry::Vacant(e) => {
160 let dep = Dependency::new();
161 let _ = e.insert(dep);
162 true
163 }
164 Entry::Occupied(_) => false,
165 }
166 }
167
168 pub fn pop(&mut self) -> Option<T> {
173 self.peek().map(T::clone).map(|key| {
174 let _ = self.remove(&key);
175 key
176 })
177 }
178
179 pub fn pop_all(&mut self) -> Vec<T> {
184 let keys = self
185 .top
186 .iter()
187 .filter(|&(_, v)| v.num_prec == 0)
188 .map(|(k, _)| k.clone())
189 .collect::<Vec<_>>();
190 for k in &keys {
191 let _ = self.remove(k);
192 }
193 keys
194 }
195
196 pub fn peek(&self) -> Option<&T> {
199 self.top
200 .iter()
201 .filter(|&(_, v)| v.num_prec == 0)
202 .map(|(k, _)| k)
203 .next()
204 }
205
206 pub fn peek_all(&self) -> Vec<&T> {
209 self.top
210 .iter()
211 .filter(|&(_, v)| v.num_prec == 0)
212 .map(|(k, _)| k)
213 .collect::<Vec<_>>()
214 }
215
216 fn remove(&mut self, prec: &T) -> Option<Dependency<T>> {
217 let result = self.top.remove(prec);
218 if let Some(ref p) = result {
219 for s in &p.succ {
220 if let Some(y) = self.top.get_mut(s) {
221 y.num_prec -= 1;
222 }
223 }
224 }
225 result
226 }
227}
228
229impl<T: PartialOrd + Eq + Hash + Clone> FromIterator<T> for TopologicalSort<T> {
230 fn from_iter<I: IntoIterator<Item = T>>(iter: I) -> TopologicalSort<T> {
231 let mut top = TopologicalSort::new();
232 let mut seen = Vec::<T>::default();
233 for item in iter {
234 let _ = top.insert(item.clone());
235 for seen_item in seen.iter().cloned() {
236 match seen_item.partial_cmp(&item) {
237 Some(Ordering::Less) => {
238 top.add_dependency(seen_item, item.clone());
239 }
240 Some(Ordering::Greater) => {
241 top.add_dependency(item.clone(), seen_item);
242 }
243 _ => (),
244 }
245 }
246 seen.push(item);
247 }
248 top
249 }
250}
251
252#[derive(Copy, Clone, Debug)]
254pub struct DependencyLink<T> {
255 pub prec: T,
257 pub succ: T,
259}
260
261impl<T> From<(T, T)> for DependencyLink<T> {
262 fn from(tuple: (T, T)) -> Self {
263 DependencyLink {
264 succ: tuple.0,
265 prec: tuple.1,
266 }
267 }
268}
269
270impl<T: Eq + Hash + Clone> FromIterator<DependencyLink<T>> for TopologicalSort<T> {
271 fn from_iter<I: IntoIterator<Item = DependencyLink<T>>>(iter: I) -> TopologicalSort<T> {
272 let mut top = TopologicalSort::new();
273 for link in iter {
274 top.add_link(link);
275 }
276 top
277 }
278}
279
280impl<T: Hash + Eq + Clone> Iterator for TopologicalSort<T> {
281 type Item = T;
282
283 fn next(&mut self) -> Option<T> {
284 self.pop()
285 }
286}
287
288impl<T: fmt::Debug + Hash + Eq> fmt::Debug for Dependency<T> {
289 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
290 write!(f, "prec={}, succ={:?}", self.num_prec, self.succ)
291 }
292}
293
294impl<T: fmt::Debug + Hash + Eq + Clone> fmt::Debug for TopologicalSort<T> {
295 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
296 write!(f, "{:?}", self.top)
297 }
298}
299
300#[cfg(test)]
301mod test {
302 use super::TopologicalSort;
303 use quickcheck_macros::quickcheck;
304 use std::iter::FromIterator;
305
306 #[test]
307 fn from_iter() {
308 let t = vec![4, 3, 3, 5, 7, 6, 8];
309 let mut ts = TopologicalSort::<i32>::from_iter(t);
310 assert_eq!(Some(3), ts.next());
311 assert_eq!(Some(4), ts.next());
312 assert_eq!(Some(5), ts.next());
313 assert_eq!(Some(6), ts.next());
314 assert_eq!(Some(7), ts.next());
315 assert_eq!(Some(8), ts.next());
316 assert_eq!(None, ts.next());
317 }
318
319 #[test]
320 fn iter() {
321 let mut ts = TopologicalSort::<i32>::new();
322 ts.add_dependency(1, 2);
323 ts.add_dependency(2, 3);
324 ts.add_dependency(3, 4);
325 assert_eq!(Some(1), ts.next());
326 assert_eq!(Some(2), ts.next());
327 assert_eq!(Some(3), ts.next());
328 assert_eq!(Some(4), ts.next());
329 assert_eq!(None, ts.next());
330 }
331
332 #[test]
333 fn pop_all() {
334 fn check(result: &[i32], ts: &mut TopologicalSort<i32>) {
335 let l = ts.len();
336 let mut v = ts.pop_all();
337 v.sort();
338 assert_eq!(result, &v[..]);
339 assert_eq!(l - result.len(), ts.len());
340 }
341
342 let mut ts = TopologicalSort::new();
343 ts.add_dependency(7, 11);
344 assert_eq!(2, ts.len());
345 ts.add_dependency(7, 8);
346 assert_eq!(3, ts.len());
347 ts.add_dependency(5, 11);
348 assert_eq!(4, ts.len());
349 ts.add_dependency(3, 8);
350 assert_eq!(5, ts.len());
351 ts.add_dependency(3, 10);
352 assert_eq!(6, ts.len());
353 ts.add_dependency(11, 2);
354 assert_eq!(7, ts.len());
355 ts.add_dependency(11, 9);
356 assert_eq!(8, ts.len());
357 ts.add_dependency(11, 10);
358 assert_eq!(8, ts.len());
359 ts.add_dependency(8, 9);
360 assert_eq!(8, ts.len());
361
362 check(&[3, 5, 7], &mut ts);
363 check(&[8, 11], &mut ts);
364 check(&[2, 9, 10], &mut ts);
365 check(&[], &mut ts);
366 }
367
368 #[test]
369 fn cyclic_deadlock() {
370 let mut ts = TopologicalSort::new();
371 ts.add_dependency("stone", "sharp");
372
373 ts.add_dependency("bucket", "hole");
374 ts.add_dependency("hole", "straw");
375 ts.add_dependency("straw", "axe");
376 ts.add_dependency("axe", "sharp");
377 ts.add_dependency("sharp", "water");
378 ts.add_dependency("water", "bucket");
379 assert_eq!(ts.pop(), Some("stone"));
380 assert!(ts.pop().is_none());
381 println!("{:?}", ts);
382 }
383
384 #[quickcheck]
385 fn topo_test_quickcheck(n: usize, edges: Vec<(usize, usize)>) {
386 use std::collections::{HashMap, HashSet};
387
388 let n = n.max(1).min(1000);
389 let mut marked = vec![false; n];
390 let edges = edges
391 .into_iter()
392 .map(|(x, y)| (x % n, y % n))
393 .collect::<Vec<_>>();
394 let mut deps = HashMap::new();
395 let mut toposort = TopologicalSort::<usize>::new();
396
397 for i in 0..n {
398 let _ = deps.insert(i, HashSet::new());
399 assert!(toposort.insert(i));
400 }
401
402 for (op, inp) in edges.iter().map(|(x, y)| (y, x)) {
403 let inps = deps.get_mut(op).unwrap();
404 let _ = inps.insert(*inp);
405 }
406
407 let deps = deps;
408 for (inp, op) in edges {
409 toposort.add_dependency(inp, op);
410 }
411 while let Some(x) = toposort.pop() {
412 for dep in deps.get(&x).unwrap().iter() {
413 assert!(marked[*dep]);
414 }
415 marked[x] = true;
416 }
417
418 if toposort.is_empty() {
419 assert!(marked.into_iter().all(|x| x));
420 } else {
421 let dep_fixed = {
422 let mut ret = (0..n)
423 .map(|i| (i, HashSet::new()))
424 .collect::<HashMap<_, _>>();
425 let mut new_to_add = deps;
426
427 while !new_to_add.is_empty() {
428 for (k, v) in new_to_add.drain() {
429 let inps = ret.get_mut(&k).unwrap();
430 inps.extend(v.into_iter());
431 }
432 for (k, vs) in ret.iter() {
433 for k2 in vs.iter() {
434 for v2 in ret.get(k2).unwrap().iter() {
435 if !vs.contains(v2) {
436 let _ = new_to_add
437 .entry(*k)
438 .or_insert_with(HashSet::new)
439 .insert(*v2);
440 }
441 }
442 }
443 }
444 }
445
446 ret
447 };
448
449 assert!(dep_fixed.into_iter().any(|(op, deps)| deps.contains(&op)));
450 }
451 }
452}