]> git.lizzy.rs Git - rust.git/commitdiff
Make HashMap::take not corrupt the map. Fixes #19292
authorAlexis Beingessner <a.beingessner@gmail.com>
Tue, 25 Nov 2014 06:16:50 +0000 (01:16 -0500)
committerAlexis Beingessner <a.beingessner@gmail.com>
Tue, 25 Nov 2014 13:41:55 +0000 (08:41 -0500)
src/libstd/collections/hash/map.rs

index 69375e8d4f84e131d5044185422b130c96589472..d34e99187c236a682af49681fa9c2e5f6a6c7053 100644 (file)
@@ -1376,7 +1376,7 @@ pub fn set(&mut self, mut value: V) -> V {
 
     /// Takes the value out of the entry, and returns it
     pub fn take(self) -> V {
-        let (_, _, v) = self.elem.take();
+        let (_, v) = pop_internal(self.elem);
         v
     }
 }
@@ -1433,6 +1433,7 @@ mod test_map {
     use hash;
     use iter::{Iterator,range_inclusive,range_step_inclusive};
     use cell::RefCell;
+    use rand::{weak_rng, Rng};
 
     struct KindaIntLike(int);
 
@@ -2062,4 +2063,37 @@ fn test_entry(){
         assert_eq!(map.get(&10).unwrap(), &1000);
         assert_eq!(map.len(), 6);
     }
+
+    #[test]
+    fn test_entry_take_doesnt_corrupt() {
+        // Test for #19292
+        fn check(m: &HashMap<int, ()>) {
+            for k in m.keys() {
+                assert!(m.contains_key(k),
+                        "{} is in keys() but not in the map?", k);
+            }
+        }
+
+        let mut m = HashMap::new();
+        let mut rng = weak_rng();
+
+        // Populate the map with some items.
+        for _ in range(0u, 50) {
+            let x = rng.gen_range(-10, 10);
+            m.insert(x, ());
+        }
+
+        for i in range(0u, 1000) {
+            let x = rng.gen_range(-10, 10);
+            match m.entry(x) {
+                Vacant(_) => {},
+                Occupied(e) => {
+                    println!("{}: remove {}", i, x);
+                    e.take();
+                },
+            }
+
+            check(&m);
+        }
+    }
 }