Rustでエラーメッセージを詳細かつ分かりやすくする方法を徹底解説

Rustは、その安全性と高パフォーマンスにより多くの開発者に支持されているプログラミング言語です。しかし、初心者にとってRustのエラーメッセージは、時に難解に感じられることがあります。エラーメッセージが分かりにくいと、問題の解決に時間がかかり、学習の障壁にもなりかねません。

本記事では、Rustのエラーメッセージをより詳細かつユーザーフレンドリーにする方法について解説します。エラー処理のための便利なクレートや、カスタムエラーメッセージの設計方法、コンパイラエラーを理解しやすくするテクニックなどを網羅しています。Rustでの開発をスムーズにし、効率的に問題解決ができるよう、エラーメッセージ改善の具体的な手法を学びましょう。

目次
  1. Rustのエラーメッセージの特徴
    1. 親切なエラー出力
    2. 他の言語との比較
  2. より良いエラーメッセージの設計原則
    1. 1. 具体的で分かりやすい内容
    2. 2. 修正方法の提示
    3. 3. 原因箇所の正確な特定
    4. 4. 簡潔で必要十分な情報
    5. 5. 一貫性と標準化
    6. 6. ユーザーの視点に立つ
  3. thiserrorクレートを使ったエラー処理
    1. thiserrorの導入
    2. 基本的な使い方
    3. 出力例
    4. thiserrorの特徴と利点
    5. 注意点
  4. anyhowクレートを活用したエラー管理
    1. anyhowの導入
    2. 基本的な使い方
    3. 出力例
    4. anyhowの特徴と利点
    5. context()とwith_context()の使い方
    6. anyhowとthiserrorの違い
  5. カスタムエラー型の実装方法
    1. カスタムエラー型の基本的な実装
    2. 出力例
    3. カスタムエラー型の利点
    4. thiserrorを併用したカスタムエラー型
  6. コンパイラエラーメッセージの理解と改善
    1. コンパイラエラーメッセージの構造
    2. よくあるエラーとその解決方法
    3. エラーコードを活用する
    4. rustc --explainコマンドの活用
    5. エラー改善のためのベストプラクティス
  7. ユーザーフレンドリーなパニックメッセージの作成
    1. panic!マクロの基本
    2. パニックメッセージに詳細情報を追加する
    3. カスタムパニックフックの設定
    4. バックトレースの表示
    5. パニックメッセージ作成のベストプラクティス
  8. エラーメッセージ改善の実践例
    1. 1. ResultとOptionのエラー処理
    2. 2. thiserrorを使ったカスタムエラー型
    3. 3. anyhowを使った柔軟なエラー処理
    4. 4. カスタムパニックメッセージ
    5. 5. エラーのログ出力
  9. まとめ

Rustのエラーメッセージの特徴

Rustのエラーメッセージは、他のプログラミング言語と比較して、非常に詳細で具体的です。コンパイル時に問題が検出された場合、Rustは単にエラーの原因を示すだけでなく、問題箇所の詳細な説明や修正案を提示します。

親切なエラー出力

Rustコンパイラは、エラー発生箇所を正確に指摘し、次のような出力を提供します:

error[E0382]: borrow of moved value: `x`
 --> src/main.rs:4:13
  |
2 |     let x = String::from("Hello");
  |         - move occurs because `x` has type `String`, which does not implement the `Copy` trait
3 |     let y = x;
  |             - value moved here
4 |     println!("{}", x);
  |                ^ value borrowed here after move

このエラーメッセージは、問題の原因(xがムーブされた後に使用されている)と、その修正方法を明示しています。

他の言語との比較

  • C/C++:エラーメッセージが抽象的で、修正方法が分かりにくいことが多い。
  • Python:トレースバックは示されるが、修正案や背景の説明は少ない。
  • Rust:エラーの背景、解決策、さらにはドキュメントへのリンクも提供される。

Rustのエラーメッセージは、開発者が効率的に問題を修正し、言語の学習を助ける工夫が施されています。

より良いエラーメッセージの設計原則

エラーメッセージをユーザーフレンドリーにするためには、いくつかの設計原則を意識する必要があります。Rustのエラーメッセージ設計は、これらの原則に基づいており、他の言語やツールでも参考になります。

1. 具体的で分かりやすい内容

