You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

117 lines
3.0KB

  1. extern crate image;
  2. extern crate syntect;
  3. extern crate tera;
  4. extern crate toml;
  5. use std::convert::Into;
  6. use std::error::Error as StdError;
  7. use std::fmt;
  8. #[derive(Debug)]
  9. pub enum ErrorKind {
  10. Msg(String),
  11. Tera(tera::Error),
  12. Io(::std::io::Error),
  13. Toml(toml::de::Error),
  14. Image(image::ImageError),
  15. Syntect(syntect::LoadingError),
  16. }
  17. /// The Error type
  18. #[derive(Debug)]
  19. pub struct Error {
  20. /// Kind of error
  21. pub kind: ErrorKind,
  22. pub source: Option<Box<dyn StdError>>,
  23. }
  24. unsafe impl Sync for Error {}
  25. unsafe impl Send for Error {}
  26. impl StdError for Error {
  27. fn source(&self) -> Option<&(dyn StdError + 'static)> {
  28. let mut source = self.source.as_ref().map(|c| &**c);
  29. if source.is_none() {
  30. match self.kind {
  31. ErrorKind::Tera(ref err) => source = err.source(),
  32. _ => (),
  33. };
  34. }
  35. source
  36. }
  37. }
  38. impl fmt::Display for Error {
  39. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  40. match self.kind {
  41. ErrorKind::Msg(ref message) => write!(f, "{}", message),
  42. ErrorKind::Tera(ref e) => write!(f, "{}", e),
  43. ErrorKind::Io(ref e) => write!(f, "{}", e),
  44. ErrorKind::Toml(ref e) => write!(f, "{}", e),
  45. ErrorKind::Image(ref e) => write!(f, "{}", e),
  46. ErrorKind::Syntect(ref e) => write!(f, "{}", e),
  47. }
  48. }
  49. }
  50. impl Error {
  51. /// Creates generic error
  52. pub fn msg(value: impl ToString) -> Self {
  53. Self { kind: ErrorKind::Msg(value.to_string()), source: None }
  54. }
  55. /// Creates generic error with a cause
  56. pub fn chain(value: impl ToString, source: impl Into<Box<dyn StdError>>) -> Self {
  57. Self { kind: ErrorKind::Msg(value.to_string()), source: Some(source.into()) }
  58. }
  59. }
  60. impl From<&str> for Error {
  61. fn from(e: &str) -> Self {
  62. Self::msg(e)
  63. }
  64. }
  65. impl From<String> for Error {
  66. fn from(e: String) -> Self {
  67. Self::msg(e)
  68. }
  69. }
  70. impl From<toml::de::Error> for Error {
  71. fn from(e: toml::de::Error) -> Self {
  72. Self { kind: ErrorKind::Toml(e), source: None }
  73. }
  74. }
  75. impl From<syntect::LoadingError> for Error {
  76. fn from(e: syntect::LoadingError) -> Self {
  77. Self { kind: ErrorKind::Syntect(e), source: None }
  78. }
  79. }
  80. impl From<tera::Error> for Error {
  81. fn from(e: tera::Error) -> Self {
  82. Self { kind: ErrorKind::Tera(e), source: None }
  83. }
  84. }
  85. impl From<::std::io::Error> for Error {
  86. fn from(e: ::std::io::Error) -> Self {
  87. Self { kind: ErrorKind::Io(e), source: None }
  88. }
  89. }
  90. impl From<image::ImageError> for Error {
  91. fn from(e: image::ImageError) -> Self {
  92. Self { kind: ErrorKind::Image(e), source: None }
  93. }
  94. }
  95. /// Convenient wrapper around std::Result.
  96. pub type Result<T> = ::std::result::Result<T, Error>;
  97. // So we can use bail! in all other crates
  98. #[macro_export]
  99. macro_rules! bail {
  100. ($e:expr) => {
  101. return Err($e.into());
  102. };
  103. ($fmt:expr, $($arg:tt)+) => {
  104. return Err(format!($fmt, $($arg)+).into());
  105. };
  106. }