1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
//! Functions and types relating to error handling.

use std::error::Error;
use std::fmt::{self, Display, Formatter};
use std::io;
use std::result;

use image::ImageError;
use sdl2::video::WindowBuildError;
use sdl2::IntegerOrSdlError;

/// A specialized `Result` type for Tetra.
///
/// All Tetra functions with a recoverable failure condition will return this type.
/// In your game code, you can either use it directly, or wrap it in your own error type.
pub type Result<T = ()> = result::Result<T, TetraError>;

/// Represents the types of error that can occur in a Tetra game.
///
/// Note that if you `match` on this enum, you will be forced to add a wildcard arm by the compiler.
/// This is so that if a new error type is added later on, it will not break your code.
#[derive(Debug)]
pub enum TetraError {
    /// An error that occurred while performing an I/O operation (e.g. while loading a file).
    Io(io::Error),

    /// An error that was returned by SDL.
    Sdl(String),

    /// An error that was returned by OpenGL.
    OpenGl(String),

    /// An error that occured while processing an image.
    Image(ImageError),

    /// This is here so that adding new error types will not be a breaking change.
    /// Can be removed once #[non_exhaustive] is stabilized.
    #[doc(hidden)]
    __Nonexhaustive,
}

impl Display for TetraError {
    fn fmt(&self, f: &mut Formatter) -> fmt::Result {
        match self {
            TetraError::Io(e) => write!(f, "IO error: {}", e),
            TetraError::Sdl(e) => write!(f, "SDL error: {}", e),
            TetraError::OpenGl(e) => write!(f, "OpenGL error: {}", e),
            TetraError::Image(e) => write!(f, "Image processing error: {}", e),
            TetraError::__Nonexhaustive => unreachable!(),
        }
    }
}

impl Error for TetraError {
    fn source(&self) -> Option<&(dyn Error + 'static)> {
        match self {
            TetraError::Io(e) => Some(e),
            TetraError::Sdl(_) => None,
            TetraError::OpenGl(_) => None,
            TetraError::Image(e) => Some(e),
            TetraError::__Nonexhaustive => unreachable!(),
        }
    }
}

impl From<io::Error> for TetraError {
    fn from(e: io::Error) -> TetraError {
        TetraError::Io(e)
    }
}

impl From<ImageError> for TetraError {
    fn from(e: ImageError) -> TetraError {
        TetraError::Image(e)
    }
}

impl From<WindowBuildError> for TetraError {
    fn from(e: WindowBuildError) -> TetraError {
        TetraError::Sdl(e.to_string())
    }
}

impl From<IntegerOrSdlError> for TetraError {
    fn from(e: IntegerOrSdlError) -> TetraError {
        TetraError::Sdl(e.to_string())
    }
}