1#![warn(missing_docs)]
2#[macro_export]
307macro_rules! quick_error {
308
309 ( $(#[$meta:meta])*
310 pub enum $name:ident { $($chunks:tt)* }
311 ) => {
312 quick_error!(SORT [pub enum $name $(#[$meta])* ]
313 items [] buf []
314 queue [ $($chunks)* ]);
315 };
316 ( $(#[$meta:meta])*
317 enum $name:ident { $($chunks:tt)* }
318 ) => {
319 quick_error!(SORT [enum $name $(#[$meta])* ]
320 items [] buf []
321 queue [ $($chunks)* ]);
322 };
323
324 ( $(#[$meta:meta])*
325 pub enum $name:ident wraps $enum_name:ident { $($chunks:tt)* }
326 ) => {
327 quick_error!(WRAPPER $enum_name [ pub struct ] $name $(#[$meta])*);
328 quick_error!(SORT [enum $enum_name $(#[$meta])* ]
329 items [] buf []
330 queue [ $($chunks)* ]);
331 };
332
333 ( $(#[$meta:meta])*
334 pub enum $name:ident wraps pub $enum_name:ident { $($chunks:tt)* }
335 ) => {
336 quick_error!(WRAPPER $enum_name [ pub struct ] $name $(#[$meta])*);
337 quick_error!(SORT [pub enum $enum_name $(#[$meta])* ]
338 items [] buf []
339 queue [ $($chunks)* ]);
340 };
341 ( $(#[$meta:meta])*
342 enum $name:ident wraps $enum_name:ident { $($chunks:tt)* }
343 ) => {
344 quick_error!(WRAPPER $enum_name [ struct ] $name $(#[$meta])*);
345 quick_error!(SORT [enum $enum_name $(#[$meta])* ]
346 items [] buf []
347 queue [ $($chunks)* ]);
348 };
349
350 ( $(#[$meta:meta])*
351 enum $name:ident wraps pub $enum_name:ident { $($chunks:tt)* }
352 ) => {
353 quick_error!(WRAPPER $enum_name [ struct ] $name $(#[$meta])*);
354 quick_error!(SORT [pub enum $enum_name $(#[$meta])* ]
355 items [] buf []
356 queue [ $($chunks)* ]);
357 };
358
359
360 (
361 WRAPPER $internal:ident [ $($strdef:tt)* ] $strname:ident
362 $(#[$meta:meta])*
363 ) => {
364 $(#[$meta])*
365 $($strdef)* $strname ( $internal );
366
367 impl ::std::fmt::Display for $strname {
368 fn fmt(&self, f: &mut ::std::fmt::Formatter)
369 -> ::std::fmt::Result
370 {
371 ::std::fmt::Display::fmt(&self.0, f)
372 }
373 }
374
375 impl From<$internal> for $strname {
376 fn from(err: $internal) -> Self {
377 $strname(err)
378 }
379 }
380
381 impl ::std::error::Error for $strname {
382 fn description(&self) -> &str {
383 self.0.description()
384 }
385 fn cause(&self) -> Option<&::std::error::Error> {
386 self.0.cause()
387 }
388 }
389 };
390
391 (SORT [enum $name:ident $( #[$meta:meta] )*]
393 items [$($( #[$imeta:meta] )*
394 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
395 {$( $ifuncs:tt )*} )* ]
396 buf [ ]
397 queue [ ]
398 ) => {
399 quick_error!(ENUM_DEFINITION [enum $name $( #[$meta] )*]
400 body []
401 queue [$($( #[$imeta] )*
402 => $iitem: $imode [$( $ivar: $ityp ),*] )*]
403 );
404 quick_error!(IMPLEMENTATIONS $name {$(
405 $iitem: $imode [$(#[$imeta])*] [$( $ivar: $ityp ),*] {$( $ifuncs )*}
406 )*});
407 $(
408 quick_error!(ERROR_CHECK $imode $($ifuncs)*);
409 )*
410 };
411 (SORT [pub enum $name:ident $( #[$meta:meta] )*]
412 items [$($( #[$imeta:meta] )*
413 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
414 {$( $ifuncs:tt )*} )* ]
415 buf [ ]
416 queue [ ]
417 ) => {
418 quick_error!(ENUM_DEFINITION [pub enum $name $( #[$meta] )*]
419 body []
420 queue [$($( #[$imeta] )*
421 => $iitem: $imode [$( $ivar: $ityp ),*] )*]
422 );
423 quick_error!(IMPLEMENTATIONS $name {$(
424 $iitem: $imode [$(#[$imeta])*] [$( $ivar: $ityp ),*] {$( $ifuncs )*}
425 )*});
426 $(
427 quick_error!(ERROR_CHECK $imode $($ifuncs)*);
428 )*
429 };
430 (SORT [$( $def:tt )*]
432 items [$($( #[$imeta:meta] )*
433 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
434 {$( $ifuncs:tt )*} )* ]
435 buf [$( #[$bmeta:meta] )*]
436 queue [ #[$qmeta:meta] $( $tail:tt )*]
437 ) => {
438 quick_error!(SORT [$( $def )*]
439 items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*]
440 buf [$( #[$bmeta] )* #[$qmeta] ]
441 queue [$( $tail )*]);
442 };
443 (SORT [$( $def:tt )*]
445 items [$($( #[$imeta:meta] )*
446 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
447 {$( $ifuncs:tt )*} )* ]
448 buf [$( #[$bmeta:meta] )*]
449 queue [ $qitem:ident $( $tail:tt )*]
450 ) => {
451 quick_error!(SORT [$( $def )*]
452 items [$( $(#[$imeta])*
453 => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*]
454 buf [$(#[$bmeta])* => $qitem : UNIT [ ] ]
455 queue [$( $tail )*]);
456 };
457 (SORT [$( $def:tt )*]
459 items [$($( #[$imeta:meta] )*
460 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
461 {$( $ifuncs:tt )*} )* ]
462 buf [$( #[$bmeta:meta] )*
463 => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ]
464 queue [ #[$qmeta:meta] $( $tail:tt )*]
465 ) => {
466 quick_error!(SORT [$( $def )*]
467 enum [$( $(#[$emeta])* => $eitem $(( $($etyp),* ))* )*
468 $(#[$bmeta])* => $bitem: $bmode $(( $($btyp),* ))*]
469 items [$($( #[$imeta:meta] )*
470 => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*
471 $bitem: $bmode [$( $bvar:$btyp ),*] {} ]
472 buf [ #[$qmeta] ]
473 queue [$( $tail )*]);
474 };
475 (SORT [$( $def:tt )*]
477 items [$($( #[$imeta:meta] )*
478 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
479 {$( $ifuncs:tt )*} )* ]
480 buf [$( #[$bmeta:meta] )* => $bitem:ident: UNIT [ ] ]
481 queue [($( $qvar:ident: $qtyp:ty ),+) $( $tail:tt )*]
482 ) => {
483 quick_error!(SORT [$( $def )*]
484 items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*]
485 buf [$( #[$bmeta] )* => $bitem: TUPLE [$( $qvar:$qtyp ),*] ]
486 queue [$( $tail )*]
487 );
488 };
489 (SORT [$( $def:tt )*]
491 items [$($( #[$imeta:meta] )*
492 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
493 {$( $ifuncs:tt )*} )* ]
494 buf [$( #[$bmeta:meta] )* => $bitem:ident: UNIT [ ] ]
495 queue [{ $( $qvar:ident: $qtyp:ty ),+} $( $tail:tt )*]
496 ) => {
497 quick_error!(SORT [$( $def )*]
498 items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*]
499 buf [$( #[$bmeta] )* => $bitem: STRUCT [$( $qvar:$qtyp ),*] ]
500 queue [$( $tail )*]);
501 };
502 (SORT [$( $def:tt )*]
504 items [$($( #[$imeta:meta] )*
505 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
506 {$( $ifuncs:tt )*} )* ]
507 buf [$( #[$bmeta:meta] )* => $bitem:ident: UNIT [ ] ]
508 queue [{$( $qvar:ident: $qtyp:ty ),+ ,} $( $tail:tt )*]
509 ) => {
510 quick_error!(SORT [$( $def )*]
511 items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*]
512 buf [$( #[$bmeta] )* => $bitem: STRUCT [$( $qvar:$qtyp ),*] ]
513 queue [$( $tail )*]);
514 };
515 (SORT [$( $def:tt )*]
517 items [$($( #[$imeta:meta] )*
518 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
519 {$( $ifuncs:tt )*} )* ]
520 buf [$( #[$bmeta:meta] )*
521 => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ]
522 queue [ {$( $qfuncs:tt )*} $( $tail:tt )*]
523 ) => {
524 quick_error!(SORT [$( $def )*]
525 items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*
526 $(#[$bmeta])* => $bitem: $bmode [$( $bvar:$btyp ),*] {$( $qfuncs )*} ]
527 buf [ ]
528 queue [$( $tail )*]);
529 };
530 (SORT [$( $def:tt )*]
532 items [$($( #[$imeta:meta] )*
533 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
534 {$( $ifuncs:tt )*} )* ]
535 buf [$( #[$bmeta:meta] )*
536 => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ]
537 queue [ $qitem:ident $( $tail:tt )*]
538 ) => {
539 quick_error!(SORT [$( $def )*]
540 items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*
541 $(#[$bmeta])* => $bitem: $bmode [$( $bvar:$btyp ),*] {} ]
542 buf [ => $qitem : UNIT [ ] ]
543 queue [$( $tail )*]);
544 };
545 (SORT [$( $def:tt )*]
547 items [$($( #[$imeta:meta] )*
548 => $iitem:ident: $imode:tt [$( $ivar:ident: $ityp:ty ),*]
549 {$( $ifuncs:tt )*} )* ]
550 buf [$( #[$bmeta:meta] )*
551 => $bitem:ident: $bmode:tt [$( $bvar:ident: $btyp:ty ),*] ]
552 queue [ ]
553 ) => {
554 quick_error!(SORT [$( $def )*]
555 items [$( $(#[$imeta])* => $iitem: $imode [$( $ivar:$ityp ),*] {$( $ifuncs )*} )*
556 $(#[$bmeta])* => $bitem: $bmode [$( $bvar:$btyp ),*] {} ]
557 buf [ ]
558 queue [ ]);
559 };
560 (ENUM_DEFINITION [pub enum $name:ident $( #[$meta:meta] )*]
562 body [$($( #[$imeta:meta] )*
563 => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ]
564 queue [ ]
565 ) => {
566 $(#[$meta])*
567 pub enum $name {
568 $(
569 $(#[$imeta])*
570 $iitem $(($( $ttyp ),*))* $({$( $svar: $styp ),*})*,
571 )*
572 }
573 };
574 (ENUM_DEFINITION [enum $name:ident $( #[$meta:meta] )*]
576 body [$($( #[$imeta:meta] )*
577 => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ]
578 queue [ ]
579 ) => {
580 $(#[$meta])*
581 enum $name {
582 $(
583 $(#[$imeta])*
584 $iitem $(($( $ttyp ),*))* $({$( $svar: $styp ),*})*,
585 )*
586 }
587 };
588 (ENUM_DEFINITION [$( $def:tt )*]
590 body [$($( #[$imeta:meta] )*
591 => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ]
592 queue [$( #[$qmeta:meta] )*
593 => $qitem:ident: UNIT [ ] $( $queue:tt )*]
594 ) => {
595 quick_error!(ENUM_DEFINITION [ $($def)* ]
596 body [$($( #[$imeta] )* => $iitem ($(($( $ttyp ),+))*) {$({$( $svar: $styp ),*})*} )*
597 $( #[$qmeta] )* => $qitem () {} ]
598 queue [ $($queue)* ]
599 );
600 };
601 (ENUM_DEFINITION [$( $def:tt )*]
603 body [$($( #[$imeta:meta] )*
604 => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ]
605 queue [$( #[$qmeta:meta] )*
606 => $qitem:ident: TUPLE [$( $qvar:ident: $qtyp:ty ),+] $( $queue:tt )*]
607 ) => {
608 quick_error!(ENUM_DEFINITION [ $($def)* ]
609 body [$($( #[$imeta] )* => $iitem ($(($( $ttyp ),+))*) {$({$( $svar: $styp ),*})*} )*
610 $( #[$qmeta] )* => $qitem (($( $qtyp ),*)) {} ]
611 queue [ $($queue)* ]
612 );
613 };
614 (ENUM_DEFINITION [$( $def:tt )*]
616 body [$($( #[$imeta:meta] )*
617 => $iitem:ident ($(($( $ttyp:ty ),+))*) {$({$( $svar:ident: $styp:ty ),*})*} )* ]
618 queue [$( #[$qmeta:meta] )*
619 => $qitem:ident: STRUCT [$( $qvar:ident: $qtyp:ty ),*] $( $queue:tt )*]
620 ) => {
621 quick_error!(ENUM_DEFINITION [ $($def)* ]
622 body [$($( #[$imeta] )* => $iitem ($(($( $ttyp ),+))*) {$({$( $svar: $styp ),*})*} )*
623 $( #[$qmeta] )* => $qitem () {{$( $qvar: $qtyp ),*}} ]
624 queue [ $($queue)* ]
625 );
626 };
627 (IMPLEMENTATIONS
628 $name:ident {$(
629 $item:ident: $imode:tt [$(#[$imeta:meta])*] [$( $var:ident: $typ:ty ),*] {$( $funcs:tt )*}
630 )*}
631 ) => {
632 #[allow(unused)]
633 impl ::std::fmt::Display for $name {
634 fn fmt(&self, fmt: &mut ::std::fmt::Formatter)
635 -> ::std::fmt::Result
636 {
637 match *self {
638 $(
639 $(#[$imeta])*
640 quick_error!(ITEM_PATTERN
641 $name $item: $imode [$( ref $var ),*]
642 ) => {
643 let display_fn = quick_error!(FIND_DISPLAY_IMPL
644 $name $item: $imode
645 {$( $funcs )*});
646
647 display_fn(self, fmt)
648 }
649 )*
650 }
651 }
652 }
653 #[allow(unused)]
654 impl ::std::error::Error for $name {
655 fn description(&self) -> &str {
656 match *self {
657 $(
658 $(#[$imeta])*
659 quick_error!(ITEM_PATTERN
660 $name $item: $imode [$( ref $var ),*]
661 ) => {
662 quick_error!(FIND_DESCRIPTION_IMPL
663 $item: $imode self fmt [$( $var ),*]
664 {$( $funcs )*})
665 }
666 )*
667 }
668 }
669 fn cause(&self) -> Option<&::std::error::Error> {
670 match *self {
671 $(
672 $(#[$imeta])*
673 quick_error!(ITEM_PATTERN
674 $name $item: $imode [$( ref $var ),*]
675 ) => {
676 quick_error!(FIND_CAUSE_IMPL
677 $item: $imode [$( $var ),*]
678 {$( $funcs )*})
679 }
680 )*
681 }
682 }
683 }
684 $(
685 quick_error!(FIND_FROM_IMPL
686 $name $item: $imode [$( $var:$typ ),*]
687 {$( $funcs )*});
688 )*
689 $(
690 quick_error!(FIND_CONTEXT_IMPL
691 $name $item: $imode [$( $var:$typ ),*]
692 {$( $funcs )*});
693 )*
694 };
695 (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt
696 { display($self_:tt) -> ($( $exprs:tt )*) $( $tail:tt )*}
697 ) => {
698 |quick_error!(IDENT $self_): &$name, f: &mut ::std::fmt::Formatter| { write!(f, $( $exprs )*) }
699 };
700 (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt
701 { display($pattern:expr) $( $tail:tt )*}
702 ) => {
703 |_, f: &mut ::std::fmt::Formatter| { write!(f, $pattern) }
704 };
705 (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt
706 { display($pattern:expr, $( $exprs:tt )*) $( $tail:tt )*}
707 ) => {
708 |_, f: &mut ::std::fmt::Formatter| { write!(f, $pattern, $( $exprs )*) }
709 };
710 (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt
711 { $t:tt $( $tail:tt )*}
712 ) => {
713 quick_error!(FIND_DISPLAY_IMPL
714 $name $item: $imode
715 {$( $tail )*})
716 };
717 (FIND_DISPLAY_IMPL $name:ident $item:ident: $imode:tt
718 { }
719 ) => {
720 |self_: &$name, f: &mut ::std::fmt::Formatter| {
721 write!(f, "{}", ::std::error::Error::description(self_))
722 }
723 };
724 (FIND_DESCRIPTION_IMPL $item:ident: $imode:tt $me:ident $fmt:ident
725 [$( $var:ident ),*]
726 { description($expr:expr) $( $tail:tt )*}
727 ) => {
728 $expr
729 };
730 (FIND_DESCRIPTION_IMPL $item:ident: $imode:tt $me:ident $fmt:ident
731 [$( $var:ident ),*]
732 { $t:tt $( $tail:tt )*}
733 ) => {
734 quick_error!(FIND_DESCRIPTION_IMPL
735 $item: $imode $me $fmt [$( $var ),*]
736 {$( $tail )*})
737 };
738 (FIND_DESCRIPTION_IMPL $item:ident: $imode:tt $me:ident $fmt:ident
739 [$( $var:ident ),*]
740 { }
741 ) => {
742 stringify!($item)
743 };
744 (FIND_CAUSE_IMPL $item:ident: $imode:tt
745 [$( $var:ident ),*]
746 { cause($expr:expr) $( $tail:tt )*}
747 ) => {
748 Some($expr)
749 };
750 (FIND_CAUSE_IMPL $item:ident: $imode:tt
751 [$( $var:ident ),*]
752 { $t:tt $( $tail:tt )*}
753 ) => {
754 quick_error!(FIND_CAUSE_IMPL
755 $item: $imode [$( $var ),*]
756 { $($tail)* })
757 };
758 (FIND_CAUSE_IMPL $item:ident: $imode:tt
759 [$( $var:ident ),*]
760 { }
761 ) => {
762 None
763 };
764 (FIND_FROM_IMPL $name:ident $item:ident: $imode:tt
766 [$( $var:ident: $typ:ty ),*]
767 { from() $( $tail:tt )*}
768 ) => {
769 $(
770 impl From<$typ> for $name {
771 fn from($var: $typ) -> $name {
772 $name::$item($var)
773 }
774 }
775 )*
776 quick_error!(FIND_FROM_IMPL
777 $name $item: $imode [$( $var:$typ ),*]
778 {$( $tail )*});
779 };
780 (FIND_FROM_IMPL $name:ident $item:ident: UNIT
781 [ ]
782 { from($ftyp:ty) $( $tail:tt )*}
783 ) => {
784 impl From<$ftyp> for $name {
785 fn from(_discarded_error: $ftyp) -> $name {
786 $name::$item
787 }
788 }
789 quick_error!(FIND_FROM_IMPL
790 $name $item: UNIT [ ]
791 {$( $tail )*});
792 };
793 (FIND_FROM_IMPL $name:ident $item:ident: TUPLE
794 [$( $var:ident: $typ:ty ),*]
795 { from($fvar:ident: $ftyp:ty) -> ($( $texpr:expr ),*) $( $tail:tt )*}
796 ) => {
797 impl From<$ftyp> for $name {
798 fn from($fvar: $ftyp) -> $name {
799 $name::$item($( $texpr ),*)
800 }
801 }
802 quick_error!(FIND_FROM_IMPL
803 $name $item: TUPLE [$( $var:$typ ),*]
804 { $($tail)* });
805 };
806 (FIND_FROM_IMPL $name:ident $item:ident: STRUCT
807 [$( $var:ident: $typ:ty ),*]
808 { from($fvar:ident: $ftyp:ty) -> {$( $tvar:ident: $texpr:expr ),*} $( $tail:tt )*}
809 ) => {
810 impl From<$ftyp> for $name {
811 fn from($fvar: $ftyp) -> $name {
812 $name::$item {
813 $( $tvar: $texpr ),*
814 }
815 }
816 }
817 quick_error!(FIND_FROM_IMPL
818 $name $item: STRUCT [$( $var:$typ ),*]
819 { $($tail)* });
820 };
821 (FIND_FROM_IMPL $name:ident $item:ident: $imode:tt
822 [$( $var:ident: $typ:ty ),*]
823 { $t:tt $( $tail:tt )*}
824 ) => {
825 quick_error!(FIND_FROM_IMPL
826 $name $item: $imode [$( $var:$typ ),*]
827 {$( $tail )*}
828 );
829 };
830 (FIND_FROM_IMPL $name:ident $item:ident: $imode:tt
831 [$( $var:ident: $typ:ty ),*]
832 { }
833 ) => {
834 };
835 (FIND_CONTEXT_IMPL $name:ident $item:ident: TUPLE
837 [$( $var:ident: $typ:ty ),*]
838 { context($cvar:ident: AsRef<$ctyp:ty>, $fvar:ident: $ftyp:ty)
839 -> ($( $texpr:expr ),*) $( $tail:tt )* }
840 ) => {
841 impl<T: AsRef<$ctyp>> From<$crate::Context<T, $ftyp>> for $name {
842 fn from(
843 $crate::Context($cvar, $fvar): $crate::Context<T, $ftyp>)
844 -> $name
845 {
846 $name::$item($( $texpr ),*)
847 }
848 }
849 quick_error!(FIND_CONTEXT_IMPL
850 $name $item: TUPLE [$( $var:$typ ),*]
851 { $($tail)* });
852 };
853 (FIND_CONTEXT_IMPL $name:ident $item:ident: TUPLE
854 [$( $var:ident: $typ:ty ),*]
855 { context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty)
856 -> ($( $texpr:expr ),*) $( $tail:tt )* }
857 ) => {
858 impl<'a> From<$crate::Context<$ctyp, $ftyp>> for $name {
859 fn from(
860 $crate::Context($cvar, $fvar): $crate::Context<$ctyp, $ftyp>)
861 -> $name
862 {
863 $name::$item($( $texpr ),*)
864 }
865 }
866 quick_error!(FIND_CONTEXT_IMPL
867 $name $item: TUPLE [$( $var:$typ ),*]
868 { $($tail)* });
869 };
870 (FIND_CONTEXT_IMPL $name:ident $item:ident: STRUCT
871 [$( $var:ident: $typ:ty ),*]
872 { context($cvar:ident: AsRef<$ctyp:ty>, $fvar:ident: $ftyp:ty)
873 -> {$( $tvar:ident: $texpr:expr ),*} $( $tail:tt )* }
874 ) => {
875 impl<T: AsRef<$ctyp>> From<$crate::Context<T, $ftyp>> for $name {
876 fn from(
877 $crate::Context($cvar, $fvar): $crate::Context<$ctyp, $ftyp>)
878 -> $name
879 {
880 $name::$item {
881 $( $tvar: $texpr ),*
882 }
883 }
884 }
885 quick_error!(FIND_CONTEXT_IMPL
886 $name $item: STRUCT [$( $var:$typ ),*]
887 { $($tail)* });
888 };
889 (FIND_CONTEXT_IMPL $name:ident $item:ident: STRUCT
890 [$( $var:ident: $typ:ty ),*]
891 { context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty)
892 -> {$( $tvar:ident: $texpr:expr ),*} $( $tail:tt )* }
893 ) => {
894 impl<'a> From<$crate::Context<$ctyp, $ftyp>> for $name {
895 fn from(
896 $crate::Context($cvar, $fvar): $crate::Context<$ctyp, $ftyp>)
897 -> $name
898 {
899 $name::$item {
900 $( $tvar: $texpr ),*
901 }
902 }
903 }
904 quick_error!(FIND_CONTEXT_IMPL
905 $name $item: STRUCT [$( $var:$typ ),*]
906 { $($tail)* });
907 };
908 (FIND_CONTEXT_IMPL $name:ident $item:ident: $imode:tt
909 [$( $var:ident: $typ:ty ),*]
910 { $t:tt $( $tail:tt )*}
911 ) => {
912 quick_error!(FIND_CONTEXT_IMPL
913 $name $item: $imode [$( $var:$typ ),*]
914 {$( $tail )*}
915 );
916 };
917 (FIND_CONTEXT_IMPL $name:ident $item:ident: $imode:tt
918 [$( $var:ident: $typ:ty ),*]
919 { }
920 ) => {
921 };
922 (ITEM_BODY $(#[$imeta:meta])* $item:ident: UNIT
924 ) => { };
925 (ITEM_BODY $(#[$imeta:meta])* $item:ident: TUPLE
926 [$( $typ:ty ),*]
927 ) => {
928 ($( $typ ),*)
929 };
930 (ITEM_BODY $(#[$imeta:meta])* $item:ident: STRUCT
931 [$( $var:ident: $typ:ty ),*]
932 ) => {
933 {$( $var:$typ ),*}
934 };
935 (ITEM_PATTERN $name:ident $item:ident: UNIT []
936 ) => {
937 $name::$item
938 };
939 (ITEM_PATTERN $name:ident $item:ident: TUPLE
940 [$( ref $var:ident ),*]
941 ) => {
942 $name::$item ($( ref $var ),*)
943 };
944 (ITEM_PATTERN $name:ident $item:ident: STRUCT
945 [$( ref $var:ident ),*]
946 ) => {
947 $name::$item {$( ref $var ),*}
948 };
949 (ERROR_CHECK $imode:tt display($self_:tt) -> ($( $exprs:tt )*) $( $tail:tt )*)
954 => { quick_error!(ERROR_CHECK $imode $($tail)*); };
955 (ERROR_CHECK $imode:tt display($pattern: expr) $( $tail:tt )*)
956 => { quick_error!(ERROR_CHECK $imode $($tail)*); };
957 (ERROR_CHECK $imode:tt display($pattern: expr, $( $exprs:tt )*) $( $tail:tt )*)
958 => { quick_error!(ERROR_CHECK $imode $($tail)*); };
959 (ERROR_CHECK $imode:tt description($expr:expr) $( $tail:tt )*)
960 => { quick_error!(ERROR_CHECK $imode $($tail)*); };
961 (ERROR_CHECK $imode:tt cause($expr:expr) $($tail:tt)*)
962 => { quick_error!(ERROR_CHECK $imode $($tail)*); };
963 (ERROR_CHECK $imode:tt from() $($tail:tt)*)
964 => { quick_error!(ERROR_CHECK $imode $($tail)*); };
965 (ERROR_CHECK $imode:tt from($ftyp:ty) $($tail:tt)*)
966 => { quick_error!(ERROR_CHECK $imode $($tail)*); };
967 (ERROR_CHECK TUPLE from($fvar:ident: $ftyp:ty) -> ($( $e:expr ),*) $( $tail:tt )*)
968 => { quick_error!(ERROR_CHECK TUPLE $($tail)*); };
969 (ERROR_CHECK STRUCT from($fvar:ident: $ftyp:ty) -> {$( $v:ident: $e:expr ),*} $( $tail:tt )*)
970 => { quick_error!(ERROR_CHECK STRUCT $($tail)*); };
971
972 (ERROR_CHECK TUPLE context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty)
973 -> ($( $e:expr ),*) $( $tail:tt )*)
974 => { quick_error!(ERROR_CHECK TUPLE $($tail)*); };
975 (ERROR_CHECK STRUCT context($cvar:ident: $ctyp:ty, $fvar:ident: $ftyp:ty)
976 -> {$( $v:ident: $e:expr ),*} $( $tail:tt )*)
977 => { quick_error!(ERROR_CHECK STRUCT $($tail)*); };
978
979 (ERROR_CHECK $imode:tt ) => {};
980 (IDENT $ident:ident) => { $ident }
982}
983
984
985#[derive(Debug)]
989pub struct Context<X, E>(pub X, pub E);
990
991pub trait ResultExt<T, E> {
993 fn context<X>(self, x: X) -> Result<T, Context<X, E>>;
1000}
1001
1002impl<T, E> ResultExt<T, E> for Result<T, E> {
1003 fn context<X>(self, x: X) -> Result<T, Context<X, E>> {
1004 self.map_err(|e| Context(x, e))
1005 }
1006}
1007
1008
1009
1010#[cfg(test)]
1011mod test {
1012 use std::num::{ParseFloatError, ParseIntError};
1013 use std::str::Utf8Error;
1014 use std::string::FromUtf8Error;
1015 use std::error::Error;
1016 use std::path::{Path, PathBuf};
1017
1018 use super::ResultExt;
1019
1020 quick_error! {
1021 #[derive(Debug)]
1022 pub enum Bare {
1023 One
1024 Two
1025 }
1026 }
1027
1028 #[test]
1029 fn bare_item_direct() {
1030 assert_eq!(format!("{}", Bare::One), "One".to_string());
1031 assert_eq!(format!("{:?}", Bare::One), "One".to_string());
1032 assert_eq!(Bare::One.description(), "One".to_string());
1033 assert!(Bare::One.cause().is_none());
1034 }
1035 #[test]
1036 fn bare_item_trait() {
1037 let err: &Error = &Bare::Two;
1038 assert_eq!(format!("{}", err), "Two".to_string());
1039 assert_eq!(format!("{:?}", err), "Two".to_string());
1040 assert_eq!(err.description(), "Two".to_string());
1041 assert!(err.cause().is_none());
1042 }
1043
1044 quick_error! {
1045 #[derive(Debug)]
1046 pub enum Wrapper wraps Wrapped {
1047 One
1048 Two(s: String) {
1049 display("two: {}", s)
1050 from()
1051 }
1052 }
1053 }
1054
1055 #[test]
1056 fn wrapper() {
1057 assert_eq!(format!("{}", Wrapper::from(Wrapped::One)),
1058 "One".to_string());
1059 assert_eq!(format!("{}",
1060 Wrapper::from(Wrapped::from(String::from("hello")))),
1061 "two: hello".to_string());
1062 assert_eq!(format!("{:?}", Wrapper::from(Wrapped::One)),
1063 "Wrapper(One)".to_string());
1064 assert_eq!(Wrapper::from(Wrapped::One).description(),
1065 "One".to_string());
1066 }
1067
1068 quick_error! {
1069 #[derive(Debug, PartialEq)]
1070 pub enum TupleWrapper {
1071 ParseFloatError(err: ParseFloatError) {
1073 from()
1074 description(err.description())
1075 display("parse float error: {err}", err=err)
1076 cause(err)
1077 }
1078 Other(descr: &'static str) {
1079 description(descr)
1080 display("Error: {}", descr)
1081 }
1082 FromUtf8Error(err: Utf8Error, source: Vec<u8>) {
1084 cause(err)
1085 display(me) -> ("{desc} at index {pos}: {err}", desc=me.description(), pos=err.valid_up_to(), err=err)
1086 description("utf8 error")
1087 from(err: FromUtf8Error) -> (err.utf8_error().clone(), err.into_bytes())
1088 }
1089 Discard {
1090 from(&'static str)
1091 }
1092 Singleton {
1093 display("Just a string")
1094 }
1095 }
1096 }
1097
1098 #[test]
1099 fn tuple_wrapper_err() {
1100 let cause = "one and a half times pi".parse::<f32>().unwrap_err();
1101 let err = TupleWrapper::ParseFloatError(cause.clone());
1102 assert_eq!(format!("{}", err), format!("parse float error: {}", cause));
1103 assert_eq!(format!("{:?}", err), format!("ParseFloatError({:?})", cause));
1104 assert_eq!(err.description(), cause.description());
1105 assert_eq!(format!("{:?}", err.cause().unwrap()), format!("{:?}", cause));
1106 }
1107
1108 #[test]
1109 fn tuple_wrapper_trait_str() {
1110 let desc = "hello";
1111 let err: &Error = &TupleWrapper::Other(desc);
1112 assert_eq!(format!("{}", err), format!("Error: {}", desc));
1113 assert_eq!(format!("{:?}", err), format!("Other({:?})", desc));
1114 assert_eq!(err.description(), desc);
1115 assert!(err.cause().is_none());
1116 }
1117
1118 #[test]
1119 fn tuple_wrapper_trait_two_fields() {
1120 let invalid_utf8: Vec<u8> = vec![0, 159, 146, 150];
1121 let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err().utf8_error();
1122 let err: &Error = &TupleWrapper::FromUtf8Error(cause.clone(), invalid_utf8.clone());
1123 assert_eq!(format!("{}", err), format!("{desc} at index {pos}: {cause}", desc=err.description(), pos=cause.valid_up_to(), cause=cause));
1124 assert_eq!(format!("{:?}", err), format!("FromUtf8Error({:?}, {:?})", cause, invalid_utf8));
1125 assert_eq!(err.description(), "utf8 error");
1126 assert_eq!(format!("{:?}", err.cause().unwrap()), format!("{:?}", cause));
1127 }
1128
1129 #[test]
1130 fn tuple_wrapper_from() {
1131 let cause = "one and a half times pi".parse::<f32>().unwrap_err();
1132 let err = TupleWrapper::ParseFloatError(cause.clone());
1133 let err_from: TupleWrapper = From::from(cause);
1134 assert_eq!(err_from, err);
1135 }
1136
1137 #[test]
1138 fn tuple_wrapper_custom_from() {
1139 let invalid_utf8: Vec<u8> = vec![0, 159, 146, 150];
1140 let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err();
1141 let err = TupleWrapper::FromUtf8Error(cause.utf8_error().clone(), invalid_utf8);
1142 let err_from: TupleWrapper = From::from(cause);
1143 assert_eq!(err_from, err);
1144 }
1145
1146 #[test]
1147 fn tuple_wrapper_discard() {
1148 let err: TupleWrapper = From::from("hello");
1149 assert_eq!(format!("{}", err), format!("Discard"));
1150 assert_eq!(format!("{:?}", err), format!("Discard"));
1151 assert_eq!(err.description(), "Discard");
1152 assert!(err.cause().is_none());
1153 }
1154
1155 #[test]
1156 fn tuple_wrapper_singleton() {
1157 let err: TupleWrapper = TupleWrapper::Singleton;
1158 assert_eq!(format!("{}", err), format!("Just a string"));
1159 assert_eq!(format!("{:?}", err), format!("Singleton"));
1160 assert_eq!(err.description(), "Singleton");
1161 assert!(err.cause().is_none());
1162 }
1163
1164 quick_error! {
1165 #[derive(Debug, PartialEq)]
1166 pub enum StructWrapper {
1167 Utf8Error{ err: Utf8Error, hint: Option<&'static str> } {
1169 cause(err)
1170 display(me) -> ("{desc} at index {pos}: {err}", desc=me.description(), pos=err.valid_up_to(), err=err)
1171 description("utf8 error")
1172 from(err: Utf8Error) -> { err: err, hint: None }
1173 }
1174 ExcessComma { descr: &'static str, } {
1176 description(descr)
1177 display("Error: {}", descr)
1178 }
1179 }
1180 }
1181
1182 #[test]
1183 fn struct_wrapper_err() {
1184 let invalid_utf8: Vec<u8> = vec![0, 159, 146, 150];
1185 let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err().utf8_error();
1186 let err: &Error = &StructWrapper::Utf8Error{ err: cause.clone(), hint: Some("nonsense") };
1187 assert_eq!(format!("{}", err), format!("{desc} at index {pos}: {cause}", desc=err.description(), pos=cause.valid_up_to(), cause=cause));
1188 assert_eq!(format!("{:?}", err), format!("Utf8Error {{ err: {:?}, hint: {:?} }}", cause, Some("nonsense")));
1189 assert_eq!(err.description(), "utf8 error");
1190 assert_eq!(format!("{:?}", err.cause().unwrap()), format!("{:?}", cause));
1191 }
1192
1193 #[test]
1194 fn struct_wrapper_struct_from() {
1195 let invalid_utf8: Vec<u8> = vec![0, 159, 146, 150];
1196 let cause = String::from_utf8(invalid_utf8.clone()).unwrap_err().utf8_error();
1197 let err = StructWrapper::Utf8Error{ err: cause.clone(), hint: None };
1198 let err_from: StructWrapper = From::from(cause);
1199 assert_eq!(err_from, err);
1200 }
1201
1202 #[test]
1203 fn struct_wrapper_excess_comma() {
1204 let descr = "hello";
1205 let err = StructWrapper::ExcessComma { descr: descr };
1206 assert_eq!(format!("{}", err), format!("Error: {}", descr));
1207 assert_eq!(format!("{:?}", err), format!("ExcessComma {{ descr: {:?} }}", descr));
1208 assert_eq!(err.description(), descr);
1209 assert!(err.cause().is_none());
1210 }
1211
1212 quick_error! {
1213 #[derive(Debug)]
1214 pub enum ContextErr {
1215 Float(src: String, err: ParseFloatError) {
1216 context(s: &'a str, e: ParseFloatError) -> (s.to_string(), e)
1217 display("Float error {:?}: {}", src, err)
1218 }
1219 Int { src: String, err: ParseIntError } {
1220 context(s: &'a str, e: ParseIntError)
1221 -> {src: s.to_string(), err: e}
1222 display("Int error {:?}: {}", src, err)
1223 }
1224 Utf8(path: PathBuf, err: Utf8Error) {
1225 context(p: AsRef<Path>, e: Utf8Error)
1226 -> (p.as_ref().to_path_buf(), e)
1227 display("Path error at {:?}: {}", path, err)
1228 }
1229 Utf8Str(s: String, err: ::std::io::Error) {
1230 context(s: AsRef<str>, e: ::std::io::Error)
1231 -> (s.as_ref().to_string(), e)
1232 display("Str error {:?}: {}", s, err)
1233 }
1234 }
1235 }
1236
1237 #[test]
1238 fn parse_float_error() {
1239 fn parse_float(s: &str) -> Result<f32, ContextErr> {
1240 Ok(try!(s.parse().context(s)))
1241 }
1242 assert_eq!(format!("{}", parse_float("12ab").unwrap_err()),
1243 r#"Float error "12ab": invalid float literal"#);
1244 }
1245
1246 #[test]
1247 fn parse_int_error() {
1248 fn parse_int(s: &str) -> Result<i32, ContextErr> {
1249 Ok(try!(s.parse().context(s)))
1250 }
1251 assert_eq!(format!("{}", parse_int("12.5").unwrap_err()),
1252 r#"Int error "12.5": invalid digit found in string"#);
1253 }
1254
1255 #[test]
1256 fn debug_context() {
1257 fn parse_int(s: &str) -> i32 {
1258 s.parse().context(s).unwrap()
1259 }
1260 assert_eq!(parse_int("12"), 12);
1261 assert_eq!(format!("{:?}", "x".parse::<i32>().context("x")),
1262 r#"Err(Context("x", ParseIntError { kind: InvalidDigit }))"#);
1263 }
1264
1265 #[test]
1266 fn path_context() {
1267 fn parse_utf<P: AsRef<Path>>(s: &[u8], p: P)
1268 -> Result<(), ContextErr>
1269 {
1270 try!(::std::str::from_utf8(s).context(p));
1271 Ok(())
1272 }
1273 let etext = parse_utf(b"a\x80\x80", "/etc").unwrap_err().to_string();
1274 assert!(etext.starts_with(
1275 "Path error at \"/etc\": invalid utf-8"));
1276 let etext = parse_utf(b"\x80\x80", PathBuf::from("/tmp")).unwrap_err()
1277 .to_string();
1278 assert!(etext.starts_with(
1279 "Path error at \"/tmp\": invalid utf-8"));
1280 }
1281
1282 #[test]
1283 fn conditional_compilation() {
1284 quick_error! {
1285 #[allow(dead_code)]
1286 #[derive(Debug)]
1287 pub enum Test {
1288 #[cfg(feature = "foo")]
1289 Variant
1290 }
1291 }
1292 }
1293}