ddr-mount 1.1.4

ddr-mount allows you to mount image files while passing bad sectors
Documentation
use std::{
    ffi::OsString,
    io::{ErrorKind, Write as IoWrite},
    process::{self, Command},
};

use crate::config::{self, DM_LOCATION};
use crate::error::{self, FileType};
use crate::parser::parse_map;
use log::{error, info};
use std::io::Error as IOError;
use std::path::Path;

/// Mounts the image
pub fn mount(image: OsString, map: OsString, block_size: u32) {
    info!(
        "mount image: {:?}, map: {:?}, block size: {block_size}",
        image, map
    );

    if block_size % 512 != 0 {
        error!("Sector size not a multiple of 512, {}", block_size % 512);
        error::sector_error();
    }

    let image = absolute_image_path(image);

    info!("Full path of image: {:?}", image);

    // mount the image
    let image_mount_path = losetup_mount(&image, block_size);
    info!("Image mounted at {image_mount_path}");
    let image_path = OsString::from(&image_mount_path);

    // mount the device mapper over image mount, creating error I/O range using map file
    let (entry, device_name) = dm_mount(&map, &image_path);

    let mut config = config::Config::read_config();
    config.write_device(image.clone(), entry, image_mount_path, device_name.clone());

    let x = image.to_string_lossy();
    println!("{x} is mounted at {DM_LOCATION}{device_name}");
}

/// Gets the absolute path of the image
fn absolute_image_path(path: OsString) -> OsString {
    let path: &Path = path.as_ref();
    let result = path.canonicalize();
    if let Ok(x) = result {
        x.as_os_str().to_os_string()
    } else {
        let file_name = if let Some(x) = path.to_str() {
            x.to_string()
        } else {
            "".to_string()
        };
        error::check_io_error(
            IOError::from(ErrorKind::NotFound),
            file_name,
            FileType::ImageFile,
        );
    }
}

/// Mounts the image using losetup. First, it finds an empty loop device using losetup -f
/// Then it mounts by doing:
/// ```losetup {loopdev} {path to image file} -b {sector size} -r```
/// Finally returns the loopdev path
fn losetup_mount(image: &OsString, block_size: u32) -> String {
    let losetup_next_loop_device = Command::new("losetup")
        .args(["-f"])
        .output()
        .unwrap_or_else(|e| {
            error!("Unable to run losetup to find nextloop: {:?}", e);
            error::mount_error()
        });
    let image_mount_path = String::from_utf8(losetup_next_loop_device.stdout)
        .unwrap_or_else(|e| {
            error!("Unable to get next loop device from losetup: {:?}", e);
            error::mount_error()
        })
        .trim_matches('\n')
        .to_string();

    let args = [
        &OsString::from(&image_mount_path),
        image,
        &OsString::from("-r"),
        &OsString::from("-b"),
        &OsString::from(block_size.to_string()),
    ];

    info!(
        "losetup{}",
        args.iter().fold("".to_string(), |carry, item| format!(
            "{carry} {}",
            item.to_string_lossy()
        ))
    );

    let image_mount_status = Command::new("losetup")
        .args(args)
        .stdin(process::Stdio::null())
        .output()
        .unwrap_or_else(|e| {
            error!("losetup run error {:?}", e);
            error::mount_error()
        });

    if !image_mount_status.status.success() {
        error!("Losetup reported an error");
        eprintln!("{}", String::from_utf8(image_mount_status.stderr).unwrap());
        error::mount_error();
    }
    image_mount_path
}

/// Mounts the image using the parse map with the following commmand
/// ```dmsetup create {device name}``` and passes parse map in stdin
fn dm_mount(map: &OsString, image_path: &OsString) -> (u32, String) {
    let entry = config::get_next_devices();
    let image_path = image_path.to_str().unwrap();

    let device_name = format!("{}{}", config::DEVICE_NAME, entry);
    let device_mapper = &parse_map(map, image_path);

    let args = ["create", &device_name];
    info!(
        "dmsetup{}",
        args.iter()
            .fold("".to_string(), |carry, item| format!("{carry} {item}"))
    );

    let mut dm_mount_process = Command::new("dmsetup")
        .args(args)
        .stdin(process::Stdio::piped())
        .spawn()
        .unwrap_or_else(|e| {
            error!("Unable to run dmsetup: {:?}", e);
            error::mount_error_clean(image_path)
        });
    dm_mount_process
        .stdin
        .take()
        .unwrap_or_else(|| {
            error!("Unable to take stdin of dmsetup");
            error::mount_error_clean(image_path)
        })
        .write_all(device_mapper.as_bytes())
        .unwrap_or_else(|e| {
            error!("Unable to write to stdin of dmsetup: {:?}", e);
            error::mount_error_clean(image_path)
        });

    let status = dm_mount_process.wait_with_output().unwrap_or_else(|e| {
        error!(
            "Unable to get status of dmsetup after writing stdin: {:?}",
            e
        );
        error::mount_error_clean(image_path)
    });

    if !status.status.success() {
        error!("Dmsetup reported an error");
        eprintln!("{}", String::from_utf8(status.stderr).unwrap());
        error::mount_error_clean(image_path);
    }

    (entry, device_name)
}