2016-10-24 00:02:02 +03:00
|
|
|
// Copyright 2016 The Grin Developers
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
//! Message types that transit over the network and related serialization code.
|
|
|
|
|
2016-12-11 06:11:49 +03:00
|
|
|
use std::net::{SocketAddr, SocketAddrV4, SocketAddrV6, Ipv4Addr, Ipv6Addr};
|
2016-10-29 16:51:24 +03:00
|
|
|
use num::FromPrimitive;
|
|
|
|
|
2016-12-11 06:11:49 +03:00
|
|
|
use futures::future::{Future, ok};
|
|
|
|
use tokio_core::net::TcpStream;
|
|
|
|
use tokio_core::io::{write_all, read_exact};
|
|
|
|
|
2016-11-01 05:03:12 +03:00
|
|
|
use core::ser::{self, Writeable, Readable, Writer, Reader};
|
2016-12-11 06:11:49 +03:00
|
|
|
use core::consensus::MAX_MSG_LEN;
|
2016-10-26 08:06:13 +03:00
|
|
|
|
2016-10-29 22:36:45 +03:00
|
|
|
use types::*;
|
|
|
|
|
2016-10-26 08:06:13 +03:00
|
|
|
/// Current latest version of the protocol
|
|
|
|
pub const PROTOCOL_VERSION: u32 = 1;
|
|
|
|
/// Grin's user agent with current version (TODO externalize)
|
|
|
|
pub const USER_AGENT: &'static str = "MW/Grin 0.1";
|
2016-10-24 00:02:02 +03:00
|
|
|
|
2016-10-25 07:35:10 +03:00
|
|
|
/// Magic number expected in the header of every message
|
|
|
|
const MAGIC: [u8; 2] = [0x1e, 0xc5];
|
|
|
|
|
2016-12-11 06:11:49 +03:00
|
|
|
/// Size in bytes of a message header
|
|
|
|
pub const HEADER_LEN: u64 = 11;
|
|
|
|
|
2016-10-25 07:35:10 +03:00
|
|
|
/// Codes for each error that can be produced reading a message.
|
2016-10-26 08:06:13 +03:00
|
|
|
pub enum ErrCodes {
|
2016-10-26 21:21:45 +03:00
|
|
|
UnsupportedVersion = 100,
|
2016-10-24 00:02:02 +03:00
|
|
|
}
|
|
|
|
|
2016-10-25 07:35:10 +03:00
|
|
|
/// Types of messages
|
2016-10-29 16:51:24 +03:00
|
|
|
enum_from_primitive! {
|
2016-11-06 02:31:45 +03:00
|
|
|
#[derive(Debug, Clone, Copy)]
|
2016-11-01 20:42:33 +03:00
|
|
|
pub enum Type {
|
|
|
|
Error,
|
|
|
|
Hand,
|
|
|
|
Shake,
|
|
|
|
Ping,
|
|
|
|
Pong,
|
|
|
|
GetPeerAddrs,
|
|
|
|
PeerAddrs,
|
|
|
|
Block,
|
|
|
|
Transaction,
|
|
|
|
}
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
|
2016-12-11 06:11:49 +03:00
|
|
|
/// Future combinator to read any message where the body is a Readable. Reads
|
|
|
|
/// the header first, handles its validation and then reads the Readable body,
|
|
|
|
/// allocating buffers of the right size.
|
|
|
|
pub fn read_msg<T>(conn: TcpStream) -> Box<Future<Item = (TcpStream, T), Error = ser::Error>>
|
|
|
|
where T: Readable<T> + 'static
|
|
|
|
{
|
|
|
|
let read_header = read_exact(conn, vec![0u8; HEADER_LEN as usize])
|
|
|
|
.map_err(|e| ser::Error::IOErr(e))
|
|
|
|
.and_then(|(reader, buf)| {
|
|
|
|
let header = try!(ser::deserialize::<MsgHeader>(&mut &buf[..]));
|
|
|
|
if header.msg_len > MAX_MSG_LEN {
|
|
|
|
// TODO add additional restrictions on a per-message-type basis to avoid 20MB
|
|
|
|
// pings
|
|
|
|
return Err(ser::Error::TooLargeReadErr);
|
|
|
|
}
|
|
|
|
Ok((reader, header))
|
|
|
|
});
|
|
|
|
|
|
|
|
let read_msg = read_header.and_then(|(reader, header)| {
|
|
|
|
read_exact(reader, vec![0u8; header.msg_len as usize]).map_err(|e| ser::Error::IOErr(e))
|
|
|
|
})
|
|
|
|
.and_then(|(reader, buf)| {
|
|
|
|
let body = try!(ser::deserialize(&mut &buf[..]));
|
|
|
|
Ok((reader, body))
|
|
|
|
});
|
|
|
|
Box::new(read_msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Future combinator to write a full message from a Writeable payload.
|
|
|
|
/// Serializes the payload first and then sends the message header and that
|
|
|
|
/// payload.
|
|
|
|
pub fn write_msg<T>(conn: TcpStream,
|
|
|
|
msg: T,
|
|
|
|
msg_type: Type)
|
|
|
|
-> Box<Future<Item = TcpStream, Error = ser::Error>>
|
|
|
|
where T: Writeable + 'static
|
|
|
|
{
|
|
|
|
let write_msg = ok((conn)).and_then(move |conn| {
|
|
|
|
// prepare the body first so we know its serialized length
|
|
|
|
let mut body_buf = vec![];
|
|
|
|
ser::serialize(&mut body_buf, &msg);
|
|
|
|
|
|
|
|
// build and send the header using the body size
|
|
|
|
let mut header_buf = vec![];
|
|
|
|
let blen = body_buf.len() as u64;
|
|
|
|
ser::serialize(&mut header_buf, &MsgHeader::new(msg_type, blen));
|
|
|
|
write_all(conn, header_buf)
|
|
|
|
.and_then(|(conn, _)| write_all(conn, body_buf))
|
|
|
|
.map(|(conn, _)| conn)
|
|
|
|
.map_err(|e| ser::Error::IOErr(e))
|
|
|
|
});
|
|
|
|
Box::new(write_msg)
|
|
|
|
}
|
|
|
|
|
2016-10-25 07:35:10 +03:00
|
|
|
/// Header of any protocol message, used to identify incoming messages.
|
|
|
|
pub struct MsgHeader {
|
|
|
|
magic: [u8; 2],
|
2016-10-31 04:23:52 +03:00
|
|
|
pub msg_type: Type,
|
2016-12-11 06:11:49 +03:00
|
|
|
pub msg_len: u64,
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl MsgHeader {
|
2016-12-11 06:11:49 +03:00
|
|
|
pub fn new(msg_type: Type, len: u64) -> MsgHeader {
|
2016-10-31 04:23:52 +03:00
|
|
|
MsgHeader {
|
|
|
|
magic: MAGIC,
|
|
|
|
msg_type: msg_type,
|
2016-12-11 06:11:49 +03:00
|
|
|
msg_len: len,
|
2016-10-31 04:23:52 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-01 20:42:33 +03:00
|
|
|
/// Serialized length of the header in bytes
|
|
|
|
pub fn serialized_len(&self) -> u64 {
|
2016-12-11 06:11:49 +03:00
|
|
|
HEADER_LEN
|
2016-11-01 20:42:33 +03:00
|
|
|
}
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Writeable for MsgHeader {
|
2016-11-01 05:03:12 +03:00
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
2016-10-25 07:35:10 +03:00
|
|
|
ser_multiwrite!(writer,
|
|
|
|
[write_u8, self.magic[0]],
|
|
|
|
[write_u8, self.magic[1]],
|
2016-12-11 06:11:49 +03:00
|
|
|
[write_u8, self.msg_type as u8],
|
|
|
|
[write_u64, self.msg_len]);
|
2016-11-01 05:03:12 +03:00
|
|
|
Ok(())
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Readable<MsgHeader> for MsgHeader {
|
|
|
|
fn read(reader: &mut Reader) -> Result<MsgHeader, ser::Error> {
|
|
|
|
try!(reader.expect_u8(MAGIC[0]));
|
|
|
|
try!(reader.expect_u8(MAGIC[1]));
|
2016-12-11 06:11:49 +03:00
|
|
|
let (t, len) = ser_multiread!(reader, read_u8, read_u64);
|
2016-11-01 20:42:33 +03:00
|
|
|
match Type::from_u8(t) {
|
|
|
|
Some(ty) => {
|
|
|
|
Ok(MsgHeader {
|
|
|
|
magic: MAGIC,
|
|
|
|
msg_type: ty,
|
2016-12-11 06:11:49 +03:00
|
|
|
msg_len: len,
|
2016-11-01 20:42:33 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
None => Err(ser::Error::CorruptedData),
|
2016-10-28 00:28:02 +03:00
|
|
|
}
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// First part of a handshake, sender advertises its version and
|
|
|
|
/// characteristics.
|
2016-10-24 00:02:02 +03:00
|
|
|
pub struct Hand {
|
2016-10-26 08:06:13 +03:00
|
|
|
/// protocol version of the sender
|
|
|
|
pub version: u32,
|
|
|
|
/// capabilities of the sender
|
|
|
|
pub capabilities: Capabilities,
|
|
|
|
/// randomly generated for each handshake, helps detect self
|
|
|
|
pub nonce: u64,
|
|
|
|
/// network address of the sender
|
|
|
|
pub sender_addr: SockAddr,
|
|
|
|
/// network address of the receiver
|
|
|
|
pub receiver_addr: SockAddr,
|
|
|
|
/// name of version of the software
|
|
|
|
pub user_agent: String,
|
2016-10-24 00:02:02 +03:00
|
|
|
}
|
|
|
|
|
2016-10-25 07:35:10 +03:00
|
|
|
impl Writeable for Hand {
|
2016-11-01 05:03:12 +03:00
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
2016-10-25 07:35:10 +03:00
|
|
|
ser_multiwrite!(writer,
|
|
|
|
[write_u32, self.version],
|
2016-10-26 08:06:13 +03:00
|
|
|
[write_u32, self.capabilities.bits()],
|
2016-10-28 00:28:02 +03:00
|
|
|
[write_u64, self.nonce]);
|
2016-10-26 08:06:13 +03:00
|
|
|
self.sender_addr.write(writer);
|
|
|
|
self.receiver_addr.write(writer);
|
2016-11-01 20:42:33 +03:00
|
|
|
writer.write_bytes(&self.user_agent)
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Readable<Hand> for Hand {
|
|
|
|
fn read(reader: &mut Reader) -> Result<Hand, ser::Error> {
|
2016-10-26 08:06:13 +03:00
|
|
|
let (version, capab, nonce) = ser_multiread!(reader, read_u32, read_u32, read_u64);
|
|
|
|
let sender_addr = try!(SockAddr::read(reader));
|
|
|
|
let receiver_addr = try!(SockAddr::read(reader));
|
|
|
|
let ua = try!(reader.read_vec());
|
2016-10-28 00:28:02 +03:00
|
|
|
let user_agent = try!(String::from_utf8(ua).map_err(|_| ser::Error::CorruptedData));
|
|
|
|
let capabilities = try!(Capabilities::from_bits(capab).ok_or(ser::Error::CorruptedData));
|
2016-10-26 08:06:13 +03:00
|
|
|
Ok(Hand {
|
2016-10-25 07:35:10 +03:00
|
|
|
version: version,
|
2016-10-26 08:06:13 +03:00
|
|
|
capabilities: capabilities,
|
2016-10-28 00:28:02 +03:00
|
|
|
nonce: nonce,
|
2016-10-26 08:06:13 +03:00
|
|
|
sender_addr: sender_addr,
|
2016-10-25 07:35:10 +03:00
|
|
|
receiver_addr: receiver_addr,
|
|
|
|
user_agent: user_agent,
|
2016-10-26 08:06:13 +03:00
|
|
|
})
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Second part of a handshake, receiver of the first part replies with its own
|
|
|
|
/// version and characteristics.
|
2016-10-24 00:02:02 +03:00
|
|
|
pub struct Shake {
|
2016-10-26 08:06:13 +03:00
|
|
|
/// sender version
|
|
|
|
pub version: u32,
|
|
|
|
/// sender capabilities
|
|
|
|
pub capabilities: Capabilities,
|
|
|
|
/// name of version of the software
|
|
|
|
pub user_agent: String,
|
2016-10-24 00:02:02 +03:00
|
|
|
}
|
|
|
|
|
2016-10-26 08:06:13 +03:00
|
|
|
impl Writeable for Shake {
|
2016-11-01 05:03:12 +03:00
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
2016-10-25 07:35:10 +03:00
|
|
|
ser_multiwrite!(writer,
|
|
|
|
[write_u32, self.version],
|
2016-10-26 08:06:13 +03:00
|
|
|
[write_u32, self.capabilities.bits()],
|
2016-11-01 20:42:33 +03:00
|
|
|
[write_bytes, &self.user_agent]);
|
2016-11-01 05:03:12 +03:00
|
|
|
Ok(())
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Readable<Shake> for Shake {
|
|
|
|
fn read(reader: &mut Reader) -> Result<Shake, ser::Error> {
|
|
|
|
let (version, capab, ua) = ser_multiread!(reader, read_u32, read_u32, read_vec);
|
2016-10-26 08:06:13 +03:00
|
|
|
let user_agent = try!(String::from_utf8(ua).map_err(|_| ser::Error::CorruptedData));
|
2016-10-28 00:28:02 +03:00
|
|
|
let capabilities = try!(Capabilities::from_bits(capab).ok_or(ser::Error::CorruptedData));
|
2016-10-26 08:06:13 +03:00
|
|
|
Ok(Shake {
|
2016-10-25 07:35:10 +03:00
|
|
|
version: version,
|
2016-10-26 08:06:13 +03:00
|
|
|
capabilities: capabilities,
|
2016-10-25 07:35:10 +03:00
|
|
|
user_agent: user_agent,
|
2016-10-26 08:06:13 +03:00
|
|
|
})
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-29 22:36:45 +03:00
|
|
|
/// Ask for other peers addresses, required for network discovery.
|
|
|
|
pub struct GetPeerAddrs {
|
|
|
|
/// Filters on the capabilities we'd like the peers to have
|
|
|
|
pub capabilities: Capabilities,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Writeable for GetPeerAddrs {
|
2016-11-01 20:42:33 +03:00
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
2016-10-29 22:36:45 +03:00
|
|
|
writer.write_u32(self.capabilities.bits())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Readable<GetPeerAddrs> for GetPeerAddrs {
|
|
|
|
fn read(reader: &mut Reader) -> Result<GetPeerAddrs, ser::Error> {
|
|
|
|
let capab = try!(reader.read_u32());
|
|
|
|
let capabilities = try!(Capabilities::from_bits(capab).ok_or(ser::Error::CorruptedData));
|
|
|
|
Ok(GetPeerAddrs { capabilities: capabilities })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Peer addresses we know of that are fresh enough, in response to
|
|
|
|
/// GetPeerAddrs.
|
|
|
|
pub struct PeerAddrs {
|
|
|
|
pub peers: Vec<SockAddr>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Writeable for PeerAddrs {
|
2016-11-01 20:42:33 +03:00
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
|
|
|
try!(writer.write_u32(self.peers.len() as u32));
|
2016-10-29 22:36:45 +03:00
|
|
|
for p in &self.peers {
|
|
|
|
p.write(writer);
|
|
|
|
}
|
2016-11-01 20:42:33 +03:00
|
|
|
Ok(())
|
2016-10-29 22:36:45 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Readable<PeerAddrs> for PeerAddrs {
|
|
|
|
fn read(reader: &mut Reader) -> Result<PeerAddrs, ser::Error> {
|
|
|
|
let peer_count = try!(reader.read_u32());
|
|
|
|
if peer_count > 1000 {
|
2016-12-11 06:11:49 +03:00
|
|
|
return Err(ser::Error::TooLargeReadErr);
|
2016-10-29 22:36:45 +03:00
|
|
|
}
|
|
|
|
let peers = try_map_vec!([0..peer_count], |_| SockAddr::read(reader));
|
|
|
|
Ok(PeerAddrs { peers: peers })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-25 07:35:10 +03:00
|
|
|
/// We found some issue in the communication, sending an error back, usually
|
|
|
|
/// followed by closing the connection.
|
2016-10-24 00:02:02 +03:00
|
|
|
pub struct PeerError {
|
2016-10-26 08:06:13 +03:00
|
|
|
/// error code
|
|
|
|
pub code: u32,
|
|
|
|
/// slightly more user friendly message
|
|
|
|
pub message: String,
|
2016-10-24 00:02:02 +03:00
|
|
|
}
|
2016-10-25 07:35:10 +03:00
|
|
|
|
|
|
|
impl Writeable for PeerError {
|
2016-11-01 05:03:12 +03:00
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
2016-11-01 20:42:33 +03:00
|
|
|
ser_multiwrite!(writer, [write_u32, self.code], [write_bytes, &self.message]);
|
2016-11-01 05:03:12 +03:00
|
|
|
Ok(())
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Readable<PeerError> for PeerError {
|
|
|
|
fn read(reader: &mut Reader) -> Result<PeerError, ser::Error> {
|
|
|
|
let (code, msg) = ser_multiread!(reader, read_u32, read_vec);
|
2016-10-26 08:06:13 +03:00
|
|
|
let message = try!(String::from_utf8(msg).map_err(|_| ser::Error::CorruptedData));
|
|
|
|
Ok(PeerError {
|
2016-10-25 07:35:10 +03:00
|
|
|
code: code,
|
|
|
|
message: message,
|
2016-10-26 08:06:13 +03:00
|
|
|
})
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-28 00:28:02 +03:00
|
|
|
/// Only necessary so we can implement Readable and Writeable. Rust disallows
|
|
|
|
/// implementing traits when both types are outside of this crate (which is the
|
|
|
|
/// case for SocketAddr and Readable/Writeable).
|
2016-10-26 08:06:13 +03:00
|
|
|
pub struct SockAddr(pub SocketAddr);
|
|
|
|
|
|
|
|
impl Writeable for SockAddr {
|
2016-11-01 05:03:12 +03:00
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
2016-10-26 08:06:13 +03:00
|
|
|
match self.0 {
|
|
|
|
SocketAddr::V4(sav4) => {
|
2016-10-25 07:35:10 +03:00
|
|
|
ser_multiwrite!(writer,
|
|
|
|
[write_u8, 0],
|
2016-10-26 08:06:13 +03:00
|
|
|
[write_fixed_bytes, &sav4.ip().octets().to_vec()],
|
2016-10-25 07:35:10 +03:00
|
|
|
[write_u16, sav4.port()]);
|
|
|
|
}
|
2016-10-26 08:06:13 +03:00
|
|
|
SocketAddr::V6(sav6) => {
|
2016-11-01 05:03:12 +03:00
|
|
|
try!(writer.write_u8(1));
|
2016-10-26 08:06:13 +03:00
|
|
|
for seg in &sav6.ip().segments() {
|
2016-11-01 05:03:12 +03:00
|
|
|
try!(writer.write_u16(*seg));
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
2016-11-01 05:03:12 +03:00
|
|
|
try!(writer.write_u16(sav6.port()));
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
2016-11-01 05:03:12 +03:00
|
|
|
Ok(())
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-26 08:06:13 +03:00
|
|
|
impl Readable<SockAddr> for SockAddr {
|
|
|
|
fn read(reader: &mut Reader) -> Result<SockAddr, ser::Error> {
|
|
|
|
let v4_or_v6 = try!(reader.read_u8());
|
2016-10-25 07:35:10 +03:00
|
|
|
if v4_or_v6 == 0 {
|
2016-10-26 08:06:13 +03:00
|
|
|
let ip = try!(reader.read_fixed_bytes(4));
|
|
|
|
let port = try!(reader.read_u16());
|
2016-10-28 00:28:02 +03:00
|
|
|
Ok(SockAddr(SocketAddr::V4(SocketAddrV4::new(Ipv4Addr::new(ip[0],
|
|
|
|
ip[1],
|
|
|
|
ip[2],
|
|
|
|
ip[3]),
|
|
|
|
port))))
|
2016-10-25 07:35:10 +03:00
|
|
|
} else {
|
2016-10-29 22:36:45 +03:00
|
|
|
let ip = try_map_vec!([0..8], |_| reader.read_u16());
|
2016-10-26 08:06:13 +03:00
|
|
|
let port = try!(reader.read_u16());
|
2016-10-28 00:28:02 +03:00
|
|
|
Ok(SockAddr(SocketAddr::V6(SocketAddrV6::new(Ipv6Addr::new(ip[0],
|
|
|
|
ip[1],
|
|
|
|
ip[2],
|
|
|
|
ip[3],
|
|
|
|
ip[4],
|
|
|
|
ip[5],
|
|
|
|
ip[6],
|
|
|
|
ip[7]),
|
|
|
|
port,
|
|
|
|
0,
|
|
|
|
0))))
|
2016-10-25 07:35:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-11 06:11:49 +03:00
|
|
|
|
|
|
|
/// Placeholder for messages like Ping and Pong that don't send anything but
|
|
|
|
/// the header.
|
|
|
|
pub struct Empty {}
|
|
|
|
|
|
|
|
impl Writeable for Empty {
|
|
|
|
fn write(&self, writer: &mut Writer) -> Result<(), ser::Error> {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Readable<Empty> for Empty {
|
|
|
|
fn read(reader: &mut Reader) -> Result<Empty, ser::Error> {
|
|
|
|
Ok(Empty {})
|
|
|
|
}
|
|
|
|
}
|