エラーメッセージは、抽象的な説明ではなく、具体的に何が問題なのかを伝えるべきです。

  • 悪い例: Error: invalid input
  • 良い例: Error: expected a number, but received a string

2. 修正方法の提示

エラーが発生した原因だけでなく、解決方法も提案するとユーザーの理解が深まります。

  • :
  error[E0382]: borrow of moved value: `x`
   --> src/main.rs:4:13
    |
  2 |     let x = String::from("Hello");
  3 |     let y = x;
    |             - value moved here
  4 |     println!("{}", x);
    |                ^ value borrowed here after move
    |
    = help: consider cloning the value if you need to use it after the move

3. 原因箇所の正確な特定

エラーが発生した箇所を正確に示すことで、ユーザーがすぐに問題を理解できます。Rustのコンパイラは、エラーが発生した行と関連する行をハイライトします。

4. 簡潔で必要十分な情報

エラーメッセージは冗長になりすぎず、必要な情報だけを伝えるべきです。長すぎるメッセージは逆に混乱を招きます。

5. 一貫性と標準化

すべてのエラーメッセージは、一貫したフォーマットとスタイルで書くと、ユーザーが理解しやすくなります。

6. ユーザーの視点に立つ

初心者や経験の浅い開発者にも分かる言葉を使い、専門用語の使用は最小限に抑えます。


これらの原則を守ることで、エラーメッセージが理解しやすくなり、問題解決がスムーズになります。

thiserrorクレートを使ったエラー処理

thiserrorはRustでカスタムエラー型を簡単に作成できるクレートです。thiserrorを使用すると、詳細で分かりやすいエラーメッセージを設計しやすくなります。

thiserrorの導入

Cargo.tomlに以下を追加してthiserrorをインストールします。

[dependencies]
thiserror = "1.0"

基本的な使い方

thiserrorクレートを使ってカスタムエラー型を作成する例を見てみましょう。

use thiserror::Error;

#[derive(Debug, Error)]
pub enum MyError {
    #[error("ファイルが見つかりません: {0}")]
    FileNotFound(String),

    #[error("ネットワーク接続エラー: {0}")]
    NetworkError(String),

    #[error("不正な入力: {0}")]
    InvalidInput(String),
}

fn open_file(filename: &str) -> Result<(), MyError> {
    if filename == "missing.txt" {
        return Err(MyError::FileNotFound(filename.to_string()));
    }
    Ok(())
}

fn main() {
    match open_file("missing.txt") {
        Ok(_) => println!("ファイルが正常に開かれました。"),
        Err(e) => eprintln!("エラーが発生しました: {}", e),
    }
}

出力例

エラーが発生しました: ファイルが見つかりません: missing.txt

thiserrorの特徴と利点

  1. シンプルな構文
    thiserrorのアトリビュートマクロを使うことで、エラー型の定義が非常にシンプルになります。
  2. 柔軟なエラーメッセージ
    フォーマット文字列を使用して、エラーの詳細をカスタマイズできます。
  3. std::error::Errorトレイトの自動実装
    thiserrorを使用することで、Errorトレイトが自動的に実装されるため、エラー処理が標準ライブラリと統合されます。
  4. 使いやすいデバッグ情報
    #[derive(Debug)]でデバッグ情報を簡単に表示できます。

注意点

  • ランタイムオーバーヘッドなし
    thiserrorはコンパイル時に処理されるため、ランタイムオーバーヘッドがありません。
  • ライブラリ用
    thiserrorはライブラリ向けで、アプリケーション全体のエラー処理にはanyhowクレートが適しています。

thiserrorを活用することで、エラー処理が効率化され、ユーザーにとって分かりやすいエラーメッセージを提供できます。

anyhowクレートを活用したエラー管理

anyhowはRustで柔軟なエラーハンドリングを可能にするクレートです。特にアプリケーション開発において、複雑なエラー処理を簡単にし、詳細なエラーメッセージを提供するのに役立ちます。

anyhowの導入

Cargo.tomlに以下を追加してanyhowをインストールします。

[dependencies]
anyhow = "1.0"

基本的な使い方

anyhowを使うと、さまざまな種類のエラーをanyhow::Result型に統一して処理できます。

use anyhow::{Context, Result};
use std::fs::File;

fn read_file(path: &str) -> Result<String> {
    let file = File::open(path).context(format!("ファイルを開けませんでした: {}", path))?;
    Ok(format!("ファイル {:?} が正常に読み込まれました", file))
}

