error.rs 1.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344
  1. use std::{process, path::PathBuf, fmt::Display, fs};
  2. #[derive(Debug, Clone)]
  3. pub struct CommandError {
  4. output: process::Output,
  5. log_path: PathBuf,
  6. }
  7. impl CommandError {
  8. pub fn new(output: process::Output, log_path: PathBuf) -> Self {
  9. Self {
  10. output, log_path
  11. }
  12. }
  13. }
  14. impl Display for CommandError {
  15. fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
  16. let log_path = fs::canonicalize(&self.log_path)
  17. .map(|p| p.to_str()
  18. .map(String::from)
  19. .unwrap_or(format!("[path not valid utf-8: {:?}]", p)))
  20. .unwrap_or_else(|e| format!("[invalid path; reason: {}]", e));
  21. write!(
  22. f,
  23. "Command exited with non-zero exit code ({}). Logs can be viewed at {}\nExcerpt from stderr:\n----\n",
  24. self.output.status.code()
  25. .map(|n| i32::to_string(&n))
  26. .unwrap_or("??".to_owned()),
  27. log_path
  28. )?;
  29. let buf: Vec<_> = self.output.stderr.rsplit(|b| *b == b'\n').take(10).collect();
  30. for line in buf.into_iter().rev() {
  31. write!(f, "{}\n", String::from_utf8(line.to_vec()).unwrap_or(String::from("[LINE NOT VALID UTF-8]")))?;
  32. }
  33. write!(f, "----\n")?;
  34. Ok(())
  35. }
  36. }