rcore 0.1.0

Core module for Rair
Documentation
/*
 * project.rs: Commands to save/load projects.
 * Copyright (C) 2019  Oddcoder
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

use core::*;
use flate2::write::{ZlibDecoder, ZlibEncoder};
use flate2::Compression;
use helper::*;
use serde::Deserialize;
use std::fs::File;
use std::io::prelude::*;
use std::mem;

#[derive(Default)]
pub struct Save {}

impl Save {
    pub fn new() -> Self {
        Default::default()
    }
}
impl Cmd for Save {
    fn run(&mut self, core: &mut Core, args: &[String]) {
        if args.len() != 1 {
            expect(core, args.len() as u64, 1);
            return;
        }
        let data = match serde_cbor::to_vec(&core) {
            Ok(data) => data,
            Err(e) => return error_msg(core, "Failed to serialize project", &e.to_string()),
        };
        let mut file = match File::create(&args[0]) {
            Ok(file) => file,
            Err(e) => return error_msg(core, "Failed to open file", &e.to_string()),
        };
        let mut compressor = ZlibEncoder::new(Vec::new(), Compression::default());
        compressor.write_all(&data).unwrap();
        let compressed_data = compressor.finish().unwrap();
        if let Err(e) = file.write_all(&compressed_data) {
            return error_msg(core, "Failed to save project", &e.to_string());
        }
    }

    fn help(&self, core: &mut Core) {
        help(core, &"save", &"", vec![("[file_path]", "Save project into given path.")]);
    }
}

#[derive(Default)]
pub struct Load {}

impl Load {
    pub fn new() -> Self {
        Default::default()
    }
}
impl Cmd for Load {
    fn run(&mut self, core: &mut Core, args: &[String]) {
        if args.len() != 1 {
            expect(core, args.len() as u64, 1);
            return;
        }
        let mut file = match File::open(&args[0]) {
            Ok(file) => file,
            Err(e) => return error_msg(core, "Failed to open file", &e.to_string()),
        };
        let mut compressed_data: Vec<u8> = Vec::new();
        if let Err(e) = file.read_to_end(&mut compressed_data) {
            return error_msg(core, "Failed to load project", &e.to_string());
        }
        let mut data = Vec::new();
        let mut decompressor = ZlibDecoder::new(data);
        if let Err(e) = decompressor.write_all(&compressed_data) {
            return error_msg(core, "Failed to decompress project", &e.to_string());
        }
        data = match decompressor.finish() {
            Ok(data) => data,
            Err(e) => return error_msg(core, "Failed to decompress project", &e.to_string()),
        };
        let mut deserializer = serde_cbor::Deserializer::from_slice(&data);

        let mut core2: Core = match Core::deserialize(&mut deserializer) {
            Ok(core) => core,
            Err(e) => return error_msg(core, "Failed to load project", &e.to_string()),
        };
        mem::swap(&mut core.stdout, &mut core2.stdout);
        mem::swap(&mut core.stderr, &mut core2.stderr);
        mem::swap(&mut core.env, &mut core2.env);
        core2.set_commands(core.commands());
        *core = core2;
    }
    fn help(&self, core: &mut Core) {
        help(core, &"load", &"", vec![("[file_path]", "load project from given path.")]);
    }
}

#[cfg(test)]

mod test_project {
    use super::*;
    use rair_io::*;
    use std::fs;
    use writer::*;
    #[test]
    fn test_project_help() {
        let mut core = Core::new_no_colors();
        core.stderr = Writer::new_buf();
        core.stdout = Writer::new_buf();
        let load = Load::new();
        let save = Save::new();
        load.help(&mut core);
        save.help(&mut core);
        assert_eq!(
            core.stdout.utf8_string().unwrap(),
            "Command: [load]\n\n\
             Usage:\n\
             load [file_path]\tload project from given path.\n\
             Command: [save]\n\n\
             Usage:\n\
             save [file_path]\tSave project into given path.\n"
        );
        assert_eq!(core.stderr.utf8_string().unwrap(), "");
    }
    #[test]
    fn test_project() {
        let mut core = Core::new_no_colors();
        core.stderr = Writer::new_buf();
        core.stdout = Writer::new_buf();
        let mut load = Load::new();
        let mut save = Save::new();
        core.io.open("malloc://0x500", IoMode::READ | IoMode::WRITE).unwrap();
        core.io.open_at("malloc://0x1337", IoMode::READ | IoMode::WRITE, 0x31000).unwrap();
        core.io.map(0x31000, 0xfff31000, 0x337).unwrap();
        save.run(&mut core, &["rair_project".to_string()]);
        core.io.close_all();
        load.run(&mut core, &["rair_project".to_string()]);
        core.run("files", &[]);
        core.run("maps", &[]);
        assert_eq!(
            core.stdout.utf8_string().unwrap(),
            "Handle\tStart address\tsize\t\tPermissions\tURI\n\
             0\t0x00000000\t0x00000500\tWRITE | READ\tmalloc://0x500\n\
             1\t0x00031000\t0x00001337\tWRITE | READ\tmalloc://0x1337\n\
             Virtual Address     Physical Address    Size\n\
             0xfff31000          0x31000             0x337\n"
        );
        assert_eq!(core.stderr.utf8_string().unwrap(), "");
        fs::remove_file("rair_project").unwrap();
    }
}