fn main() {
    match read_file("missing.txt") {
        Ok(message) => println!("{}", message),
        Err(e) => eprintln!("エラー: {:?}", e),
    }
}

出力例

エラー: ファイルを開けませんでした: missing.txt

Caused by:
    No such file or directory (os error 2)

anyhowの特徴と利点

  1. シンプルなエラー処理
    anyhow::Result型を使うことで、関数の戻り値として簡単にエラーを返せます。
  2. エラーコンテキストの追加
    context()with_context()を使って、エラーに追加情報を加え、エラーの原因を明確にできます。
  3. 複数エラー型の統一
    異なるエラー型を一つのanyhow::Error型にまとめて扱えるため、エラー処理がシンプルになります。
  4. エラーチェーンの表示
    エラーの原因が複数ある場合、それらをチェーンとして表示し、デバッグを助けます。

context()with_context()の使い方

context()with_context()でエラーに説明を追加する例です。

use anyhow::{Context, Result};
use std::fs::read_to_string;

fn load_config() -> Result<String> {
    read_to_string("config.toml").context("設定ファイルの読み込みに失敗しました")
}

fn main() {
    match load_config() {
        Ok(config) => println!("設定: {}", config),
        Err(e) => eprintln!("エラー: {:?}", e),
    }
}

anyhowthiserrorの違い

  • anyhow:アプリケーション全体のエラー処理向け。エラー型の定義が不要で、柔軟にエラーを扱えます。
  • thiserror:ライブラリ向けのカスタムエラー型定義。具体的なエラー型を明確にしたい場合に使います。

anyhowを活用すると、エラー処理が柔軟かつシンプルになり、アプリケーションの開発効率が向上します。

カスタムエラー型の実装方法

Rustでは、独自のカスタムエラー型を作成することで、エラー処理を細かく制御し、詳細なエラーメッセージを提供できます。カスタムエラー型は、プロジェクトの要件に応じたエラー分類やエラー内容のカスタマイズを可能にします。

カスタムエラー型の基本的な実装

カスタムエラー型を実装するには、std::fmt::Displayトレイトとstd::error::Errorトレイトを実装する必要があります。

以下は、カスタムエラー型の基本的な例です。

use std::fmt;

// カスタムエラー型の定義
#[derive(Debug)]
enum MyError {
    NotFound(String),
    PermissionDenied(String),
    Unknown,
}

// Displayトレイトの実装
impl fmt::Display for MyError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match self {
            MyError::NotFound(msg) => write!(f, "リソースが見つかりません: {}", msg),
            MyError::PermissionDenied(msg) => write!(f, "許可が拒否されました: {}", msg),
            MyError::Unknown => write!(f, "不明なエラーが発生しました"),
        }
    }
}

// Errorトレイトの実装
impl std::error::Error for MyError {}

// カスタムエラー型を使った関数
fn perform_action(action: &str) -> Result<(), MyError> {
    match action {
        "read_file" => Err(MyError::NotFound("config.txt".to_string())),
        "write_file" => Err(MyError::PermissionDenied("config.txt".to_string())),
        _ => Err(MyError::Unknown),
    }
}

fn main() {
    match perform_action("read_file") {
        Ok(_) => println!("操作が成功しました。"),
        Err(e) => eprintln!("エラー: {}", e),
    }
}

出力例

エラー: リソースが見つかりません: config.txt

カスタムエラー型の利点

  1. エラーの分類
    エラーの種類ごとに型を分けることで、エラー処理が明確になります。
  2. 詳細なエラーメッセージ
    エラーごとに異なるメッセージや追加情報を提供できます。
  3. 統一されたエラー処理
    プロジェクト全体で一貫したエラー処理を実装できます。
  4. コンパイル時の安全性
    型システムにより、エラーの種類を明示的に扱えるため、ミスが減ります。

thiserrorを併用したカスタムエラー型

thiserrorを使うと、カスタムエラー型の定義がより簡単になります。

use thiserror::Error;

#[derive(Debug, Error)]
enum MyError {
    #[error("リソースが見つかりません: {0}")]
    NotFound(String),

    #[error("許可が拒否されました: {0}")]
    PermissionDenied(String),

    #[error("不明なエラーが発生しました")]
    Unknown,
}

