#![warn(missing_docs)] //! A macro which makes errors easy to write //! //! Minimum type is like this: //! //! ```rust //! #[macro_use] extern crate quick_error; //! # fn main() {} //! //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! Variant1 {} //! } //! } //! ``` //! Both ``pub`` and non-public types may be declared, and all meta attributes //! (such as ``#[derive(Debug)]``) are forwarded as is. The `Debug` must be //! implemented (but you may do that yourself if you like). The documentation //! comments ``/// something`` (as well as other meta attrbiutes) on variants //! are allowed. //! //! # Allowed Syntax //! //! You may add arbitrary parameters to any struct variant: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! /// IO Error //! Io(err: std::io::Error) {} //! /// Utf8 Error //! Utf8(err: std::str::Utf8Error) {} //! } //! } //! ``` //! //! Note unlike in normal Enum declarations you declare names of fields (which //! are omitted from type). How they can be used is outlined below. //! //! Now you might have noticed trailing braces `{}`. They are used to define //! implementations. By default: //! //! * `Error::description()` returns variant name as static string //! * `Error::cause()` returns None (even if type wraps some value) //! * `Display` outputs `description()` //! * No `From` implementations are defined //! //! To define description simply add `description(value)` inside braces: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! Io(err: std::io::Error) { //! description(err.description()) //! } //! Utf8(err: std::str::Utf8Error) { //! description("utf8 error") //! } //! } //! } //! ``` //! //! Normal rules for borrowing apply. So most of the time description either //! returns constant string or forwards description from enclosed type. //! //! To change `cause` method to return some error, add `cause(value)`, for //! example: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! Io(err: std::io::Error) { //! cause(err) //! description(err.description()) //! } //! Utf8(err: std::str::Utf8Error) { //! description("utf8 error") //! } //! Other(err: Box) { //! cause(&**err) //! description(err.description()) //! } //! } //! } //! ``` //! Note you don't need to wrap value in `Some`, its implicit. In case you want //! `None` returned just omit the `cause`. You can't return `None` //! conditionally. //! //! To change how each clause is `Display`ed add `display(pattern,..args)`, //! for example: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! Io(err: std::io::Error) { //! display("I/O error: {}", err) //! } //! Utf8(err: std::str::Utf8Error) { //! display("Utf8 error, valid up to {}", err.valid_up_to()) //! } //! } //! } //! ``` //! //! If you need a reference to the error when `Display`ing, you can instead use //! `display(x) -> (pattern, ..args)`, where `x` sets the name of the reference. //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! use std::error::Error; // put methods like `description()` of this trait into scope //! //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! Io(err: std::io::Error) { //! display(x) -> ("{}: {}", x.description(), err) //! } //! Utf8(err: std::str::Utf8Error) { //! display(self_) -> ("{}, valid up to {}", self_.description(), err.valid_up_to()) //! } //! } //! } //! ``` //! //! To convert to the type from any other, use one of the three forms of //! `from` clause. //! //! For example, to convert simple wrapper use bare `from()`: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! Io(err: std::io::Error) { //! from() //! } //! } //! } //! ``` //! //! This implements ``From``. //! //! To convert to singleton enumeration type (discarding the value), use //! the `from(type)` form: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! FormatError { //! from(std::fmt::Error) //! } //! } //! } //! ``` //! //! And the most powerful form is `from(var: type) -> (arguments...)`. It //! might be used to convert to type with multiple arguments or for arbitrary //! value conversions: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # fn main() {} //! # //! quick_error! { //! #[derive(Debug)] //! pub enum SomeError { //! FailedOperation(s: &'static str, errno: i32) { //! from(errno: i32) -> ("os error", errno) //! from(e: std::io::Error) -> ("io error", e.raw_os_error().unwrap()) //! } //! /// Converts from both kinds of utf8 errors //! Utf8(err: std::str::Utf8Error) { //! from() //! from(err: std::string::FromUtf8Error) -> (err.utf8_error()) //! } //! } //! } //! ``` //! # Context //! //! Since quick-error 1.1 we also have a `context` declaration, which is //! similar to (the longest form of) `from`, but allows adding some context to //! the error. We need a longer example to demonstrate this: //! //! ```rust //! # #[macro_use] extern crate quick_error; //! # use std::io; //! # use std::fs::File; //! # use std::path::{Path, PathBuf}; //! # //! use quick_error::ResultExt; //! //! quick_error! { //! #[derive(Debug)] //! pub enum Error { //! File(filename: PathBuf, err: io::Error) { //! context(path: &'a Path, err: io::Error) //! -> (path.to_path_buf(), err) //! } //! } //! } //! //! fn openfile(path: &Path) -> Result<(), Error> { //! try!(File::open(path).context(path)); //! //! // If we didn't have context, the line above would be written as; //! // //! // try!(File::open(path) //! // .map_err(|err| Error::File(path.to_path_buf(), err))); //! //! Ok(()) //! } //! //! # fn main() { //! # openfile(Path::new("/etc/somefile")).ok(); //! # } //! ``` //! //! Each `context(a: A, b: B)` clause implements //! `From> for Error`. Which means multiple `context` clauses //! are a subject to the normal coherence rules. Unfortunately, we can't //! provide full support of generics for the context, but you may either use a //! lifetime `'a` for references or `AsRef` (the latter means `A: //! AsRef`, and `Type` must be concrete). It's also occasionally useful //! to use a tuple as a type of the first argument. //! //! You also need to `use quick_error::ResultExt` extension trait to get //! working `.context()` method. //! //! More info on context in [this article](http://bit.ly/1PsuxDt). //! //! All forms of `from`, `display`, `description`, `cause`, and `context` //! clauses can be combined and put in arbitrary order. Only `from` and //! `context` can be used multiple times in single variant of enumeration. //! Docstrings are also okay. Empty braces can be omitted as of quick_error //! 0.1.3. //! //! # Private Enums //! //! Since quick-error 1.2.0 we have a way to make a private enum that is //! wrapped by public structure: //! //! ```rust //! #[macro_use] extern crate quick_error; //! # fn main() {} //! //! quick_error! { //! #[derive(Debug)] //! pub enum PubError wraps ErrorEnum { //! Variant1 {} //! } //! } //! ``` //! //! This generates data structures like this //! //! ```rust //! //! pub struct PubError(ErrorEnum); //! //! enum ErrorEnum { //! Variant1, //! } //! //! ``` //! //! Which in turn allows you to export just `PubError` in your crate and keep //! actual enumeration private to the crate. This is useful to keep backwards //! compatibility for error types. Currently there is no shorcuts to define //! error constructors for the inner type, but we consider adding some in //! future versions. //! //! It's possible to declare internal enum as public too. //! //! /// Main macro that does all the work #[macro_export] macro_rules! quick_error { ( $(#[$meta:meta])* pub enum $name:ident { $($chunks:tt)* } ) => { quick_error!(SORT [pub enum $name $(#[$meta])* ] items [] buf [] queue [ $($chunks)* ]); }; ( $(#[$meta:meta])* enum $name:ident { $($chunks:tt)* } ) => { quick_error!(SORT [enum $name $(#[$meta])* ] items [] buf [] queue [ $($chunks)* ]); }; ( $(#[$meta:meta])* pub enum $name:ident wraps $enum_name:ident { $($chunks:tt)* } ) => { quick_error!(WRAPPER $enum_name [ pub struct ] $name $(#[$meta])*); quick_error!(SORT [enum $enum_name $(#[$meta])* ] items [] buf [] queue [ $($chunks)* ]); }; ( $(#[$meta:meta])* pub enum $name:ident wraps pub $enum_name:ident { $($chunks:tt)* } ) => { quick_error!(WRAPPER $enum_name [ pub struct ] $name $(#[$meta])*); quick_error!(SORT [pub enum $enum_name $(#[$meta])* ] items [] buf [] queue [ $($chunks)* ]); }; ( $(#[$meta:meta])* enum $name:ident wraps $enum_name:ident { $($chunks:tt)* } ) => { quick_error!(WRAPPER $enum_name [ struct ] $name $(#[$meta])*); quick_error!(SORT [enum $enum_name $(#[$meta])* ] items [] buf [] queue [ $($chunks)* ]); }; ( $(#[$meta:meta])* enum $name:ident wraps pub $enum_name:ident { $($chunks:tt)* } ) => { quick_error!(WRAPPER $enum_name [ struct ] $name $(#[$meta])*); quick_error!(SORT [pub enum $enum_name $(#[$meta])* ] items [] buf [] queue [ $($chunks)* ]); }; ( WRAPPER $internal:ident [ $($strdef:tt)* ] $strname:ident $(#[$meta:meta])* ) => { $(#[$meta])* $($strdef)* $strname ( $internal ); impl ::std::fmt::Display for $strname { fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result { ::std::fmt::Display::fmt(&self.0, f) } } impl From<$internal> for $strname { fn from(err: $internal) -> Self { $strname(err) } } impl ::std::error::Error for $strname { fn description(&self) -> &str { self.0.description() } fn cause(&self) -> Option<&::std::error::Error> { self.0.cause() } } }; // Queue is empty, can do the work (SORT [enum $name:ident $( #[$meta:meta] )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [ ] queue [ ] ) => { quick_error!(ENUM_DEFINITION [enum $name $( #[$meta] )*] body [] queue [$($( #[$imeta] )* => $iitem: $imode [$( $ivar: $ityp ),*] )*] ); quick_error!(IMPLEMENTATIONS $name {$( $iitem: $imode [$(#[$imeta])*] [$( $ivar: $ityp ),*] {$( $ifuncs )*} )*}); $( quick_error!(ERROR_CHECK $imode $($ifuncs)*); )* }; (SORT [pub enum $name:ident $( #[$meta:meta] )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [ ] queue [ ] ) => { quick_error!(ENUM_DEFINITION [pub enum $name $( #[$meta] )*] body [] queue [$($( #[$imeta] )* => $iitem: $imode [$( $ivar: $ityp ),*] )*] ); quick_error!(IMPLEMENTATIONS $name {$( $iitem: $imode [$(#[$imeta])*] [$( $ivar: $ityp ),*] {$( $ifuncs )*} )*}); $( quick_error!(ERROR_CHECK $imode $($ifuncs)*); )* }; // Add meta to buffer (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )*] queue [ #[$qmeta:meta] $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*] buf [$( #[$bmeta] )* #[$qmeta] ] queue [$( $tail )*]); }; // Add ident to buffer (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )*] queue [ $qitem:ident $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*] buf [$(#[$bmeta])* => $qitem : UNIT [ ] ] queue [$( $tail )*]); }; // Flush buffer on meta after ident (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )* => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ] queue [ #[$qmeta:meta] $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] enum [$( $(#[$emeta])* => $eitem $(( $($etyp),* ))* )* $(#[$bmeta])* => $bitem: $bmode $(( $($btyp),* ))*] items [$($( #[$imeta:meta] )* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )* $bitem: $bmode [$( $bvar:$btyp ),*] {} ] buf [ #[$qmeta] ] queue [$( $tail )*]); }; // Add tuple enum-variant (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )* => $bitem:ident: UNIT [ ] ] queue [($( $qvar:ident: $qtyp:ty ),+) $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*] buf [$( #[$bmeta] )* => $bitem: TUPLE [$( $qvar:$qtyp ),*] ] queue [$( $tail )*] ); }; // Add struct enum-variant - e.g. { descr: &'static str } (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )* => $bitem:ident: UNIT [ ] ] queue [{ $( $qvar:ident: $qtyp:ty ),+} $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*] buf [$( #[$bmeta] )* => $bitem: STRUCT [$( $qvar:$qtyp ),*] ] queue [$( $tail )*]); }; // Add struct enum-variant, with excess comma - e.g. { descr: &'static str, } (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )* => $bitem:ident: UNIT [ ] ] queue [{$( $qvar:ident: $qtyp:ty ),+ ,} $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*] buf [$( #[$bmeta] )* => $bitem: STRUCT [$( $qvar:$qtyp ),*] ] queue [$( $tail )*]); }; // Add braces and flush always on braces (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )* => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ] queue [ {$( $qfuncs:tt )*} $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )* $(#[$bmeta])* => $bitem: $bmode [$( $bvar:$btyp ),*] {$( $qfuncs )*} ] buf [ ] queue [$( $tail )*]); }; // Flush buffer on double ident (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )* => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ] queue [ $qitem:ident $( $tail:tt )*] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )* $(#[$bmeta])* => $bitem: $bmode [$( $bvar:$btyp ),*] {} ] buf [ => $qitem : UNIT [ ] ] queue [$( $tail )*]); }; // Flush buffer on end (SORT [$( $def:tt )*] items [$($( #[$imeta:meta] )* => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*] {$( $ifuncs:tt )*} )* ] buf [$( #[$bmeta:meta] )* => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ] queue [ ] ) => { quick_error!(SORT [$( $def )*] items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )* $(#[$bmeta])* => $bitem: $bmode [$( $bvar:$btyp ),*] {} ] buf [ ] queue [ ]); }; // Public enum (Queue Empty) (ENUM_DEFINITION [pub enum $name:ident $( #[$meta:meta] )*] body [$($( #[$imeta:meta] )* => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ] queue [ ] ) => { #[allow(unknown_lints)] // no unused_doc_comments in older rust #[allow(unused_doc_comment)] $(#[$meta])* pub enum $name { $( $(#[$imeta])* $iitem $(($( $ttyp ),*))* $({$( $svar: $styp ),*})*, )* } }; // Private enum (Queue Empty) (ENUM_DEFINITION [enum $name:ident $( #[$meta:meta] )*] body [$($( #[$imeta:meta] )* => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ] queue [ ] ) => { #[allow(unknown_lints)] // no unused_doc_comments in older rust #[allow(unused_doc_comment)] $(#[$meta])* enum $name { $( $(#[$imeta])* $iitem $(($( $ttyp ),*))* $({$( $svar: $styp ),*})*, )* } }; // Unit variant (ENUM_DEFINITION [$( $def:tt )*] body [$($( #[$imeta:meta] )* => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ] queue [$( #[$qmeta:meta] )* => $qitem:ident: UNIT [ ] $( $queue:tt )*] ) => { quick_error!(ENUM_DEFINITION [ $($def)* ] body [$($( #[$imeta] )* => $iitem ($(($( $ttyp ),+))*) {$({$( $svar: $styp ),*})*} )* $( #[$qmeta] )* => $qitem () {} ] queue [ $($queue)* ] ); }; // Tuple variant (ENUM_DEFINITION [$( $def:tt )*] body [$($( #[$imeta:meta] )* => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ] queue [$( #[$qmeta:meta] )* => $qitem:ident: TUPLE [$( $qvar:ident: $qtyp:ty ),+] $( $queue:tt )*] ) => { quick_error!(ENUM_DEFINITION [ $($def)* ] body [$($( #[$imeta] )* => $iitem ($(($( $ttyp ),+))*) {$({$( $svar: $styp ),*})*} )* $( #[$qmeta] )* => $qitem (($( $qtyp ),*)) {} ] queue [ $($queue)* ] ); }; // Struct variant (ENUM_DEFINITION [$( $def:tt )*] body [$($( #[$imeta:meta] )* => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ] queue [$( #[$qmeta:meta] )* => $qitem:ident: STRUCT [$( $qvar:ident: $qtyp:ty ),*] $( $queue:tt )*] ) => { quick_error!(ENUM_DEFINITION [ $($def)* ] body [$($( #[$imeta] )* => $iitem ($(($( $ttyp ),+))*) {$({$( $svar: $styp ),*})*} )* $( #[$qmeta] )* => $qitem () {{$( $qvar: $qtyp ),*}} ] queue [ $($queue)* ] ); }; (IMPLEMENTATIONS $name:ident {$( $item:ident: $imode:tt [$(#[$imeta:meta])*] [$( $var:ident: $typ:ty ),*] {$( $funcs:tt )*} )*} ) => { #[allow(unused)] #[allow(unknown_lints)] // no unused_doc_comments in older rust #[allow(unused_doc_comment)] impl ::std::fmt::Display for $name { fn fmt(&self, fmt: &mut ::std::fmt::Formatter) -> ::std::fmt::Result { match *self { $( $(#[$imeta])* quick_error!(ITEM_PATTERN $name $item: $imode [$( ref $var ),*] ) => { let display_fn = quick_error!(FIND_DISPLAY_IMPL $name $item: $imode {$( $funcs )*}); display_fn(self, fmt) } )* } } } #[allow(unused)] #[allow(unknown_lints)] // no unused_doc_comments in older rust #[allow(unused_doc_comment)] impl ::std::error::Error for $name { fn description(&self) -> &str { match *self { $( $(#[$imeta])* quick_error!(ITEM_PATTERN $name $item: $imode [$( ref $var ),*] ) => { quick_error!(FIND_DESCRIPTION_IMPL $item: $imode self fmt [$( $var ),*] {$( $funcs )*}) } )* } } fn cause(&self) -> Option<&::std::error::Error> { match *self { $( $(#[$imeta])* quick_error!(ITEM_PATTERN $name $item: $imode [$( ref $var ),*] ) => { quick_error!(FIND_CAUSE_IMPL $item: $imode [$( $var ),*] {$( $funcs )*}) } )* } } } $( quick_error!(FIND_FROM_IMPL $name $item: $imode [$( $var:$typ ),*] {$( $funcs )*}); )* $( quick_error!(FIND_CONTEXT_IMPL $name $item: $imode [$( $var:$typ ),*] {$( $funcs )*}); )* }; (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt { display($self_:tt) -> ($( $exprs:tt )*) $( $tail:tt )*} ) => { |quick_error!(IDENT $self_): &$name, f: &mut ::std::fmt::Formatter| { write!(f, $( $exprs )*) } }; (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt { display($pattern:expr) $( $tail:tt )*} ) => { |_, f: &mut ::std::fmt::Formatter| { write!(f, $pattern) } }; (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt { display($pattern:expr, $( $exprs:tt )*) $( $tail:tt )*} ) => { |_, f: &mut ::std::fmt::Formatter| { write!(f, $pattern, $( $exprs )*) } }; (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt { $t:tt $( $tail:tt )*} ) => { quick_error!(FIND_DISPLAY_IMPL $name $item: $imode {$( $tail )*}) }; (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt { } ) => { |self_: &$name, f: &mut ::std::fmt::Formatter| { write!(f, "{}", ::std::error::Error::description(self_)) } }; (FIND_DESCRIPTION_IMPL $item:ident: $imode:tt $me:ident $fmt:ident [$( $var:ident ),*] { description($expr:expr) $( $tail:tt )*} ) => { $expr }; (FIND_DESCRIPTION_IMPL $item:ident: $imode:tt $me:ident $fmt:ident [$( $var:ident ),*] { $t:tt $( $tail:tt )*} ) => { quick_error!(FIND_DESCRIPTION_IMPL $item: $imode $me $fmt [$( $var ),*] {$( $tail )*}) }; (FIND_DESCRIPTION_IMPL $item:ident: $imode:tt $me:ident $fmt:ident [$( $var:ident ),*] { } ) => { stringify!($item) }; (FIND_CAUSE_IMPL $item:ident: $imode:tt [$( $var:ident ),*] { cause($expr:expr) $( $tail:tt )*} ) => { Some($expr) }; (FIND_CAUSE_IMPL $item:ident: $imode:tt [$( $var:ident ),*] { $t:tt $( $tail:tt )*} ) => { quick_error!(FIND_CAUSE_IMPL $item: $imode [$( $var ),*] { $($tail)* }) }; (FIND_CAUSE_IMPL $item:ident: $imode:tt [$( $var:ident ),*] { } ) => { None }; // ----------------------------- FROM IMPL -------------------------- (FIND_FROM_IMPL $name:ident $item:ident: $imode:tt [$( $var:ident: $typ:ty ),*] { from() $( $tail:tt )*} ) => { $( impl From<$typ> for $name { fn from($var: $typ) -> $name { $name::$item($var) } } )* quick_error!(FIND_FROM_IMPL $name $item: $imode [$( $var:$typ ),*] {$( $tail )*}); }; (FIND_FROM_IMPL $name:ident $item:ident: UNIT [ ] { from($ftyp:ty) $( $tail:tt )*} ) => { impl From<$ftyp> for $name { fn from(_discarded_error: $ftyp) -> $name { $name::$item } } quick_error!(FIND_FROM_IMPL $name $item: UNIT [ ] {$( $tail )*}); }; (FIND_FROM_IMPL $name:ident $item:ident: TUPLE [$( $var:ident: $typ:ty ),*] { from($fvar:ident: $ftyp:ty) -> ($( $texpr:expr ),*) $( $tail:tt )*} ) => { impl From<$ftyp> for $name { fn from($fvar: $ftyp) -> $name { $name::$item($( $texpr ),*) } } quick_error!(FIND_FROM_IMPL $name $item: TUPLE [$( $var:$typ ),*] { $($tail)* }); }; (FIND_FROM_IMPL $name:ident $item:ident: STRUCT [$( $var:ident: $typ:ty ),*] { from($fvar:ident: $ftyp:ty) -> {$( $tvar:ident: $texpr:expr ),*} $( $tail:tt )*} ) => { impl From<$ftyp> for $name { fn from($fvar: $ftyp) -> $name { $name::$item { $( $tvar: $texpr ),* } } } quick_error!(FIND_FROM_IMPL $name $item: STRUCT [$( $var:$typ ),*] { $($tail)* }); }; (FIND_FROM_IMPL $name:ident $item:ident: $imode:tt [$( $var:ident: $typ:ty ),*] { $t:tt $( $tail:tt )*} ) => { quick_error!(FIND_FROM_IMPL $name $item: $imode [$( $var:$typ ),*] {$( $tail )*} ); }; (FIND_FROM_IMPL $name:ident $item:ident: $imode:tt [$( $var:ident: $typ:ty ),*] { } ) => { }; // ----------------------------- CONTEXT IMPL -------------------------- (FIND_CONTEXT_IMPL $name:ident $item:ident: TUPLE [$( $var:ident: $typ:ty ),*] { context($cvar:ident: AsRef<$ctyp:ty>, $fvar:ident: $ftyp:ty) -> ($( $texpr:expr ),*) $( $tail:tt )* } ) => { impl> From<$crate::Context> for $name { fn from( $crate::Context($cvar, $fvar): $crate::Context) -> $name { $name::$item($( $texpr ),*) } } quick_error!(FIND_CONTEXT_IMPL $name $item: TUPLE [$( $var:$typ ),*] { $($tail)* }); }; (FIND_CONTEXT_IMPL $name:ident $item:ident: TUPLE [$( $var:ident: $typ:ty ),*] { context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty) -> ($( $texpr:expr ),*) $( $tail:tt )* } ) => { impl<'a> From<$crate::Context<$ctyp, $ftyp>> for $name { fn from( $crate::Context($cvar, $fvar): $crate::Context<$ctyp, $ftyp>) -> $name { $name::$item($( $texpr ),*) } } quick_error!(FIND_CONTEXT_IMPL $name $item: TUPLE [$( $var:$typ ),*] { $($tail)* }); }; (FIND_CONTEXT_IMPL $name:ident $item:ident: STRUCT [$( $var:ident: $typ:ty ),*] { context($cvar:ident: AsRef<$ctyp:ty>, $fvar:ident: $ftyp:ty) -> {$( $tvar:ident: $texpr:expr ),*} $( $tail:tt )* } ) => { impl> From<$crate::Context> for $name { fn from( $crate::Context($cvar, $fvar): $crate::Context<$ctyp, $ftyp>) -> $name { $name::$item { $( $tvar: $texpr ),* } } } quick_error!(FIND_CONTEXT_IMPL $name $item: STRUCT [$( $var:$typ ),*] { $($tail)* }); }; (FIND_CONTEXT_IMPL $name:ident $item:ident: STRUCT [$( $var:ident: $typ:ty ),*] { context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty) -> {$( $tvar:ident: $texpr:expr ),*} $( $tail:tt )* } ) => { impl<'a> From<$crate::Context<$ctyp, $ftyp>> for $name { fn from( $crate::Context($cvar, $fvar): $crate::Context<$ctyp, $ftyp>) -> $name { $name::$item { $( $tvar: $texpr ),* } } } quick_error!(FIND_CONTEXT_IMPL $name $item: STRUCT [$( $var:$typ ),*] { $($tail)* }); }; (FIND_CONTEXT_IMPL $name:ident $item:ident: $imode:tt [$( $var:ident: $typ:ty ),*] { $t:tt $( $tail:tt )*} ) => { quick_error!(FIND_CONTEXT_IMPL $name $item: $imode [$( $var:$typ ),*] {$( $tail )*} ); }; (FIND_CONTEXT_IMPL $name:ident $item:ident: $imode:tt [$( $var:ident: $typ:ty ),*] { } ) => { }; // ----------------------------- ITEM IMPL -------------------------- (ITEM_BODY $(#[$imeta:meta])* $item:ident: UNIT ) => { }; (ITEM_BODY $(#[$imeta:meta])* $item:ident: TUPLE [$( $typ:ty ),*] ) => { ($( $typ ),*) }; (ITEM_BODY $(#[$imeta:meta])* $item:ident: STRUCT [$( $var:ident: $typ:ty ),*] ) => { {$( $var:$typ ),*} }; (ITEM_PATTERN $name:ident $item:ident: UNIT [] ) => { $name::$item }; (ITEM_PATTERN $name:ident $item:ident: TUPLE [$( ref $var:ident ),*] ) => { $name::$item ($( ref $var ),*) }; (ITEM_PATTERN $name:ident $item:ident: STRUCT [$( ref $var:ident ),*] ) => { $name::$item {$( ref $var ),*} }; // This one should match all allowed sequences in "funcs" but not match // anything else. // This is to contrast FIND_* clauses which just find stuff they need and // skip everything else completely (ERROR_CHECK $imode:tt display($self_:tt) -> ($( $exprs:tt )*) $( $tail:tt )*) => { quick_error!(ERROR_CHECK $imode $($tail)*); }; (ERROR_CHECK $imode:tt display($pattern: expr) $( $tail:tt )*) => { quick_error!(ERROR_CHECK $imode $($tail)*); }; (ERROR_CHECK $imode:tt display($pattern: expr, $( $exprs:tt )*) $( $tail:tt )*) => { quick_error!(ERROR_CHECK $imode $($tail)*); }; (ERROR_CHECK $imode:tt description($expr:expr) $( $tail:tt )*) => { quick_error!(ERROR_CHECK $imode $($tail)*); }; (ERROR_CHECK $imode:tt cause($expr:expr) $($tail:tt)*) => { quick_error!(ERROR_CHECK $imode $($tail)*); }; (ERROR_CHECK $imode:tt from() $($tail:tt)*) => { quick_error!(ERROR_CHECK $imode $($tail)*); }; (ERROR_CHECK $imode:tt from($ftyp:ty) $($tail:tt)*) => { quick_error!(ERROR_CHECK $imode $($tail)*); }; (ERROR_CHECK TUPLE from($fvar:ident: $ftyp:ty) -> ($( $e:expr ),*) $( $tail:tt )*) => { quick_error!(ERROR_CHECK TUPLE $($tail)*); }; (ERROR_CHECK STRUCT from($fvar:ident: $ftyp:ty) -> {$( $v:ident: $e:expr ),*} $( $tail:tt )*) => { quick_error!(ERROR_CHECK STRUCT $($tail)*); }; (ERROR_CHECK TUPLE context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty) -> ($( $e:expr ),*) $( $tail:tt )*) => { quick_error!(ERROR_CHECK TUPLE $($tail)*); }; (ERROR_CHECK STRUCT context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty) -> {$( $v:ident: $e:expr ),*} $( $tail:tt )*) => { quick_error!(ERROR_CHECK STRUCT $($tail)*); }; (ERROR_CHECK $imode:tt ) => {}; // Utility functions (IDENT $ident:ident) => { $ident } } /// Generic context type /// /// Used mostly as a transport for `ResultExt::context` method #[derive(Debug)] pub struct Context(pub X, pub E); /// Result extension trait adding a `context` method pub trait ResultExt { /// The method is use to add context information to current operation /// /// The context data is then used in error constructor to store additional /// information within error. For example, you may add a filename as a /// context for file operation. See crate documentation for the actual /// example. fn context(self, x: X) -> Result>; } impl ResultExt for Result { fn context(self, x: X) -> Result> { self.map_err(|e| Context(x, e)) } } #[cfg(test)] mod test { use std::num::{ParseFloatError, ParseIntError}; use std::str::Utf8Error; use std::string::FromUtf8Error; use std::error::Error; use std::path::{Path, PathBuf}; use super::ResultExt; quick_error! { #[derive(Debug)] pub enum Bare { One Two } } #[test] fn bare_item_direct() { assert_eq!(format!("{}", Bare::One), "One".to_string()); assert_eq!(format!("{:?}", Bare::One), "One".to_string()); assert_eq!(Bare::One.description(), "One".to_string()); assert!(Bare::One.cause().is_none()); } #[test] fn bare_item_trait() { let err: &Error = &Bare::Two; assert_eq!(format!("{}", err), "Two".to_string()); assert_eq!(format!("{:?}", err), "Two".to_string()); assert_eq!(err.description(), "Two".to_string()); assert!(err.cause().is_none()); } quick_error! { #[derive(Debug)] pub enum Wrapper wraps Wrapped { One Two(s: String) { display("two: {}", s) from() } } } #[test] fn wrapper() { assert_eq!(format!("{}", Wrapper::from(Wrapped::One)), "One".to_string()); assert_eq!(format!("{}", Wrapper::from(Wrapped::from(String::from("hello")))), "two: hello".to_string()); assert_eq!(format!("{:?}", Wrapper::from(Wrapped::One)), "Wrapper(One)".to_string()); assert_eq!(Wrapper::from(Wrapped::One).description(), "One".to_string()); } quick_error! { #[derive(Debug, PartialEq)] pub enum TupleWrapper { /// ParseFloat Error ParseFloatError(err: ParseFloatError) { from() description(err.description()) display("parse float error: {err}", err=err) cause(err) } Other(descr: &'static str) { description(descr) display("Error: {}", descr) } /// FromUtf8 Error FromUtf8Error(err: Utf8Error, source: Vec) { cause(err) display(me) -> ("{desc} at index {pos}: {err}", desc=me.description(), pos=err.valid_up_to(), err=err) description("utf8 error") from(err: FromUtf8Error) -> (err.utf8_error().clone(), err.into_bytes()) } Discard { from(&'static str) } Singleton { display("Just a string") } } } #[test] fn tuple_wrapper_err() { let cause = "one and a half times pi".parse::().unwrap_err(); let err = TupleWrapper::ParseFloatError(cause.clone()); assert_eq!(format!("{}", err), format!("parse float error: {}", cause)); assert_eq!(format!("{:?}", err), format!("ParseFloatError({:?})", cause)); assert_eq!(err.description(), cause.description()); assert_eq!(format!("{:?}", err.cause().unwrap()), format!("{:?}", cause)); } #[test] fn tuple_wrapper_trait_str() { let desc = "hello"; let err: &Error = &TupleWrapper::Other(desc); assert_eq!(format!("{}", err), format!("Error: {}", desc)); assert_eq!(format!("{:?}", err), format!("Other({:?})", desc)); assert_eq!(err.description(), desc); assert!(err.cause().is_none()); } #[test] fn tuple_wrapper_trait_two_fields() { let invalid_utf8: Vec = vec![0, 159, 146, 150]; let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err().utf8_error(); let err: &Error = &TupleWrapper::FromUtf8Error(cause.clone(), invalid_utf8.clone()); assert_eq!(format!("{}", err), format!("{desc} at index {pos}: {cause}", desc=err.description(), pos=cause.valid_up_to(), cause=cause)); assert_eq!(format!("{:?}", err), format!("FromUtf8Error({:?}, {:?})", cause, invalid_utf8)); assert_eq!(err.description(), "utf8 error"); assert_eq!(format!("{:?}", err.cause().unwrap()), format!("{:?}", cause)); } #[test] fn tuple_wrapper_from() { let cause = "one and a half times pi".parse::().unwrap_err(); let err = TupleWrapper::ParseFloatError(cause.clone()); let err_from: TupleWrapper = From::from(cause); assert_eq!(err_from, err); } #[test] fn tuple_wrapper_custom_from() { let invalid_utf8: Vec = vec![0, 159, 146, 150]; let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err(); let err = TupleWrapper::FromUtf8Error(cause.utf8_error().clone(), invalid_utf8); let err_from: TupleWrapper = From::from(cause); assert_eq!(err_from, err); } #[test] fn tuple_wrapper_discard() { let err: TupleWrapper = From::from("hello"); assert_eq!(format!("{}", err), format!("Discard")); assert_eq!(format!("{:?}", err), format!("Discard")); assert_eq!(err.description(), "Discard"); assert!(err.cause().is_none()); } #[test] fn tuple_wrapper_singleton() { let err: TupleWrapper = TupleWrapper::Singleton; assert_eq!(format!("{}", err), format!("Just a string")); assert_eq!(format!("{:?}", err), format!("Singleton")); assert_eq!(err.description(), "Singleton"); assert!(err.cause().is_none()); } quick_error! { #[derive(Debug, PartialEq)] pub enum StructWrapper { // Utf8 Error Utf8Error{ err: Utf8Error, hint: Option<&'static str> } { cause(err) display(me) -> ("{desc} at index {pos}: {err}", desc=me.description(), pos=err.valid_up_to(), err=err) description("utf8 error") from(err: Utf8Error) -> { err: err, hint: None } } // Utf8 Error ExcessComma { descr: &'static str, } { description(descr) display("Error: {}", descr) } } } #[test] fn struct_wrapper_err() { let invalid_utf8: Vec = vec![0, 159, 146, 150]; let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err().utf8_error(); let err: &Error = &StructWrapper::Utf8Error{ err: cause.clone(), hint: Some("nonsense") }; assert_eq!(format!("{}", err), format!("{desc} at index {pos}: {cause}", desc=err.description(), pos=cause.valid_up_to(), cause=cause)); assert_eq!(format!("{:?}", err), format!("Utf8Error {{ err: {:?}, hint: {:?} }}", cause, Some("nonsense"))); assert_eq!(err.description(), "utf8 error"); assert_eq!(format!("{:?}", err.cause().unwrap()), format!("{:?}", cause)); } #[test] fn struct_wrapper_struct_from() { let invalid_utf8: Vec = vec![0, 159, 146, 150]; let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err().utf8_error(); let err = StructWrapper::Utf8Error{ err: cause.clone(), hint: None }; let err_from: StructWrapper = From::from(cause); assert_eq!(err_from, err); } #[test] fn struct_wrapper_excess_comma() { let descr = "hello"; let err = StructWrapper::ExcessComma { descr: descr }; assert_eq!(format!("{}", err), format!("Error: {}", descr)); assert_eq!(format!("{:?}", err), format!("ExcessComma {{ descr: {:?} }}", descr)); assert_eq!(err.description(), descr); assert!(err.cause().is_none()); } quick_error! { #[derive(Debug)] pub enum ContextErr { Float(src: String, err: ParseFloatError) { context(s: &'a str, e: ParseFloatError) -> (s.to_string(), e) display("Float error {:?}: {}", src, err) } Int { src: String, err: ParseIntError } { context(s: &'a str, e: ParseIntError) -> {src: s.to_string(), err: e} display("Int error {:?}: {}", src, err) } Utf8(path: PathBuf, err: Utf8Error) { context(p: AsRef, e: Utf8Error) -> (p.as_ref().to_path_buf(), e) display("Path error at {:?}: {}", path, err) } Utf8Str(s: String, err: ::std::io::Error) { context(s: AsRef, e: ::std::io::Error) -> (s.as_ref().to_string(), e) display("Str error {:?}: {}", s, err) } } } #[test] fn parse_float_error() { fn parse_float(s: &str) -> Result { Ok(try!(s.parse().context(s))) } assert_eq!(format!("{}", parse_float("12ab").unwrap_err()), r#"Float error "12ab": invalid float literal"#); } #[test] fn parse_int_error() { fn parse_int(s: &str) -> Result { Ok(try!(s.parse().context(s))) } assert_eq!(format!("{}", parse_int("12.5").unwrap_err()), r#"Int error "12.5": invalid digit found in string"#); } #[test] fn debug_context() { fn parse_int(s: &str) -> i32 { s.parse().context(s).unwrap() } assert_eq!(parse_int("12"), 12); assert_eq!(format!("{:?}", "x".parse::().context("x")), r#"Err(Context("x", ParseIntError { kind: InvalidDigit }))"#); } #[test] fn path_context() { fn parse_utf>(s: &[u8], p: P) -> Result<(), ContextErr> { try!(::std::str::from_utf8(s).context(p)); Ok(()) } let etext = parse_utf(b"a\x80\x80", "/etc").unwrap_err().to_string(); assert!(etext.starts_with( "Path error at \"/etc\": invalid utf-8")); let etext = parse_utf(b"\x80\x80", PathBuf::from("/tmp")).unwrap_err() .to_string(); assert!(etext.starts_with( "Path error at \"/tmp\": invalid utf-8")); } #[test] fn conditional_compilation() { quick_error! { #[allow(dead_code)] #[derive(Debug)] pub enum Test { #[cfg(feature = "foo")] Variant } } } }