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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
use std::{fmt, ops::Deref, sync::Arc};

use derive_more::From;
use parking_lot::Mutex;
use serde::{
    de::{self, Deserializer},
    ser::Serializer,
    Deserialize, Serialize,
};

/// Used to send values that cannot be serialized.
/// Works inside the current node only.
///
/// Messages must be instances of `Clone`, `Serialize` and `Deserialize` because
/// of network communication and message dumping. However, in some cases, it's
/// desired to have messages that cannot satisfy requirements.
/// For instance, when sending channels.
///
/// `Local<T>` implements `Debug`, `Serialize` and `Deserialize` for any `T`.
/// Meanwhile, it can be serialized (but w/o useful information), it cannot be
/// deserialized (it returns an error on attempts).
///
/// # Example
/// ```ignore
/// #[message]
/// pub struct OpenDirectChannel {
///     pub tx: Local<mpsc::Sender>,
/// }
/// ```
#[derive(Clone, Copy, PartialEq, Eq, Default, From)]
pub struct Local<T>(T);

impl<T> Local<T> {
    #[inline]
    pub fn into_inner(self) -> T {
        self.0
    }
}

impl<T> Deref for Local<T> {
    type Target = T;

    #[inline]
    fn deref(&self) -> &Self::Target {
        &self.0
    }
}

impl<T> fmt::Debug for Local<T> {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        write!(f, "<local>")
    }
}

impl<T> Serialize for Local<T> {
    #[inline]
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        // TODO: fail to serialize in the network context.
        serializer.serialize_str("<local>")
    }
}

impl<'de, T> Deserialize<'de> for Local<T> {
    fn deserialize<D>(_deserializer: D) -> Result<Self, D::Error>
    where
        D: Deserializer<'de>,
    {
        Err(de::Error::custom("Local<T> cannot be deserialized"))
    }
}

/// Used to transfer ownership over messaging.
///
/// Messages must be instances of `Clone`, `Serialize` and `Deserialize` because
/// of network communication and message dumping. However, in some cases, it's
/// desired to have messages that cannot satisfy requirements.
/// For instance, when sending sockets or files.
///
/// `MoveOwnership<T>` implements `Debug`, `Clone`, `Serialize` and
/// `Deserialize` for any `T`. Meanwhile, it can be serialized (but w/o useful
/// information), it cannot be deserialized (it returns an error on attempts).
///
/// # Example
/// ```ignore
/// #[message]
/// pub struct HandleFile {
///     pub path: PathBuf,
///     pub file: MoveOwnership<File>,
/// }
/// ```
pub struct MoveOwnership<T>(Arc<Mutex<Option<T>>>);

impl<T> MoveOwnership<T> {
    /// Takes the value. Next calls will return `None`.
    pub fn take(&self) -> Option<T> {
        self.0.lock().take()
    }
}

impl<T> From<T> for MoveOwnership<T> {
    fn from(value: T) -> Self {
        Self(Arc::new(Mutex::new(Some(value))))
    }
}

impl<T> Clone for MoveOwnership<T> {
    fn clone(&self) -> MoveOwnership<T> {
        Self(self.0.clone())
    }
}

impl<T> fmt::Debug for MoveOwnership<T> {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        write!(f, "<local>")
    }
}

impl<T> Serialize for MoveOwnership<T> {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        // TODO: fail to serialize in the network context.
        serializer.serialize_str("<local>")
    }
}

impl<'de, T> Deserialize<'de> for MoveOwnership<T> {
    fn deserialize<D>(_deserializer: D) -> Result<Self, D::Error>
    where
        D: Deserializer<'de>,
    {
        Err(de::Error::custom("MoveOwnership<T> cannot be deserialized"))
    }
}