thiserrorを使えば、手動でDisplayErrorトレイトを実装する必要がなくなります。


カスタムエラー型を実装することで、エラー処理が柔軟になり、ユーザーにとって分かりやすいエラーメッセージを提供できます。

コンパイラエラーメッセージの理解と改善

Rustのコンパイラは非常に詳細で親切なエラーメッセージを提供しますが、最初はその内容を理解するのが難しいかもしれません。ここでは、Rustのコンパイラエラーメッセージの構造と、エラーに対応するための改善方法を解説します。

コンパイラエラーメッセージの構造

Rustコンパイラのエラーメッセージは、一般的に以下の要素で構成されています:

  1. エラーの種類とコード
    エラーの種類(errorwarning など)と、エラーコード(例:E0382)が表示されます。
  2. エラーメッセージの概要
    問題の概要が分かりやすい文章で説明されます。
  3. 問題の発生箇所
    エラーが発生したファイル、行番号、列番号が示され、該当箇所がハイライトされます。
  4. ヘルプや修正案
    解決方法の提案や追加のヘルプが提供されることがあります。

error[E0382]: borrow of moved value: `x`
 --> src/main.rs:4:13
  |
2 |     let x = String::from("Hello");
  |         - move occurs because `x` has type `String`, which does not implement the `Copy` trait
3 |     let y = x;
  |             - value moved here
4 |     println!("{}", x);
  |                ^ value borrowed here after move
  |
  = help: consider cloning the value if you need to use it after the move

よくあるエラーとその解決方法

1. 借用エラー (Borrow Checker)

エラーコード: E0382
原因:所有権が移動した後に値を使用しようとしています。

解決方法:値をクローンするか、参照を使います。

let x = String::from("Hello");
let y = x.clone(); // クローンして新しい所有権を作成
println!("{}", x);

2. 型推論エラー

エラーコード: E0282
原因:コンパイラが型を推論できません。

解決方法:型を明示的に指定します。

let x: i32 = "42".parse().unwrap();

3. ライフタイムエラー

エラーコード: E0106
原因:ライフタイムが不明確で、借用が不正です。

解決方法:ライフタイムを明示的に指定します。

fn example<'a>(s: &'a str) -> &'a str {
    s
}

エラーコードを活用する

Rustのエラーコード(例:E0382)は、公式ドキュメントで詳細な解説が提供されています。エラーコードを検索することで、解決策や具体例を確認できます。

rustc --explainコマンドの活用

エラーコードについてさらに詳しく知りたい場合は、ターミナルで以下のようにrustcを使います。

rustc --explain E0382

これにより、エラーの詳細な説明と解決方法が表示されます。

エラー改善のためのベストプラクティス

  1. エラーメッセージをよく読む
    コンパイラが提案する解決方法を参考にしましょう。
  2. エラーコードを検索する
    エラーコードを検索して、他の開発者の解決例を参考にします。
  3. 小さな変更で問題を分離
    問題箇所を特定するために、コードを小さく分割してテストします。
  4. ツールを活用する
    clippyrust-analyzerなどのツールを使うと、コード品質を向上させる提案を受けられます。

Rustコンパイラのエラーメッセージを正しく理解し、提案された修正方法を適用することで、効率的に問題解決ができるようになります。

ユーザーフレンドリーなパニックメッセージの作成

Rustでは、致命的なエラーが発生した場合、panic!マクロを使用してプログラムをクラッシュさせることができます。しかし、デフォルトのパニックメッセージはシンプルで、場合によっては原因が分かりにくいことがあります。ユーザーフレンドリーなパニックメッセージを作成することで、エラー発生時のデバッグや修正が容易になります。

panic!マクロの基本

panic!マクロは、プログラムが継続できない状態にあるときに使用します。

fn main() {
    panic!("致命的なエラーが発生しました!");
}

出力例:

thread 'main' panicked at '致命的なエラーが発生しました!', src/main.rs:2:5
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace

パニックメッセージに詳細情報を追加する

エラーの原因や解決方法を含めたパニックメッセージを作成すると、デバッグが容易になります。

fn divide(a: i32, b: i32) -> i32 {
    if b == 0 {
        panic!("ゼロでの除算が発生しました!a = {}, b = {}。除算する前に引数を確認してください。", a, b);
    }
    a / b
}

