aeruginous 3.7.17

The Aeruginous Open Source Development Toolbox.
Documentation
/*********************** GNU General Public License 3.0 ***********************\
|                                                                              |
|  Copyright (C) 2023 Kevin Matthes                                            |
|                                                                              |
|  This program is free software: you can redistribute it and/or modify        |
|  it under the terms of the GNU 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 General Public License for more details.                                |
|                                                                              |
|  You should have received a copy of the GNU General Public License           |
|  along with this program.  If not, see <https://www.gnu.org/licenses/>.      |
|                                                                              |
\******************************************************************************/

use crate::{AppendAsLine, ToStderr};
use std::{io::stdin, path::PathBuf};
use sysexits::Result;

/// Read from common sources of input.
#[deprecated(since = "3.7.3", note = "use `aeruginous_io` instead")]
pub trait ReadFile {
    /// The shared logic of all methods.
    ///
    /// This method defines the common behaviour of all methods this trait
    /// provides.  Implementations should also consider the possibilities to
    /// read from both (a) file(s) and [`std::io::Stdin`].
    ///
    /// `show_error_messages` shall control whether to write error messages to
    /// [`std::io::Stderr`], if appropriate.
    ///
    /// # Errors
    ///
    /// See [`sysexits::ExitCode`].
    fn behaviour(&self, show_error_messages: bool) -> Result<String>;

    /// Read the input stream(s) and write error messages.
    ///
    /// # Errors
    ///
    /// See [`Self::behaviour`].
    fn read(&self) -> Result<String> {
        self.behaviour(true)
    }

    /// Read the input stream(s) without writing error messages.
    ///
    /// # Errors
    ///
    /// See [`Self::behaviour`].
    fn read_silently(&self) -> Result<String> {
        self.behaviour(false)
    }
}

#[allow(deprecated)]
impl ReadFile for Option<PathBuf> {
    fn behaviour(&self, show_error_messages: bool) -> Result<String> {
        self.as_ref().map_or_else(
            || stdin().behaviour(show_error_messages),
            |path| ReadFile::behaviour(path, show_error_messages),
        )
    }
}

#[allow(deprecated)]
impl ReadFile for &Option<PathBuf> {
    fn behaviour(&self, show_error_messages: bool) -> Result<String> {
        (*self).behaviour(show_error_messages)
    }
}

#[allow(deprecated)]
impl ReadFile for PathBuf {
    fn behaviour(&self, show_error_messages: bool) -> Result<String> {
        if self.is_dir() {
            match self.read_dir() {
                Ok(directory) => {
                    let mut result = String::new();

                    for entry in directory {
                        match entry {
                            Ok(entry) => {
                                match entry
                                    .path()
                                    .behaviour(show_error_messages)
                                {
                                    Ok(string) => result.push_str(&string),
                                    Err(error) => return Err(error),
                                }
                            }
                            Err(error) => {
                                return error.to_stderr(show_error_messages);
                            }
                        }
                    }

                    Ok(result)
                }
                Err(error) => error.to_stderr(show_error_messages),
            }
        } else {
            match std::fs::read_to_string(self) {
                Ok(string) => Ok(string),
                Err(error) => error.to_stderr(show_error_messages),
            }
        }
    }
}

#[allow(deprecated)]
impl ReadFile for std::io::Stdin {
    fn behaviour(&self, show_error_messages: bool) -> Result<String> {
        let mut result = String::new();

        for line in stdin().lines() {
            match line {
                Ok(string) => result.append_as_line(string),
                Err(error) => return error.to_stderr(show_error_messages),
            }
        }

        Ok(result)
    }
}

#[allow(deprecated)]
impl ReadFile for &str {
    fn behaviour(&self, show_error_messages: bool) -> Result<String> {
        PathBuf::from(self).behaviour(show_error_messages)
    }
}

#[allow(deprecated)]
impl ReadFile for &Vec<PathBuf> {
    fn behaviour(&self, show_error_messages: bool) -> Result<String> {
        if self.is_empty() {
            stdin().behaviour(show_error_messages)
        } else {
            let mut result = String::new();

            for file in *self {
                result
                    .push_str(&ReadFile::behaviour(file, show_error_messages)?);
            }

            Ok(result)
        }
    }
}

/******************************************************************************/