轉換錯誤型別

use std::error::Error;
use std::fmt::{self, Display, Formatter};
use std::fs::{self, File};
use std::io::{self, Read};

#[derive(Debug)]
enum ReadUsernameError {
    IoError(io::Error),
    EmptyUsername(String),
}

impl Error for ReadUsernameError {}

impl Display for ReadUsernameError {
    fn fmt(&self, f: &mut Formatter) -> fmt::Result {
        match self {
            Self::IoError(e) => write!(f, "IO error: {e}"),
            Self::EmptyUsername(filename) => write!(f, "Found no username in {filename}"),
        }
    }
}

impl From<io::Error> for ReadUsernameError {
    fn from(err: io::Error) -> ReadUsernameError {
        ReadUsernameError::IoError(err)
    }
}

fn read_username(path: &str) -> Result<String, ReadUsernameError> {
    let mut username = String::with_capacity(100);
    File::open(path)?.read_to_string(&mut username)?;
    if username.is_empty() {
        return Err(ReadUsernameError::EmptyUsername(String::from(path)));
    }
    Ok(username)
}

fn main() {
    //fs::write("config.dat", "").unwrap();
    let username = read_username("config.dat");
    println!("username or error: {username:?}");
}

重要須知:

  • username 變數可以是 Ok(string)Err(error)
  • 請使用 fs::write 呼叫來測試以下不同情況:沒有檔案、空白檔案、含使用者名稱的檔案。

對所有不需要是 no_std 的錯誤型別來說,實作 std::error::Error 是很好的做法,std::error::Error 會需要 DebugDisplaycoreError Crate 僅於 nightly 提供,因此與 no_std 尚未完全相容。

對這種錯誤型別來說,在可能的情況下實作 CloneEq 通常也很有用,不僅有利於程式庫的測試,使用者也會更輕鬆。但在本例中,我們無法輕易這麼做,因為 io::Error 並未實作 CloneEq