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
89
90
91
92
93
94
95
96
97
98
99
//! Conversion method for infallible results
//!
//! This crate provides a convenience trait `UnwrapInfallible`,
//! adding method `unwrap_infallible` to `Result` types where an `Err` variant
//! is statically known to never occur.
//!
//! # Example
//!
//! ```
//! # #![cfg_attr(feature = "never_type", feature(never_type))]
//! #
//! use unwrap_infallible::UnwrapInfallible;
//! # #[cfg(not(feature = "blanket_impl"))]
//! use std::convert::Infallible;
//! # #[cfg(feature = "blanket_impl")]
//! # type Infallible = !;
//!
//! fn always_sunny() -> Result<String, Infallible> {
//!     Ok("it's always sunny!".into())
//! }
//!
//! fn main() {
//!     println!("{}", always_sunny().unwrap_infallible());
//! }
//! ```

#![warn(rust_2018_idioms)]
#![warn(clippy::all)]
#![warn(missing_docs)]
#![no_std]
#![cfg_attr(feature = "never_type", feature(never_type))]

#[cfg(not(feature = "blanket_impl"))]
use core::convert::Infallible;

/// Unwrapping an infallible result into its success value.
pub trait UnwrapInfallible {
    /// Type of the `Ok` variant of the result.
    type Ok;

    /// Unwraps a result, returning the content of an `Ok`.
    ///
    /// Unlike `Result::unwrap`, this method is known to never panic
    /// on the result types it is implemented for. Therefore, it can be used
    /// instead of `unwrap` as a maintainability safeguard that will fail
    /// to compile if the error type of the `Result` is later changed
    /// to an error that can actually occur.
    fn unwrap_infallible(self) -> Self::Ok;
}

#[cfg(feature = "blanket_impl")]
impl<T, E: Into<!>> UnwrapInfallible for Result<T, E> {
    type Ok = T;
    fn unwrap_infallible(self) -> T {
        match self {
            Ok(v) => v,
            Err(e) => e.into(),
        }
    }
}

#[cfg(all(feature = "never_type", not(feature = "blanket_impl")))]
impl<T> UnwrapInfallible for Result<T, !> {
    type Ok = T;
    fn unwrap_infallible(self) -> T {
        self.unwrap_or_else(|never| never)
    }
}

#[cfg(not(feature = "blanket_impl"))]
impl<T> UnwrapInfallible for Result<T, Infallible> {
    type Ok = T;
    fn unwrap_infallible(self) -> T {
        self.unwrap_or_else(|never| match never {})
    }
}

#[cfg(test)]
mod tests {
    use super::UnwrapInfallible;

    // Hmm, Infallible is not Into<!> yet
    #[cfg(not(feature = "blanket_impl"))]
    #[test]
    fn with_infallible() {
        use core::convert::TryFrom;

        let a = 42u8;
        let a = u64::try_from(a).unwrap_infallible();
        assert_eq!(a, 42u64);
    }

    #[cfg(feature = "never_type")]
    #[test]
    fn with_never_type() {
        let r: Result<bool, !> = Ok(true);
        assert!(r.unwrap_infallible());
    }
}