mirror of
https://github.com/fafhrd91/actix-web
synced 2025-01-18 22:01:50 +01:00
385 lines
11 KiB
Rust
385 lines
11 KiB
Rust
use byteorder::{ByteOrder, LittleEndian, NetworkEndian};
|
|
use bytes::{BufMut, Bytes, BytesMut};
|
|
use log::debug;
|
|
use rand;
|
|
|
|
use crate::ws::mask::apply_mask;
|
|
use crate::ws::proto::{CloseCode, CloseReason, OpCode};
|
|
use crate::ws::ProtocolError;
|
|
|
|
/// A struct representing a `WebSocket` frame.
|
|
#[derive(Debug)]
|
|
pub struct Parser;
|
|
|
|
impl Parser {
|
|
fn parse_metadata(
|
|
src: &[u8],
|
|
server: bool,
|
|
max_size: usize,
|
|
) -> Result<Option<(usize, bool, OpCode, usize, Option<u32>)>, ProtocolError> {
|
|
let chunk_len = src.len();
|
|
|
|
let mut idx = 2;
|
|
if chunk_len < 2 {
|
|
return Ok(None);
|
|
}
|
|
|
|
let first = src[0];
|
|
let second = src[1];
|
|
let finished = first & 0x80 != 0;
|
|
|
|
// check masking
|
|
let masked = second & 0x80 != 0;
|
|
if !masked && server {
|
|
return Err(ProtocolError::UnmaskedFrame);
|
|
} else if masked && !server {
|
|
return Err(ProtocolError::MaskedFrame);
|
|
}
|
|
|
|
// Op code
|
|
let opcode = OpCode::from(first & 0x0F);
|
|
|
|
if let OpCode::Bad = opcode {
|
|
return Err(ProtocolError::InvalidOpcode(first & 0x0F));
|
|
}
|
|
|
|
let len = second & 0x7F;
|
|
let length = if len == 126 {
|
|
if chunk_len < 4 {
|
|
return Ok(None);
|
|
}
|
|
let len = NetworkEndian::read_uint(&src[idx..], 2) as usize;
|
|
idx += 2;
|
|
len
|
|
} else if len == 127 {
|
|
if chunk_len < 10 {
|
|
return Ok(None);
|
|
}
|
|
let len = NetworkEndian::read_uint(&src[idx..], 8);
|
|
if len > max_size as u64 {
|
|
return Err(ProtocolError::Overflow);
|
|
}
|
|
idx += 8;
|
|
len as usize
|
|
} else {
|
|
len as usize
|
|
};
|
|
|
|
// check for max allowed size
|
|
if length > max_size {
|
|
return Err(ProtocolError::Overflow);
|
|
}
|
|
|
|
let mask = if server {
|
|
if chunk_len < idx + 4 {
|
|
return Ok(None);
|
|
}
|
|
|
|
let mask: &[u8] = &src[idx..idx + 4];
|
|
let mask_u32 = LittleEndian::read_u32(mask);
|
|
idx += 4;
|
|
Some(mask_u32)
|
|
} else {
|
|
None
|
|
};
|
|
|
|
Ok(Some((idx, finished, opcode, length, mask)))
|
|
}
|
|
|
|
/// Parse the input stream into a frame.
|
|
pub fn parse(
|
|
src: &mut BytesMut,
|
|
server: bool,
|
|
max_size: usize,
|
|
) -> Result<Option<(bool, OpCode, Option<BytesMut>)>, ProtocolError> {
|
|
// try to parse ws frame metadata
|
|
let (idx, finished, opcode, length, mask) =
|
|
match Parser::parse_metadata(src, server, max_size)? {
|
|
None => return Ok(None),
|
|
Some(res) => res,
|
|
};
|
|
|
|
// not enough data
|
|
if src.len() < idx + length {
|
|
return Ok(None);
|
|
}
|
|
|
|
// remove prefix
|
|
src.split_to(idx);
|
|
|
|
// no need for body
|
|
if length == 0 {
|
|
return Ok(Some((finished, opcode, None)));
|
|
}
|
|
|
|
let mut data = src.split_to(length);
|
|
|
|
// control frames must have length <= 125
|
|
match opcode {
|
|
OpCode::Ping | OpCode::Pong if length > 125 => {
|
|
return Err(ProtocolError::InvalidLength(length));
|
|
}
|
|
OpCode::Close if length > 125 => {
|
|
debug!("Received close frame with payload length exceeding 125. Morphing to protocol close frame.");
|
|
return Ok(Some((true, OpCode::Close, None)));
|
|
}
|
|
_ => (),
|
|
}
|
|
|
|
// unmask
|
|
if let Some(mask) = mask {
|
|
apply_mask(&mut data, mask);
|
|
}
|
|
|
|
Ok(Some((finished, opcode, Some(data))))
|
|
}
|
|
|
|
/// Parse the payload of a close frame.
|
|
pub fn parse_close_payload(payload: &[u8]) -> Option<CloseReason> {
|
|
if payload.len() >= 2 {
|
|
let raw_code = NetworkEndian::read_u16(payload);
|
|
let code = CloseCode::from(raw_code);
|
|
let description = if payload.len() > 2 {
|
|
Some(String::from_utf8_lossy(&payload[2..]).into())
|
|
} else {
|
|
None
|
|
};
|
|
Some(CloseReason { code, description })
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
/// Generate binary representation
|
|
pub fn write_message<B: Into<Bytes>>(
|
|
dst: &mut BytesMut,
|
|
pl: B,
|
|
op: OpCode,
|
|
fin: bool,
|
|
mask: bool,
|
|
) {
|
|
let payload = pl.into();
|
|
let one: u8 = if fin {
|
|
0x80 | Into::<u8>::into(op)
|
|
} else {
|
|
op.into()
|
|
};
|
|
let payload_len = payload.len();
|
|
let (two, p_len) = if mask {
|
|
(0x80, payload_len + 4)
|
|
} else {
|
|
(0, payload_len)
|
|
};
|
|
|
|
if payload_len < 126 {
|
|
dst.reserve(p_len + 2 + if mask { 4 } else { 0 });
|
|
dst.put_slice(&[one, two | payload_len as u8]);
|
|
} else if payload_len <= 65_535 {
|
|
dst.reserve(p_len + 4 + if mask { 4 } else { 0 });
|
|
dst.put_slice(&[one, two | 126]);
|
|
dst.put_u16_be(payload_len as u16);
|
|
} else {
|
|
dst.reserve(p_len + 10 + if mask { 4 } else { 0 });
|
|
dst.put_slice(&[one, two | 127]);
|
|
dst.put_u64_be(payload_len as u64);
|
|
};
|
|
|
|
if mask {
|
|
let mask = rand::random::<u32>();
|
|
dst.put_u32_le(mask);
|
|
dst.put_slice(payload.as_ref());
|
|
let pos = dst.len() - payload_len;
|
|
apply_mask(&mut dst[pos..], mask);
|
|
} else {
|
|
dst.put_slice(payload.as_ref());
|
|
}
|
|
}
|
|
|
|
/// Create a new Close control frame.
|
|
#[inline]
|
|
pub fn write_close(dst: &mut BytesMut, reason: Option<CloseReason>, mask: bool) {
|
|
let payload = match reason {
|
|
None => Vec::new(),
|
|
Some(reason) => {
|
|
let mut code_bytes = [0; 2];
|
|
NetworkEndian::write_u16(&mut code_bytes, reason.code.into());
|
|
|
|
let mut payload = Vec::from(&code_bytes[..]);
|
|
if let Some(description) = reason.description {
|
|
payload.extend(description.as_bytes());
|
|
}
|
|
payload
|
|
}
|
|
};
|
|
|
|
Parser::write_message(dst, payload, OpCode::Close, true, mask)
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use bytes::Bytes;
|
|
|
|
struct F {
|
|
finished: bool,
|
|
opcode: OpCode,
|
|
payload: Bytes,
|
|
}
|
|
|
|
fn is_none(
|
|
frm: &Result<Option<(bool, OpCode, Option<BytesMut>)>, ProtocolError>,
|
|
) -> bool {
|
|
match *frm {
|
|
Ok(None) => true,
|
|
_ => false,
|
|
}
|
|
}
|
|
|
|
fn extract(
|
|
frm: Result<Option<(bool, OpCode, Option<BytesMut>)>, ProtocolError>,
|
|
) -> F {
|
|
match frm {
|
|
Ok(Some((finished, opcode, payload))) => F {
|
|
finished,
|
|
opcode,
|
|
payload: payload
|
|
.map(|b| b.freeze())
|
|
.unwrap_or_else(|| Bytes::from("")),
|
|
},
|
|
_ => unreachable!("error"),
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse() {
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 0b0000_0001u8][..]);
|
|
assert!(is_none(&Parser::parse(&mut buf, false, 1024)));
|
|
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 0b0000_0001u8][..]);
|
|
buf.extend(b"1");
|
|
|
|
let frame = extract(Parser::parse(&mut buf, false, 1024));
|
|
assert!(!frame.finished);
|
|
assert_eq!(frame.opcode, OpCode::Text);
|
|
assert_eq!(frame.payload.as_ref(), &b"1"[..]);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_length0() {
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 0b0000_0000u8][..]);
|
|
let frame = extract(Parser::parse(&mut buf, false, 1024));
|
|
assert!(!frame.finished);
|
|
assert_eq!(frame.opcode, OpCode::Text);
|
|
assert!(frame.payload.is_empty());
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_length2() {
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 126u8][..]);
|
|
assert!(is_none(&Parser::parse(&mut buf, false, 1024)));
|
|
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 126u8][..]);
|
|
buf.extend(&[0u8, 4u8][..]);
|
|
buf.extend(b"1234");
|
|
|
|
let frame = extract(Parser::parse(&mut buf, false, 1024));
|
|
assert!(!frame.finished);
|
|
assert_eq!(frame.opcode, OpCode::Text);
|
|
assert_eq!(frame.payload.as_ref(), &b"1234"[..]);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_length4() {
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 127u8][..]);
|
|
assert!(is_none(&Parser::parse(&mut buf, false, 1024)));
|
|
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 127u8][..]);
|
|
buf.extend(&[0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 4u8][..]);
|
|
buf.extend(b"1234");
|
|
|
|
let frame = extract(Parser::parse(&mut buf, false, 1024));
|
|
assert!(!frame.finished);
|
|
assert_eq!(frame.opcode, OpCode::Text);
|
|
assert_eq!(frame.payload.as_ref(), &b"1234"[..]);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_frame_mask() {
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 0b1000_0001u8][..]);
|
|
buf.extend(b"0001");
|
|
buf.extend(b"1");
|
|
|
|
assert!(Parser::parse(&mut buf, false, 1024).is_err());
|
|
|
|
let frame = extract(Parser::parse(&mut buf, true, 1024));
|
|
assert!(!frame.finished);
|
|
assert_eq!(frame.opcode, OpCode::Text);
|
|
assert_eq!(frame.payload, Bytes::from(vec![1u8]));
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_frame_no_mask() {
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 0b0000_0001u8][..]);
|
|
buf.extend(&[1u8]);
|
|
|
|
assert!(Parser::parse(&mut buf, true, 1024).is_err());
|
|
|
|
let frame = extract(Parser::parse(&mut buf, false, 1024));
|
|
assert!(!frame.finished);
|
|
assert_eq!(frame.opcode, OpCode::Text);
|
|
assert_eq!(frame.payload, Bytes::from(vec![1u8]));
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_frame_max_size() {
|
|
let mut buf = BytesMut::from(&[0b0000_0001u8, 0b0000_0010u8][..]);
|
|
buf.extend(&[1u8, 1u8]);
|
|
|
|
assert!(Parser::parse(&mut buf, true, 1).is_err());
|
|
|
|
if let Err(ProtocolError::Overflow) = Parser::parse(&mut buf, false, 0) {
|
|
} else {
|
|
unreachable!("error");
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_ping_frame() {
|
|
let mut buf = BytesMut::new();
|
|
Parser::write_message(&mut buf, Vec::from("data"), OpCode::Ping, true, false);
|
|
|
|
let mut v = vec![137u8, 4u8];
|
|
v.extend(b"data");
|
|
assert_eq!(&buf[..], &v[..]);
|
|
}
|
|
|
|
#[test]
|
|
fn test_pong_frame() {
|
|
let mut buf = BytesMut::new();
|
|
Parser::write_message(&mut buf, Vec::from("data"), OpCode::Pong, true, false);
|
|
|
|
let mut v = vec![138u8, 4u8];
|
|
v.extend(b"data");
|
|
assert_eq!(&buf[..], &v[..]);
|
|
}
|
|
|
|
#[test]
|
|
fn test_close_frame() {
|
|
let mut buf = BytesMut::new();
|
|
let reason = (CloseCode::Normal, "data");
|
|
Parser::write_close(&mut buf, Some(reason.into()), false);
|
|
|
|
let mut v = vec![136u8, 6u8, 3u8, 232u8];
|
|
v.extend(b"data");
|
|
assert_eq!(&buf[..], &v[..]);
|
|
}
|
|
|
|
#[test]
|
|
fn test_empty_close_frame() {
|
|
let mut buf = BytesMut::new();
|
|
Parser::write_close(&mut buf, None, false);
|
|
assert_eq!(&buf[..], &vec![0x88, 0x00][..]);
|
|
}
|
|
}
|