Skip to content

Commit

Permalink
feat: PanicMessageWriter
Browse files Browse the repository at this point in the history
Signed-off-by: tison <wander4096@gmail.com>
  • Loading branch information
tisonkun committed Nov 22, 2023
1 parent ba2a784 commit a32428d
Show file tree
Hide file tree
Showing 3 changed files with 78 additions and 47 deletions.
2 changes: 2 additions & 0 deletions rustfmt.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
group_imports = "StdExternalCrate"
imports_granularity = "Module"
112 changes: 69 additions & 43 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -45,13 +45,13 @@
#![cfg_attr(feature = "nightly", feature(panic_info_message))]

pub mod report;
use report::{Method, Report};

use std::borrow::Cow;
use std::io::Result as IoResult;
use std::io::{Result as IoResult, Write};
use std::panic::PanicInfo;
use std::path::{Path, PathBuf};

use report::{Method, Report};

/// A convenient metadata struct that describes a crate
///
/// See [`metadata!`]
Expand Down Expand Up @@ -106,8 +106,9 @@ macro_rules! setup_panic {
($meta:expr) => {{
#[allow(unused_imports)]
use std::panic::{self, PanicInfo};

#[allow(unused_imports)]
use $crate::{handle_dump, print_msg, Metadata};
use $crate::{handle_dump, print_msg_with_writer, DefaultPanicMessageWriter, Metadata};

match $crate::PanicStyle::default() {
$crate::PanicStyle::Debug => {}
Expand All @@ -116,7 +117,7 @@ macro_rules! setup_panic {

panic::set_hook(Box::new(move |info: &PanicInfo| {
let file_path = handle_dump(&meta, info);
print_msg(file_path, &meta)
print_msg_with_writer(DefaultPanicMessageWriter, file_path, &meta)
.expect("human-panic: printing error message to console failed");
}));
}
Expand Down Expand Up @@ -153,69 +154,94 @@ impl Default for PanicStyle {
/// Utility function that prints a message to our human users
#[cfg(feature = "color")]
pub fn print_msg<P: AsRef<Path>>(file_path: Option<P>, meta: &Metadata) -> IoResult<()> {
use std::io::Write as _;
print_msg_with_writer(DefaultPanicMessageWriter, file_path, meta)
}

#[cfg(feature = "color")]
pub fn print_msg_with_writer<P: AsRef<Path>, W: PanicMessageWriter>(
writer: W,
file_path: Option<P>,
meta: &Metadata,
) -> IoResult<()> {
let stderr = anstream::stderr();
let mut stderr = stderr.lock();

write!(stderr, "{}", anstyle::AnsiColor::Red.render_fg())?;
write_msg(&mut stderr, file_path, meta)?;
writer.write_panic_msg(&mut stderr, file_path.as_ref(), meta)?;
write!(stderr, "{}", anstyle::Reset.render())?;

Ok(())
}

#[cfg(not(feature = "color"))]
pub fn print_msg<P: AsRef<Path>>(file_path: Option<P>, meta: &Metadata) -> IoResult<()> {
pub fn print_msg_with_writer<P: AsRef<Path>, W: PanicMessageWriter>(
writer: W,
file_path: Option<P>,
meta: &Metadata,
) -> IoResult<()> {
let stderr = std::io::stderr();
let mut stderr = stderr.lock();

write_msg(&mut stderr, file_path, meta)?;
writer.write_panic_msg(&mut stderr, file_path.as_ref(), meta)?;

Ok(())
}

fn write_msg<P: AsRef<Path>>(
buffer: &mut impl std::io::Write,
file_path: Option<P>,
meta: &Metadata,
) -> IoResult<()> {
let (_version, name, authors, homepage) =
(&meta.version, &meta.name, &meta.authors, &meta.homepage);
pub trait PanicMessageWriter {
fn write_panic_msg<P: AsRef<Path>, W: Write>(
&self,
buffer: &mut W,
file_path: Option<&P>,
meta: &Metadata,
) -> IoResult<()>;
}

writeln!(buffer, "Well, this is embarrassing.\n")?;
writeln!(
buffer,
"{name} had a problem and crashed. To help us diagnose the \
pub struct DefaultPanicMessageWriter;

impl PanicMessageWriter for DefaultPanicMessageWriter {
fn write_panic_msg<P: AsRef<Path>, W: Write>(
&self,
buffer: &mut W,
file_path: Option<&P>,
meta: &Metadata,
) -> IoResult<()> {
let (_version, name, authors, homepage) =
(&meta.version, &meta.name, &meta.authors, &meta.homepage);

writeln!(buffer, "Well, this is embarrassing.\n")?;
writeln!(
buffer,
"{name} had a problem and crashed. To help us diagnose the \
problem you can send us a crash report.\n"
)?;
writeln!(
buffer,
"We have generated a report file at \"{}\". Submit an \
)?;
writeln!(
buffer,
"We have generated a report file at \"{}\". Submit an \
issue or email with the subject of \"{} Crash Report\" and include the \
report as an attachment.\n",
match file_path {
Some(fp) => format!("{}", fp.as_ref().display()),
None => "<Failed to store file to disk>".to_string(),
},
name
)?;

if !homepage.is_empty() {
writeln!(buffer, "- Homepage: {homepage}")?;
}
if !authors.is_empty() {
writeln!(buffer, "- Authors: {authors}")?;
}
writeln!(
buffer,
"\nWe take privacy seriously, and do not perform any \
match file_path {
Some(fp) => format!("{}", fp.as_ref().display()),
None => "<Failed to store file to disk>".to_string(),
},
name
)?;

if !homepage.is_empty() {
writeln!(buffer, "- Homepage: {homepage}")?;
}
if !authors.is_empty() {
writeln!(buffer, "- Authors: {authors}")?;
}
writeln!(
buffer,
"\nWe take privacy seriously, and do not perform any \
automated error collection. In order to improve the software, we rely on \
people to submit reports.\n"
)?;
writeln!(buffer, "Thank you kindly!")?;
)?;
writeln!(buffer, "Thank you kindly!")?;

Ok(())
Ok(())
}
}

/// Utility function which will handle dumping information to disk
Expand Down
11 changes: 7 additions & 4 deletions src/report.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,15 @@
//! A `Report` contains the metadata collected about the event
//! to construct a helpful error message.
use backtrace::Backtrace;
use serde_derive::Serialize;
use std::error::Error;
use std::fmt::Write as FmtWrite;
use std::mem;
use std::{env, fs::File, io::Write, path::Path, path::PathBuf};
use std::fs::File;
use std::io::Write;
use std::path::{Path, PathBuf};
use std::{env, mem};

use backtrace::Backtrace;
use serde_derive::Serialize;
use uuid::Uuid;

/// Method of failure.
Expand Down

0 comments on commit a32428d

Please sign in to comment.