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
#[derive(Debug, Clone)]
pub struct Inner<IntoError, IntoWarning>
where
    IntoError: Clone + Send,
    IntoWarning: Clone + Send,
{
    pub error: Option<IntoError>,
    pub warning: Option<IntoWarning>,
}

#[derive(Debug, Clone)]
pub struct Flag<IntoError, IntoWarning>(
    std::sync::Arc<std::sync::Mutex<Inner<IntoError, IntoWarning>>>,
)
where
    IntoError: Clone + Send,
    IntoWarning: Clone + Send;

impl<IntoError, IntoWarning> Flag<IntoError, IntoWarning>
where
    IntoError: Clone + Send,
    IntoWarning: Clone + Send,
{
    pub fn new() -> Self {
        Self(std::sync::Arc::new(std::sync::Mutex::new(Inner {
            error: None,
            warning: None,
        })))
    }

    pub fn store_error_if_not_set<Error>(&self, error: Error)
    where
        Error: Into<IntoError>,
    {
        self.0
            .lock()
            .expect("mutex is not poisoned")
            .error
            .get_or_insert(error.into());
    }

    pub fn store_warning_if_not_set<Warning>(&self, warning: Warning)
    where
        Warning: Into<IntoWarning>,
    {
        self.0
            .lock()
            .expect("mutex is not poisoned")
            .warning
            .get_or_insert(warning.into());
    }

    pub fn load_error(&self) -> Result<(), IntoError> {
        match self.0.lock().expect("mutex is not poisoned").error.take() {
            Some(error) => Err(error),
            None => Ok(()),
        }
    }

    pub fn load_warning(&self) -> Option<IntoWarning> {
        self.0.lock().expect("mutex is not poisoned").warning.take()
    }
}

impl<IntoError, IntoWarning> Default for Flag<IntoError, IntoWarning>
where
    IntoError: Clone + Send,
    IntoWarning: Clone + Send,
{
    fn default() -> Self {
        Self(std::sync::Arc::new(std::sync::Mutex::new(Inner {
            error: None,
            warning: None,
        })))
    }
}