Skip to content

Commit af95aab

Browse files
droundyBurntSushi
authored andcommitted
add OsString and PathBuf impls for Arbitrary
1 parent 531a25a commit af95aab

File tree

4 files changed

+117
-22
lines changed

4 files changed

+117
-22
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,4 @@ tags
44
target
55
build
66
Cargo.lock
7+
*~

src/arbitrary.rs

+93-16
Original file line numberDiff line numberDiff line change
@@ -1,24 +1,21 @@
11
use std::char;
22
use std::collections::{
3-
BTreeMap,
4-
BTreeSet,
3+
BTreeMap, BTreeSet,
54
BinaryHeap,
6-
HashMap,
7-
HashSet,
5+
HashMap, HashSet,
86
LinkedList,
97
VecDeque,
108
};
9+
use std::env;
10+
use std::ffi::OsString;
1111
use std::hash::{BuildHasher, Hash};
1212
use std::iter::{empty, once};
1313
use std::net::{
14-
IpAddr,
15-
Ipv4Addr,
16-
Ipv6Addr,
17-
SocketAddr,
18-
SocketAddrV4,
19-
SocketAddrV6,
14+
IpAddr, Ipv4Addr, Ipv6Addr,
15+
SocketAddr, SocketAddrV4, SocketAddrV6,
2016
};
2117
use std::ops::{Range, RangeFrom, RangeTo, RangeFull};
18+
use std::path::PathBuf;
2219
use std::sync::Arc;
2320
use std::time::Duration;
2421

@@ -376,7 +373,11 @@ impl<K: Arbitrary + Ord, V: Arbitrary> Arbitrary for BTreeMap<K, V> {
376373
}
377374
}
378375

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> {
380381
fn arbitrary<G: Gen>(g: &mut G) -> Self {
381382
let vec: Vec<(K, V)> = Arbitrary::arbitrary(g);
382383
vec.into_iter().collect()
@@ -414,7 +415,10 @@ impl<T: Arbitrary + Ord> Arbitrary for BinaryHeap<T> {
414415
}
415416
}
416417

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> {
418422
fn arbitrary<G: Gen>(g: &mut G) -> Self {
419423
let vec: Vec<T> = Arbitrary::arbitrary(g);
420424
vec.into_iter().collect()
@@ -495,6 +499,66 @@ impl Arbitrary for SocketAddrV6 {
495499
}
496500
}
497501

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+
498562
impl Arbitrary for String {
499563
fn arbitrary<G: Gen>(g: &mut G) -> String {
500564
let size = { let s = g.size(); g.gen_range(0, s) };
@@ -536,8 +600,8 @@ impl Arbitrary for char {
536600
'\t',
537601
'\n',
538602
'~', '`', '!', '@', '#', '$', '%', '^', '&', '*', '(', ')',
539-
'_', '-', '=', '+','[', ']', '{', '}',':',';','\'','"','\\',
540-
'|',',','<','>','.','/','?',
603+
'_', '-', '=', '+','[', ']', '{', '}', ':', ';', '\'', '"',
604+
'\\', '|',',','<','>','.','/','?',
541605
'0', '1','2','3','4','5','6','7','8','9',
542606
]).unwrap()
543607
}
@@ -584,6 +648,7 @@ impl Arbitrary for char {
584648
}
585649
}
586650

651+
587652
macro_rules! unsigned_shrinker {
588653
($ty:ty) => {
589654
mod shrinker {
@@ -799,7 +864,7 @@ impl<A: Arbitrary + Sync> Arbitrary for Arc<A> {
799864
fn arbitrary<G: Gen>(g: &mut G) -> Arc<A> {
800865
Arc::new(A::arbitrary(g))
801866
}
802-
867+
803868
fn shrink(&self) -> Box<Iterator<Item=Arc<A>>> {
804869
Box::new((**self).shrink().map(Arc::new))
805870
}
@@ -847,6 +912,7 @@ mod test {
847912
};
848913
use std::fmt::Debug;
849914
use std::hash::Hash;
915+
use std::path::PathBuf;
850916
use super::Arbitrary;
851917

852918
#[test]
@@ -1003,7 +1069,10 @@ mod test {
10031069
for n in v {
10041070
let found = shrunk.iter().any(|&i| i == n);
10051071
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));
10071076
}
10081077
}
10091078
}
@@ -1151,4 +1220,12 @@ mod test {
11511220
ordered_eq(..3, vec![..0, ..2]);
11521221
ordered_eq(.., vec![]);
11531222
}
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+
}
11541231
}

src/tester.rs

+9-4
Original file line numberDiff line numberDiff line change
@@ -103,9 +103,12 @@ impl<G: Gen> QuickCheck<G> {
103103

104104
/// Set the minimum number of tests that needs to pass.
105105
///
106-
/// This actually refers to the minimum number of *valid* *passed* tests that
107-
/// needs to pass for the property to be considered successful.
108-
pub fn min_tests_passed(mut self, min_tests_passed: usize) -> QuickCheck<G> {
106+
/// This actually refers to the minimum number of *valid* *passed* tests
107+
/// that needs to pass for the property to be considered successful.
108+
pub fn min_tests_passed(
109+
mut self,
110+
min_tests_passed: usize,
111+
) -> QuickCheck<G> {
109112
self.min_tests_passed = min_tests_passed;
110113
self
111114
}
@@ -171,7 +174,9 @@ impl<G: Gen> QuickCheck<G> {
171174
if n_tests_passed >= self.min_tests_passed {
172175
info!("(Passed {} QuickCheck tests.)", n_tests_passed)
173176
} else {
174-
panic!("(Unable to generate enough tests, {} not discarded.)", n_tests_passed)
177+
panic!(
178+
"(Unable to generate enough tests, {} not discarded.)",
179+
n_tests_passed)
175180
}
176181

177182
}

src/tests.rs

+14-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ use rand;
33
use super::{QuickCheck, StdGen, TestResult, quickcheck};
44
use std::collections::{HashSet, HashMap};
55
use std::hash::{BuildHasherDefault, SipHasher};
6+
use std::path::PathBuf;
67

78
#[test]
89
fn prop_oob() {
@@ -239,6 +240,13 @@ fn all_tests_discarded_min_tests_passed_missing() {
239240
}
240241

241242
quickcheck! {
243+
/// The following is a very simplistic test, which only verifies
244+
/// that our PathBuf::arbitrary does not panic. Still, that's
245+
/// something! :)
246+
fn pathbuf(_p: PathBuf) -> bool {
247+
true
248+
}
249+
242250
fn basic_hashset(_set: HashSet<u8>) -> bool {
243251
true
244252
}
@@ -247,11 +255,15 @@ quickcheck! {
247255
true
248256
}
249257

250-
fn substitute_hashset(_set: HashSet<u8, BuildHasherDefault<SipHasher>>) -> bool {
258+
fn substitute_hashset(
259+
_set: HashSet<u8, BuildHasherDefault<SipHasher>>
260+
) -> bool {
251261
true
252262
}
253263

254-
fn substitute_hashmap(_map: HashMap<u8, u8, BuildHasherDefault<SipHasher>>) -> bool {
264+
fn substitute_hashmap(
265+
_map: HashMap<u8, u8, BuildHasherDefault<SipHasher>>
266+
) -> bool {
255267
true
256268
}
257269
}

0 commit comments

Comments
 (0)