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
mod msg;
pub use self::msg::*;
use std::io;
use std::str;
use serde_json;
use bytes::{BufMut, BytesMut};
use tokio_io::codec::{Encoder, Decoder, Framed};
use tokio_core::net::TcpStream;
use comms::{Client, Room};
use utils::*;
pub type MsgClient<I> = Client<I, MsgTransport>;
pub type MsgRoom<I> = Room<I, MsgTransport>;
#[derive(PartialEq, Clone, Copy, Debug, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")]
pub enum ClientKind {
Player,
Spectator,
}
pub type MsgTransport = Framed<TcpStream, MsgCodec>;
#[derive(Clone, Copy, Debug)]
pub struct MsgCodec;
impl Decoder for MsgCodec {
type Item = Msg;
type Error = io::Error;
fn decode(&mut self, buf: &mut BytesMut) -> io::Result<Option<Msg>> {
if let Some(n) = buf.as_ref().iter().position(|b| *b == b'\n') {
let line = buf.split_to(n);
buf.split_to(1);
let line = match str::from_utf8(line.as_ref()) {
Ok(s) => s,
Err(_) => return Err(io_error_from_str("invalid string")),
};
return match serde_json::from_str(line) {
Ok(msg) => Ok(Some(msg)),
Err(e) => Err(io_error_from_error(&e)),
};
}
Ok(None)
}
}
impl Encoder for MsgCodec {
type Item = Msg;
type Error = io::Error;
fn encode(&mut self, msg: Msg, buf: &mut BytesMut) -> io::Result<()> {
let msg_str = serde_json::to_string(&msg).map_err(|e| io_error_from_error(&e))?;
buf.extend(msg_str.as_bytes());
buf.put(b'\n');
Ok(())
}
}