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
// Network encoding for lightning network peer protocol data types
// Written in 2020-2022 by
//     Dr. Maxim Orlovsky <orlovsky@pandoracore.com>
//
// To the extent possible under law, the author(s) have dedicated all
// copyright and related and neighboring rights to this software to
// the public domain worldwide. This software is distributed without
// any warranty.
//
// You should have received a copy of the MIT License
// along with this software.
// If not, see <https://opensource.org/licenses/MIT>.

use std::io;
use std::ops::Deref;

use super::{Error, LightningDecode, LightningEncode};

impl LightningEncode for &[u8] {
    fn lightning_encode<E: io::Write>(&self, mut e: E) -> Result<usize, Error> {
        let mut len = self.len();
        // We handle oversize problems at the level of `usize` value
        // serializaton
        len += len.lightning_encode(&mut e)?;
        e.write_all(self)?;
        Ok(len)
    }
}

impl<const LEN: usize> LightningEncode for [u8; LEN] {
    fn lightning_encode<E: io::Write>(&self, mut e: E) -> Result<usize, Error> {
        e.write_all(self)?;
        Ok(self.len())
    }
}

impl<const LEN: usize> LightningDecode for [u8; LEN] {
    fn lightning_decode<D: io::Read>(mut d: D) -> Result<Self, Error> {
        let mut ret = [0u8; LEN];
        d.read_exact(&mut ret)?;
        Ok(ret)
    }
}

impl LightningEncode for Box<[u8]> {
    fn lightning_encode<E: io::Write>(&self, e: E) -> Result<usize, Error> {
        self.deref().lightning_encode(e)
    }
}

impl LightningDecode for Box<[u8]> {
    fn lightning_decode<D: io::Read>(mut d: D) -> Result<Self, Error> {
        let len = usize::lightning_decode(&mut d)?;
        let mut ret = vec![0u8; len];
        d.read_exact(&mut ret)?;
        Ok(ret.into_boxed_slice())
    }
}

impl LightningEncode for &str {
    fn lightning_encode<E: io::Write>(&self, e: E) -> Result<usize, Error> {
        self.as_bytes().lightning_encode(e)
    }
}

impl LightningEncode for String {
    fn lightning_encode<E: io::Write>(&self, e: E) -> Result<usize, Error> {
        self.as_bytes().lightning_encode(e)
    }
}

impl LightningDecode for String {
    fn lightning_decode<D: io::Read>(d: D) -> Result<Self, Error> {
        Ok(String::from_utf8_lossy(&Vec::<u8>::lightning_decode(d)?)
            .to_string())
    }
}