use std::ffi::OsString;
use std::fs::OpenOptions;
use std::io::{Error, ErrorKind, Read, Result, Write, stderr, stdin, stdout};
use std::path::PathBuf;
use uucore::display::Quotable;
use uucore::error::{UResult, strip_errno};
use uucore::translate;
mod cli;
pub use crate::cli::uu_app;
use crate::cli::{Options, OutputErrorMode, options};
#[cfg(target_os = "linux")]
use uucore::signals::ensure_stdout_not_broken;
#[cfg(unix)]
use uucore::signals::{disable_pipe_errors, ignore_interrupts};
#[uucore::main]
pub fn uumain(args: impl uucore::Args) -> UResult<()> {
let matches = uucore::clap_localization::handle_clap_result(uu_app(), args)?;
let append = matches.get_flag(options::APPEND);
let ignore_interrupts = matches.get_flag(options::IGNORE_INTERRUPTS);
let ignore_pipe_errors = matches.get_flag(options::IGNORE_PIPE_ERRORS);
let output_error = matches
.get_one::<String>(options::OUTPUT_ERROR)
.map(|s| match s.as_str() {
"warn" => OutputErrorMode::Warn,
"warn-nopipe" => OutputErrorMode::WarnNoPipe,
"exit" => OutputErrorMode::Exit,
"exit-nopipe" => OutputErrorMode::ExitNoPipe,
_ => unreachable!("clap excluded it"),
})
.or_else(|| ignore_pipe_errors.then_some(OutputErrorMode::WarnNoPipe));
let files = matches
.get_many::<OsString>(options::FILE)
.map(|v| v.cloned().collect())
.unwrap_or_default();
let options = Options {
append,
ignore_interrupts,
ignore_pipe_errors,
files,
output_error,
};
tee(&options).map_err(|_| 1.into())
}
fn tee(options: &Options) -> Result<()> {
#[cfg(unix)]
{
if options.ignore_interrupts {
ignore_interrupts().map_err(|_| Error::from(ErrorKind::Other))?;
}
if options.output_error.is_some() {
disable_pipe_errors().map_err(|_| Error::from(ErrorKind::Other))?;
}
}
let mut writers: Vec<NamedWriter> = options
.files
.iter()
.filter_map(|file| open(file, options.append, options.output_error.as_ref()))
.collect::<Result<Vec<NamedWriter>>>()?;
let had_open_errors = writers.len() != options.files.len();
writers.insert(
0,
NamedWriter {
name: translate!("tee-standard-output").into(),
inner: Writer::Stdout(stdout()),
},
);
let mut output = MultiWriter::new(writers, options.output_error.clone());
let input = NamedReader { inner: stdin() };
#[cfg(target_os = "linux")]
if options.ignore_pipe_errors && !ensure_stdout_not_broken()? && output.writers.len() == 1 {
return Ok(());
}
let res = match copy(input, &mut output) {
Err(e) if e.kind() != ErrorKind::Other => Err(e),
_ => Ok(()),
};
if had_open_errors || res.is_err() || output.flush().is_err() || output.error_occurred() {
Err(Error::from(ErrorKind::Other))
} else {
Ok(())
}
}
fn copy(mut input: impl Read, mut output: impl Write) -> Result<usize> {
const FIRST_BUF_SIZE: usize = if cfg!(target_os = "espidf") {
512
} else {
8 * 1024
};
let mut buffer = [0u8; FIRST_BUF_SIZE];
let mut len = 0;
match input.read(&mut buffer) {
Ok(0) => return Ok(0),
Ok(bytes_count) => {
output.write_all(&buffer[0..bytes_count])?;
len = bytes_count;
if bytes_count < FIRST_BUF_SIZE {
output.flush()?;
return Ok(len);
}
}
Err(e) if e.kind() == ErrorKind::Interrupted => (),
Err(e) => return Err(e),
}
let mut buffer = vec![0u8; 4 * FIRST_BUF_SIZE]; loop {
match input.read(&mut buffer) {
Ok(0) => return Ok(len), Ok(received) => {
output.write_all(&buffer[..received])?;
output.flush()?;
len += received;
}
Err(e) if e.kind() == ErrorKind::Interrupted => {}
Err(e) => return Err(e),
}
}
}
fn open(
name: &OsString,
append: bool,
output_error: Option<&OutputErrorMode>,
) -> Option<Result<NamedWriter>> {
let path = PathBuf::from(name);
let mut options = OpenOptions::new();
let mode = if append {
options.append(true)
} else {
options.truncate(true)
};
match mode.write(true).create(true).open(path.as_path()) {
Ok(file) => Some(Ok(NamedWriter {
inner: Writer::File(file),
name: name.clone(),
})),
Err(f) => {
let _ = writeln!(stderr(), "{}: {f}", name.maybe_quote());
match output_error {
Some(OutputErrorMode::Exit | OutputErrorMode::ExitNoPipe) => Some(Err(f)),
_ => None,
}
}
}
}
struct MultiWriter {
writers: Vec<NamedWriter>,
output_error_mode: Option<OutputErrorMode>,
ignored_errors: usize,
}
impl MultiWriter {
fn new(writers: Vec<NamedWriter>, output_error_mode: Option<OutputErrorMode>) -> Self {
Self {
writers,
output_error_mode,
ignored_errors: 0,
}
}
fn error_occurred(&self) -> bool {
self.ignored_errors != 0
}
}
fn process_error(
mode: Option<&OutputErrorMode>,
f: Error,
writer: &NamedWriter,
ignored_errors: &mut usize,
) -> Result<()> {
match mode {
Some(OutputErrorMode::Warn) => {
let _ = writeln!(stderr(), "{}: {f}", writer.name.maybe_quote());
*ignored_errors += 1;
Ok(())
}
Some(OutputErrorMode::WarnNoPipe) | None => {
if f.kind() != ErrorKind::BrokenPipe {
let _ = writeln!(stderr(), "{}: {f}", writer.name.maybe_quote());
*ignored_errors += 1;
}
Ok(())
}
Some(OutputErrorMode::Exit) => {
let _ = writeln!(stderr(), "{}: {f}", writer.name.maybe_quote());
Err(f)
}
Some(OutputErrorMode::ExitNoPipe) => {
if f.kind() == ErrorKind::BrokenPipe {
Ok(())
} else {
let _ = writeln!(stderr(), "{}: {f}", writer.name.maybe_quote());
Err(f)
}
}
}
}
impl Write for MultiWriter {
fn write(&mut self, buf: &[u8]) -> Result<usize> {
let mut aborted = None;
let mode = self.output_error_mode.clone();
let mut errors = 0;
self.writers.retain_mut(|writer| {
let result = writer.write_all(buf);
match result {
Err(f) => {
if let Err(e) = process_error(mode.as_ref(), f, writer, &mut errors) {
if aborted.is_none() {
aborted = Some(e);
}
}
false
}
_ => true,
}
});
self.ignored_errors += errors;
if let Some(e) = aborted {
Err(e)
} else if self.writers.is_empty() {
Err(Error::from(ErrorKind::Other))
} else {
Ok(buf.len())
}
}
fn flush(&mut self) -> Result<()> {
let mut aborted = None;
let mode = self.output_error_mode.clone();
let mut errors = 0;
self.writers.retain_mut(|writer| {
let result = writer.flush();
match result {
Err(f) => {
if let Err(e) = process_error(mode.as_ref(), f, writer, &mut errors) {
if aborted.is_none() {
aborted = Some(e);
}
}
false
}
_ => true,
}
});
self.ignored_errors += errors;
if let Some(e) = aborted {
Err(e)
} else {
Ok(())
}
}
}
enum Writer {
File(std::fs::File),
Stdout(std::io::Stdout),
}
impl Write for Writer {
fn write(&mut self, buf: &[u8]) -> Result<usize> {
match self {
Self::File(f) => f.write(buf),
Self::Stdout(s) => s.write(buf),
}
}
fn flush(&mut self) -> Result<()> {
match self {
Self::File(f) => f.flush(),
Self::Stdout(s) => s.flush(),
}
}
}
struct NamedWriter {
inner: Writer,
pub name: OsString,
}
impl Write for NamedWriter {
fn write(&mut self, buf: &[u8]) -> Result<usize> {
self.inner.write(buf)
}
fn flush(&mut self) -> Result<()> {
self.inner.flush()
}
}
struct NamedReader {
inner: std::io::Stdin,
}
impl Read for NamedReader {
fn read(&mut self, buf: &mut [u8]) -> Result<usize> {
match self.inner.read(buf) {
Err(f) => {
let _ = writeln!(
stderr(),
"tee: {}",
translate!("tee-error-stdin", "error" => strip_errno(&f))
);
Err(f)
}
okay => okay,
}
}
}