1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
//! This crate allows creation and automatic deletion (based on Drop trait) of files.
//! This is aimed mostly for testing purposes, for example when testing a parser you probably
//! want to read/write file and validate their content

use rand::Rng;
use std::io::Read;
use std::path::Path;
use std::io::Write;
use std::fs::OpenOptions;
use std::io::SeekFrom;
use std::io::Seek;

#[derive(Debug)]
pub struct TestTempFile {
    filename: String,
    random_number: i32,
    final_filename: String,
    file: std::fs::File
}

impl Drop for TestTempFile {
    fn drop(&mut self) {
        self.delete_file();
    }
}

impl Write for TestTempFile {
    fn write(&mut self, buf: &[u8]) -> std::io::Result<usize> {
        self.file.write(buf)
    }

    fn flush(&mut self) -> std::io::Result<()> {
        self.file.flush()
    }

    fn write_all(&mut self, buf: &[u8]) -> std::io::Result<()> {
        self.file.write_all(buf)
    }
}

impl Read for TestTempFile {
    fn read(&mut self, buf: &mut [u8]) -> std::io::Result<usize> {
        self.file.read(buf)
    }
}

impl Seek for TestTempFile {
    fn seek(&mut self, pos: SeekFrom) -> std::io::Result<u64> { self.file.seek(pos) }
}

impl TestTempFile {
    ///
    ///  /// # Arguments
    ///
    /// * `filename` - A String containing the file name
    ///
    /// # Examples
    /// ```
    /// use test_temp_file::TestTempFile;
    /// let mut t = TestTempFile::new(String::from("file_name.txt"));
    /// ```
    pub fn new(filename: String) -> TestTempFile {
        TestTempFile::gen_random_name(filename)
    }

    fn gen_random_name(filename: String) -> TestTempFile {
        let mut rng = rand::thread_rng();
        let random_number = rng.gen_range(0, i32::max_value());
        let final_filename = format!("_{}_{}", random_number, filename);
        let file = OpenOptions::new().
            create(true).
            write(true).
            read(true).
            open(final_filename.clone()).unwrap();
        TestTempFile {
            filename,
            random_number,
            final_filename,
            file
        }
    }

    fn delete_file(&mut self) {
        let final_filename = self.final_filename.as_str();
        if Path::new(final_filename).exists() {
            std::fs::remove_file(final_filename);
        }
    }
}


#[cfg(test)]
mod tests {
    use super::*;
    
    use std::str;

    static FILE_NAME: &'static str = "test_file.txt";

    #[test]
    fn test_constructor() {
        let t = TestTempFile::new(String::from(FILE_NAME));
        assert_eq!(FILE_NAME, t.filename);
    }

    #[test]
    fn test_write() {
        let mut t = TestTempFile::new(String::from(FILE_NAME));
        match t.write_all(b"some bytes") {
            Ok(_) => assert!(true),
            Err(e) => assert!(false, e)
        }
    }

    #[test]
    fn test_write_and_read() {
        let mut t = TestTempFile::new(String::from(FILE_NAME));
        let mut buffer = [0; 10];
        let msg = b"some bytes";
        match t.write_all(msg) {
            Ok(_) => assert!(true),
            Err(e) => assert!(false, e)
        }

        // Need to rewind pointer inside file, since after the write we're pointing to the end

        t.seek(SeekFrom::Start(0));

        match t.read(&mut buffer[..]) {
            Ok(n) => {
                assert_eq!(msg,
                           &buffer[..n],
                           "Left:{:#?}\nRight:{:#?}\n{:#?}",
                           msg, &buffer[..n], t)
            },
            Err(e) => assert!(false, format!("Error:{}\n{:#?}", e.to_string(), t))
        }
    }
}