fn main() {
    let result = divide(10, 0);
    println!("結果: {}", result);
}

出力例:

thread 'main' panicked at 'ゼロでの除算が発生しました!a = 10, b = 0。除算する前に引数を確認してください。', src/main.rs:3:9

カスタムパニックフックの設定

パニック時の挙動をカスタマイズするために、カスタムパニックフックを設定できます。これにより、エラー情報をログに記録したり、エラーレポートを生成することが可能です。

use std::panic;

fn main() {
    panic::set_hook(Box::new(|info| {
        eprintln!("カスタムパニックメッセージ: {}", info);
    }));

    panic!("予期しないエラーが発生しました!");
}

出力例:

カスタムパニックメッセージ: panicked at '予期しないエラーが発生しました!', src/main.rs:7:5

バックトレースの表示

エラー発生時のスタックトレースを表示すると、問題の原因がさらに分かりやすくなります。環境変数RUST_BACKTRACEを設定してバックトレースを有効にします。

RUST_BACKTRACE=1 cargo run

出力例:

thread 'main' panicked at '致命的なエラーが発生しました!', src/main.rs:2:5
stack backtrace:
   0: rust_begin_unwind
             at /rustc/xxxx/src/libstd/panicking.rs:487
   1: core::panicking::panic_fmt
             at /rustc/xxxx/src/libcore/panicking.rs:85
   2: main
             at src/main.rs:2

パニックメッセージ作成のベストプラクティス

  1. 明確なエラーメッセージ:何が問題なのかを具体的に記述します。
  2. エラーの原因と解決策:可能ならエラーの原因と修正方法を提示します。
  3. コンテキスト情報:変数の値や実行状況を含めると、デバッグがしやすくなります。
  4. カスタムパニックフック:エラーログやレポート生成が必要な場合はカスタムパニックフックを利用します。

これらの手法を活用することで、Rustプログラムのパニックメッセージを詳細かつユーザーフレンドリーにし、エラー発生時のデバッグや問題解決を効率化できます。

エラーメッセージ改善の実践例

ここでは、Rustにおけるエラーメッセージを改善する具体的な手法を、コード例と共に紹介します。これにより、開発者が問題を素早く理解し、修正できるようになります。


1. ResultOptionのエラー処理

標準ライブラリのResultOptionを使い、エラー時に分かりやすいメッセージを返す方法です。

use std::fs::File;
use std::io::{self, Read};

fn read_file_content(filename: &str) -> Result<String, String> {
    let mut file = File::open(filename).map_err(|_| format!("ファイルが開けませんでした: {}", filename))?;
    let mut content = String::new();
    file.read_to_string(&mut content).map_err(|_| format!("ファイルの読み取りに失敗しました: {}", filename))?;
    Ok(content)
}

fn main() {
    match read_file_content("example.txt") {
        Ok(content) => println!("ファイルの内容:\n{}", content),
        Err(e) => eprintln!("エラー: {}", e),
    }
}

出力例:

エラー: ファイルが開けませんでした: example.txt

2. thiserrorを使ったカスタムエラー型

thiserrorを使うと、エラー型をカスタマイズし、明確なエラーメッセージを提供できます。

use thiserror::Error;
use std::fs::File;
use std::io::{self, Read};

#[derive(Debug, Error)]
enum FileError {
    #[error("ファイルが見つかりません: {0}")]
    NotFound(String),

    #[error("ファイルの読み取りエラー: {0}")]
    ReadError(String),
}

fn read_file(filename: &str) -> Result<String, FileError> {
    let mut file = File::open(filename).map_err(|_| FileError::NotFound(filename.to_string()))?;
    let mut content = String::new();
    file.read_to_string(&mut content).map_err(|_| FileError::ReadError(filename.to_string()))?;
    Ok(content)
}

fn main() {
    match read_file("missing.txt") {
        Ok(content) => println!("ファイル内容:\n{}", content),
        Err(e) => eprintln!("エラー: {}", e),
    }
}

出力例:

エラー: ファイルが見つかりません: missing.txt

3. anyhowを使った柔軟なエラー処理

anyhowを使うと、複数のエラー型を統一し、エラーにコンテキストを追加できます。

use anyhow::{Context, Result};
use std::fs::File;
use std::io::Read;

