rust/tests/ui/write-fmt-errors.rs

56 lines
1.7 KiB
Rust

//@ run-pass
//@ needs-unwind
#![feature(io_error_uncategorized)]
use std::fmt;
use std::io::{self, Error, Write, sink};
use std::panic::catch_unwind;
struct ErrorDisplay;
impl fmt::Display for ErrorDisplay {
fn fmt(&self, _: &mut fmt::Formatter) -> fmt::Result {
Err(fmt::Error)
}
}
struct ErrorWriter;
const WRITER_ERROR: io::ErrorKind = io::ErrorKind::NotConnected;
impl Write for ErrorWriter {
fn write(&mut self, _buf: &[u8]) -> io::Result<usize> {
Err(Error::new(WRITER_ERROR, "not connected"))
}
fn flush(&mut self) -> io::Result<()> { Ok(()) }
}
fn main() {
// Test that an underlying error is propagated
let res = write!(ErrorWriter, "abc");
assert!(res.is_err(), "writer error did not propagate");
// Test that the error from the formatter is detected.
let res = catch_unwind(|| write!(sink(), "{} {} {}", 1, ErrorDisplay, "bar"));
let err = res.expect_err("formatter error did not lead to panic").downcast::<&str>().unwrap();
assert!(
err.contains("formatting trait implementation returned an error"),
"unexpected panic: {}", err
);
// Writer error when there's some string before the first `{}`
let res = write!(ErrorWriter, "abc {}", ErrorDisplay);
assert!(res.is_err(), "writer error did not propagate");
assert_eq!(res.unwrap_err().kind(), WRITER_ERROR);
// Formatter error when the `{}` comes first
let res = catch_unwind(|| write!(ErrorWriter, "{} abc", ErrorDisplay));
let err = res.expect_err("formatter error did not lead to panic").downcast::<&str>().unwrap();
assert!(
err.contains("formatting trait implementation returned an error"),
"unexpected panic: {}", err
);
}