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
/*! Data packet
*/

use super::*;

use tox_binary_io::*;
use crate::dht::{CookieRequest, CookieResponse, CryptoHandshake, CryptoData};
use crate::relay::connection_id::ConnectionId;

/** Sent by client to server.
The client sends data with `connection_id` and the server
relays it to the given connection

Serialized form:

Length   | Content
-------- | ------
`1`      | connection_id [ `0x10` .. `0xFF` ]
variable | Data

*/
#[derive(Debug, PartialEq, Clone)]
pub struct Data {
    /// The id of the connection of the client
    pub connection_id: ConnectionId,
    /// Data payload
    pub data: DataPayload,
}

impl FromBytes for Data {
    named!(from_bytes<Data>, do_parse!(
        connection_id: call!(ConnectionId::from_bytes) >>
        data: call!(DataPayload::from_bytes) >>
        (Data { connection_id, data })
    ));
}

impl ToBytes for Data {
    fn to_bytes<'a>(&self, buf: (&'a mut [u8], usize)) -> Result<(&'a mut [u8], usize), GenError> {
        do_gen!(buf,
            gen_call!(|buf, connection_id| ConnectionId::to_bytes(connection_id, buf), &self.connection_id) >>
            gen_call!(|buf, packet| DataPayload::to_bytes(packet, buf), &self.data)
        )
    }
}

/// Data payload enum.
#[derive(Debug, PartialEq, Clone)]
pub enum DataPayload {
    /// [`CookieRequest`](../../dht/packet/struct.CookieRequest.html) structure.
    CookieRequest(CookieRequest),
    /// [`CookieResponse`](../../dht/packet/struct.CookieResponse.html) structure.
    CookieResponse(CookieResponse),
    /// [`CryptoHandshake`](../../dht/packet/struct.CryptoHandshake.html) structure.
    CryptoHandshake(CryptoHandshake),
    /// [`CryptoData`](../../dht/packet/struct.CryptoData.html) structure.
    CryptoData(CryptoData),
}

impl ToBytes for DataPayload {
    fn to_bytes<'a>(&self, buf: (&'a mut [u8], usize)) -> Result<(&'a mut [u8], usize), GenError> {
        match *self {
            DataPayload::CookieRequest(ref p) => p.to_bytes(buf),
            DataPayload::CookieResponse(ref p) => p.to_bytes(buf),
            DataPayload::CryptoHandshake(ref p) => p.to_bytes(buf),
            DataPayload::CryptoData(ref p) => p.to_bytes(buf),
        }
    }
}

impl FromBytes for DataPayload {
    named!(from_bytes<DataPayload>, alt!(
        map!(CookieRequest::from_bytes, DataPayload::CookieRequest) |
        map!(CookieResponse::from_bytes, DataPayload::CookieResponse) |
        map!(CryptoHandshake::from_bytes, DataPayload::CryptoHandshake) |
        map!(CryptoData::from_bytes, DataPayload::CryptoData)
    ));
}

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

    encode_decode_test!(
        tox_crypto::crypto_init().unwrap(),
        data_encode_decode,
        Data {
            connection_id: ConnectionId::from_index(1),
            data: DataPayload::CryptoData(CryptoData {
                nonce_last_bytes: 42,
                payload: vec![42; 123],
            }),
        }
    );
}