fn read_file_with_anyhow(filename: &str) -> Result<String> {
    let mut file = File::open(filename).with_context(|| format!("ファイルが開けませんでした: {}", filename))?;
    let mut content = String::new();
    file.read_to_string(&mut content).with_context(|| format!("ファイルの読み取りに失敗しました: {}", filename))?;
    Ok(content)
}

fn main() {
    if let Err(e) = read_file_with_anyhow("missing.txt") {
        eprintln!("エラー: {:?}", e);
    }
}

出力例:

エラー: ファイルが開けませんでした: missing.txt

Caused by:
    No such file or directory (os error 2)

4. カスタムパニックメッセージ

パニック時に詳細な情報を提供する例です。

fn divide(a: i32, b: i32) -> i32 {
    if b == 0 {
        panic!("ゼロでの除算は許可されていません。a = {}, b = {}", a, b);
    }
    a / b
}

fn main() {
    divide(10, 0);
}

出力例:

thread 'main' panicked at 'ゼロでの除算は許可されていません。a = 10, b = 0', src/main.rs:4:9

5. エラーのログ出力

logクレートを使ってエラー情報をログに記録する方法です。

Cargo.tomlに追加:

[dependencies]
log = "0.4"
env_logger = "0.10"

コード:

use log::{error, info};

fn main() {
    env_logger::init();

    info!("アプリケーションを開始します。");

    if let Err(e) = std::fs::read_to_string("missing.txt") {
        error!("ファイル読み込みエラー: {}", e);
    }
}

出力例:

[ERROR] ファイル読み込みエラー: No such file or directory (os error 2)

これらの実践例を使うことで、Rustのエラーメッセージがより分かりやすく、ユーザーフレンドリーになります。適切なクレートや手法を選択し、効率的にエラー処理を改善しましょう。

まとめ

本記事では、Rustにおけるエラーメッセージを詳細かつユーザーフレンドリーにするための手法について解説しました。Rustのエラーメッセージの特徴や、エラー処理を改善するための具体的な方法として、thiserroranyhowクレートの活用、カスタムエラー型の実装、カスタムパニックメッセージの作成、コンパイラエラーメッセージの理解と改善、エラーのログ出力を紹介しました。

これらの手法を使うことで、エラー発生時のデバッグが容易になり、開発効率が向上します。Rustの強力なエラーメッセージシステムを最大限に活用し、分かりやすく修正しやすいコードを書きましょう。

コメント

コメントする

目次
  1. Rustのエラーメッセージの特徴
    1. 親切なエラー出力
    2. 他の言語との比較
  2. より良いエラーメッセージの設計原則
    1. 1. 具体的で分かりやすい内容
    2. 2. 修正方法の提示
    3. 3. 原因箇所の正確な特定
    4. 4. 簡潔で必要十分な情報
    5. 5. 一貫性と標準化
    6. 6. ユーザーの視点に立つ
  3. thiserrorクレートを使ったエラー処理
    1. thiserrorの導入
    2. 基本的な使い方
    3. 出力例
    4. thiserrorの特徴と利点
    5. 注意点
  4. anyhowクレートを活用したエラー管理
    1. anyhowの導入
    2. 基本的な使い方
    3. 出力例
    4. anyhowの特徴と利点
    5. context()とwith_context()の使い方
    6. anyhowとthiserrorの違い
  5. カスタムエラー型の実装方法
    1. カスタムエラー型の基本的な実装
    2. 出力例
    3. カスタムエラー型の利点
    4. thiserrorを併用したカスタムエラー型
  6. コンパイラエラーメッセージの理解と改善
    1. コンパイラエラーメッセージの構造
    2. よくあるエラーとその解決方法
    3. エラーコードを活用する
    4. rustc --explainコマンドの活用
    5. エラー改善のためのベストプラクティス
  7. ユーザーフレンドリーなパニックメッセージの作成
    1. panic!マクロの基本
    2. パニックメッセージに詳細情報を追加する
    3. カスタムパニックフックの設定
    4. バックトレースの表示
    5. パニックメッセージ作成のベストプラクティス
  8. エラーメッセージ改善の実践例
    1. 1. ResultとOptionのエラー処理
    2. 2. thiserrorを使ったカスタムエラー型
    3. 3. anyhowを使った柔軟なエラー処理
    4. 4. カスタムパニックメッセージ
    5. 5. エラーのログ出力
  9. まとめ