|
1 | 1 | use std::char;
|
2 | 2 | use std::collections::{
|
3 |
| - BTreeMap, |
4 |
| - BTreeSet, |
| 3 | + BTreeMap, BTreeSet, |
5 | 4 | BinaryHeap,
|
6 |
| - HashMap, |
7 |
| - HashSet, |
| 5 | + HashMap, HashSet, |
8 | 6 | LinkedList,
|
9 | 7 | VecDeque,
|
10 | 8 | };
|
| 9 | +use std::env; |
| 10 | +use std::ffi::OsString; |
11 | 11 | use std::hash::{BuildHasher, Hash};
|
12 | 12 | use std::iter::{empty, once};
|
13 | 13 | use std::net::{
|
14 |
| - IpAddr, |
15 |
| - Ipv4Addr, |
16 |
| - Ipv6Addr, |
17 |
| - SocketAddr, |
18 |
| - SocketAddrV4, |
19 |
| - SocketAddrV6, |
| 14 | + IpAddr, Ipv4Addr, Ipv6Addr, |
| 15 | + SocketAddr, SocketAddrV4, SocketAddrV6, |
20 | 16 | };
|
21 | 17 | use std::ops::{Range, RangeFrom, RangeTo, RangeFull};
|
| 18 | +use std::path::PathBuf; |
22 | 19 | use std::sync::Arc;
|
23 | 20 | use std::time::Duration;
|
24 | 21 |
|
@@ -376,7 +373,11 @@ impl<K: Arbitrary + Ord, V: Arbitrary> Arbitrary for BTreeMap<K, V> {
|
376 | 373 | }
|
377 | 374 | }
|
378 | 375 |
|
379 |
| -impl<K: Arbitrary + Eq + Hash, V: Arbitrary, S: BuildHasher + Default + Clone + Send + 'static> Arbitrary for HashMap<K, V, S> { |
| 376 | +impl< |
| 377 | + K: Arbitrary + Eq + Hash, |
| 378 | + V: Arbitrary, |
| 379 | + S: BuildHasher + Default + Clone + Send + 'static, |
| 380 | +> Arbitrary for HashMap<K, V, S> { |
380 | 381 | fn arbitrary<G: Gen>(g: &mut G) -> Self {
|
381 | 382 | let vec: Vec<(K, V)> = Arbitrary::arbitrary(g);
|
382 | 383 | vec.into_iter().collect()
|
@@ -414,7 +415,10 @@ impl<T: Arbitrary + Ord> Arbitrary for BinaryHeap<T> {
|
414 | 415 | }
|
415 | 416 | }
|
416 | 417 |
|
417 |
| -impl<T: Arbitrary + Eq + Hash, S: BuildHasher + Default + Clone + Send + 'static> Arbitrary for HashSet<T, S> { |
| 418 | +impl< |
| 419 | + T: Arbitrary + Eq + Hash, |
| 420 | + S: BuildHasher + Default + Clone + Send + 'static, |
| 421 | +> Arbitrary for HashSet<T, S> { |
418 | 422 | fn arbitrary<G: Gen>(g: &mut G) -> Self {
|
419 | 423 | let vec: Vec<T> = Arbitrary::arbitrary(g);
|
420 | 424 | vec.into_iter().collect()
|
@@ -495,6 +499,66 @@ impl Arbitrary for SocketAddrV6 {
|
495 | 499 | }
|
496 | 500 | }
|
497 | 501 |
|
| 502 | +impl Arbitrary for PathBuf { |
| 503 | + fn arbitrary<G: Gen>(g: &mut G) -> PathBuf { |
| 504 | + // use some real directories as guesses, so we may end up with |
| 505 | + // actual working directories in case that is relevant. |
| 506 | + let here = env::current_dir() |
| 507 | + .unwrap_or(PathBuf::from("/test/directory")); |
| 508 | + let temp = env::temp_dir(); |
| 509 | + let home = env::home_dir() |
| 510 | + .unwrap_or(PathBuf::from("/home/user")); |
| 511 | + let choices = &[ |
| 512 | + here, |
| 513 | + temp, |
| 514 | + home, |
| 515 | + PathBuf::from("."), |
| 516 | + PathBuf::from(".."), |
| 517 | + PathBuf::from("../../.."), |
| 518 | + PathBuf::new(), |
| 519 | + ]; |
| 520 | + let mut p = g.choose(choices).unwrap().clone(); |
| 521 | + p.extend(Vec::<OsString>::arbitrary(g).iter()); |
| 522 | + p |
| 523 | + } |
| 524 | + |
| 525 | + fn shrink(&self) -> Box<Iterator<Item=PathBuf>> { |
| 526 | + let mut shrunk = vec![]; |
| 527 | + let mut popped = self.clone(); |
| 528 | + if popped.pop() { |
| 529 | + shrunk.push(popped); |
| 530 | + } |
| 531 | + |
| 532 | + // Iterating over a Path performs a small amount of normalization. |
| 533 | + let normalized = self.iter().collect::<PathBuf>(); |
| 534 | + if normalized.as_os_str() != self.as_os_str() { |
| 535 | + shrunk.push(normalized); |
| 536 | + } |
| 537 | + |
| 538 | + // Add the canonicalized variant only if canonicalizing the path |
| 539 | + // actually does something, making it (hopefully) smaller. Also, ignore |
| 540 | + // canonicalization if canonicalization errors. |
| 541 | + if let Ok(canonicalized) = self.canonicalize() { |
| 542 | + if canonicalized.as_os_str() != self.as_os_str() { |
| 543 | + shrunk.push(canonicalized); |
| 544 | + } |
| 545 | + } |
| 546 | + |
| 547 | + Box::new(shrunk.into_iter()) |
| 548 | + } |
| 549 | +} |
| 550 | + |
| 551 | +impl Arbitrary for OsString { |
| 552 | + fn arbitrary<G: Gen>(g: &mut G) -> OsString { |
| 553 | + OsString::from(String::arbitrary(g)) |
| 554 | + } |
| 555 | + |
| 556 | + fn shrink(&self) -> Box<Iterator<Item=OsString>> { |
| 557 | + let mystring: String = self.clone().into_string().unwrap(); |
| 558 | + Box::new(mystring.shrink().map(|s| OsString::from(s))) |
| 559 | + } |
| 560 | +} |
| 561 | + |
498 | 562 | impl Arbitrary for String {
|
499 | 563 | fn arbitrary<G: Gen>(g: &mut G) -> String {
|
500 | 564 | let size = { let s = g.size(); g.gen_range(0, s) };
|
@@ -536,8 +600,8 @@ impl Arbitrary for char {
|
536 | 600 | '\t',
|
537 | 601 | '\n',
|
538 | 602 | '~', '`', '!', '@', '#', '$', '%', '^', '&', '*', '(', ')',
|
539 |
| - '_', '-', '=', '+','[', ']', '{', '}',':',';','\'','"','\\', |
540 |
| - '|',',','<','>','.','/','?', |
| 603 | + '_', '-', '=', '+','[', ']', '{', '}', ':', ';', '\'', '"', |
| 604 | + '\\', '|',',','<','>','.','/','?', |
541 | 605 | '0', '1','2','3','4','5','6','7','8','9',
|
542 | 606 | ]).unwrap()
|
543 | 607 | }
|
@@ -584,6 +648,7 @@ impl Arbitrary for char {
|
584 | 648 | }
|
585 | 649 | }
|
586 | 650 |
|
| 651 | + |
587 | 652 | macro_rules! unsigned_shrinker {
|
588 | 653 | ($ty:ty) => {
|
589 | 654 | mod shrinker {
|
@@ -799,7 +864,7 @@ impl<A: Arbitrary + Sync> Arbitrary for Arc<A> {
|
799 | 864 | fn arbitrary<G: Gen>(g: &mut G) -> Arc<A> {
|
800 | 865 | Arc::new(A::arbitrary(g))
|
801 | 866 | }
|
802 |
| - |
| 867 | + |
803 | 868 | fn shrink(&self) -> Box<Iterator<Item=Arc<A>>> {
|
804 | 869 | Box::new((**self).shrink().map(Arc::new))
|
805 | 870 | }
|
@@ -847,6 +912,7 @@ mod test {
|
847 | 912 | };
|
848 | 913 | use std::fmt::Debug;
|
849 | 914 | use std::hash::Hash;
|
| 915 | + use std::path::PathBuf; |
850 | 916 | use super::Arbitrary;
|
851 | 917 |
|
852 | 918 | #[test]
|
@@ -1003,7 +1069,10 @@ mod test {
|
1003 | 1069 | for n in v {
|
1004 | 1070 | let found = shrunk.iter().any(|&i| i == n);
|
1005 | 1071 | if !found {
|
1006 |
| - panic!(format!("Element {:?} was not found in shrink results {:?}", n, shrunk)); |
| 1072 | + panic!(format!( |
| 1073 | + "Element {:?} was not found \ |
| 1074 | + in shrink results {:?}", |
| 1075 | + n, shrunk)); |
1007 | 1076 | }
|
1008 | 1077 | }
|
1009 | 1078 | }
|
@@ -1151,4 +1220,12 @@ mod test {
|
1151 | 1220 | ordered_eq(..3, vec![..0, ..2]);
|
1152 | 1221 | ordered_eq(.., vec![]);
|
1153 | 1222 | }
|
| 1223 | + |
| 1224 | + #[test] |
| 1225 | + fn pathbuf() { |
| 1226 | + ordered_eq(PathBuf::from("/home/foo//.././bar"), vec![ |
| 1227 | + PathBuf::from("/home/foo//.."), |
| 1228 | + PathBuf::from("/home/foo/../bar"), |
| 1229 | + ]); |
| 1230 | + } |
1154 | 1231 | }
|
0 commit comments