frame: cleanup frame API

Signed-off-by: Alexey Galakhov <agalakhov@snapview.de>
pull/43/head
Alexey Galakhov 7 years ago
parent 75aa0d54f3
commit 50d1371e52
  1. 131
      src/protocol/frame/frame.rs
  2. 17
      src/protocol/mod.rs

@ -50,7 +50,7 @@ pub struct FrameHeader {
/// WebSocket protocol opcode. /// WebSocket protocol opcode.
pub opcode: OpCode, pub opcode: OpCode,
/// A frame mask, if any. /// A frame mask, if any.
pub mask: Option<[u8; 4]>, mask: Option<[u8; 4]>,
} }
impl Default for FrameHeader { impl Default for FrameHeader {
@ -120,6 +120,13 @@ impl FrameHeader {
Ok(()) Ok(())
} }
/// Generate a random frame mask and store this in the header.
///
/// Of course this does not change frame contents. It just generates a mask.
pub(crate) fn set_random_mask(&mut self) {
self.mask = Some(generate_mask())
}
} }
impl FrameHeader { impl FrameHeader {
@ -218,34 +225,16 @@ impl Frame {
self.header.len(length as u64) + length self.header.len(length as u64) + length
} }
/// Test whether the frame is a final frame. /// Get a reference to the frame's header.
#[inline]
pub fn is_final(&self) -> bool {
self.header.is_final
}
/// Test whether the first reserved bit is set.
#[inline]
pub fn has_rsv1(&self) -> bool {
self.header.rsv1
}
/// Test whether the second reserved bit is set.
#[inline]
pub fn has_rsv2(&self) -> bool {
self.header.rsv2
}
/// Test whether the third reserved bit is set.
#[inline] #[inline]
pub fn has_rsv3(&self) -> bool { pub fn header(&self) -> &FrameHeader {
self.header.rsv3 &self.header
} }
/// Get the OpCode of the frame. /// Get a mutable reference to the frame's header.
#[inline] #[inline]
pub fn opcode(&self) -> OpCode { pub fn header_mut(&mut self) -> &mut FrameHeader {
self.header.opcode &mut self.header
} }
/// Get a reference to the frame's payload. /// Get a reference to the frame's payload.
@ -254,86 +243,34 @@ impl Frame {
&self.payload &self.payload
} }
// Test whether the frame is masked. /// Get a mutable reference to the frame's payload.
#[doc(hidden)]
#[inline]
pub fn is_masked(&self) -> bool {
self.header.mask.is_some()
}
// Get an optional reference to the frame's mask.
#[doc(hidden)]
#[allow(dead_code)]
#[inline]
pub fn mask(&self) -> Option<&[u8; 4]> {
self.header.mask.as_ref()
}
/// Make this frame a final frame.
#[allow(dead_code)]
#[inline] #[inline]
pub fn set_final(&mut self, is_final: bool) -> &mut Frame { pub fn payload_mut(&mut self) -> &mut Vec<u8> {
self.header.is_final = is_final; &mut self.payload
self
}
/// Set the first reserved bit.
#[inline]
pub fn set_rsv1(&mut self, has_rsv1: bool) -> &mut Frame {
self.header.rsv1 = has_rsv1;
self
}
/// Set the second reserved bit.
#[inline]
pub fn set_rsv2(&mut self, has_rsv2: bool) -> &mut Frame {
self.header.rsv2 = has_rsv2;
self
}
/// Set the third reserved bit.
#[inline]
pub fn set_rsv3(&mut self, has_rsv3: bool) -> &mut Frame {
self.header.rsv3 = has_rsv3;
self
}
/// Set the OpCode.
#[allow(dead_code)]
#[inline]
pub fn set_opcode(&mut self, opcode: OpCode) -> &mut Frame {
self.header.opcode = opcode;
self
} }
/// Edit the frame's payload. /// Test whether the frame is masked.
#[allow(dead_code)]
#[inline] #[inline]
pub fn payload_mut(&mut self) -> &mut Vec<u8> { pub(crate) fn is_masked(&self) -> bool {
&mut self.payload self.header.mask.is_some()
} }
// Generate a new mask for this frame. /// Generate a random mask for the frame.
// ///
// This method simply generates and stores the mask. It does not change the payload data. /// This just generates a mask, payload is not changed. The actual masking is performed
// Instead, the payload data will be masked with the generated mask when the frame is sent /// either on `format()` or on `apply_mask()` call.
// to the other endpoint.
#[doc(hidden)]
#[inline] #[inline]
pub fn set_mask(&mut self) -> &mut Frame { pub(crate) fn set_random_mask(&mut self) {
self.header.mask = Some(generate_mask()); self.header.set_random_mask()
self
} }
// This method unmasks the payload and should only be called on frames that are actually /// This method unmasks the payload and should only be called on frames that are actually
// masked. In other words, those frames that have just been received from a client endpoint. /// masked. In other words, those frames that have just been received from a client endpoint.
#[doc(hidden)]
#[inline] #[inline]
pub fn remove_mask(&mut self) { pub(crate) fn apply_mask(&mut self) {
self.header.mask.and_then(|mask| { if let Some(mask) = self.header.mask.take() {
Some(apply_mask(&mut self.payload, &mask)) apply_mask(&mut self.payload, &mask)
}); }
self.header.mask = None;
} }
/// Consume the frame into its payload as binary. /// Consume the frame into its payload as binary.
@ -350,7 +287,7 @@ impl Frame {
/// Consume the frame into a closing frame. /// Consume the frame into a closing frame.
#[inline] #[inline]
pub fn into_close(self) -> Result<Option<CloseFrame<'static>>> { pub(crate) fn into_close(self) -> Result<Option<CloseFrame<'static>>> {
match self.payload.len() { match self.payload.len() {
0 => Ok(None), 0 => Ok(None),
1 => Err(Error::Protocol("Invalid close sequence".into())), 1 => Err(Error::Protocol("Invalid close sequence".into())),
@ -435,7 +372,7 @@ impl Frame {
/// Write a frame out to a buffer /// Write a frame out to a buffer
pub fn format(mut self, output: &mut impl Write) -> Result<()> { pub fn format(mut self, output: &mut impl Write) -> Result<()> {
self.header.format(self.payload.len() as u64, output)?; self.header.format(self.payload.len() as u64, output)?;
self.remove_mask(); self.apply_mask();
output.write_all(self.payload())?; output.write_all(self.payload())?;
Ok(()) Ok(())
} }

@ -246,8 +246,11 @@ impl<Stream: Read + Write> WebSocket<Stream> {
// the negotiated extensions defines the meaning of such a nonzero // the negotiated extensions defines the meaning of such a nonzero
// value, the receiving endpoint MUST _Fail the WebSocket // value, the receiving endpoint MUST _Fail the WebSocket
// Connection_. // Connection_.
if frame.has_rsv1() || frame.has_rsv2() || frame.has_rsv3() { {
return Err(Error::Protocol("Reserved bits are non-zero".into())) let hdr = frame.header();
if hdr.rsv1 || hdr.rsv2 || hdr.rsv3 {
return Err(Error::Protocol("Reserved bits are non-zero".into()))
}
} }
match self.role { match self.role {
@ -255,7 +258,7 @@ impl<Stream: Read + Write> WebSocket<Stream> {
if frame.is_masked() { if frame.is_masked() {
// A server MUST remove masking for data frames received from a client // A server MUST remove masking for data frames received from a client
// as described in Section 5.3. (RFC 6455) // as described in Section 5.3. (RFC 6455)
frame.remove_mask() frame.apply_mask()
} else { } else {
// The server MUST close the connection upon receiving a // The server MUST close the connection upon receiving a
// frame that is not masked. (RFC 6455) // frame that is not masked. (RFC 6455)
@ -270,13 +273,13 @@ impl<Stream: Read + Write> WebSocket<Stream> {
} }
} }
match frame.opcode() { match frame.header().opcode {
OpCode::Control(ctl) => { OpCode::Control(ctl) => {
match ctl { match ctl {
// All control frames MUST have a payload length of 125 bytes or less // All control frames MUST have a payload length of 125 bytes or less
// and MUST NOT be fragmented. (RFC 6455) // and MUST NOT be fragmented. (RFC 6455)
_ if !frame.is_final() => { _ if !frame.header().is_final => {
Err(Error::Protocol("Fragmented control frame".into())) Err(Error::Protocol("Fragmented control frame".into()))
} }
_ if frame.payload().len() > 125 => { _ if frame.payload().len() > 125 => {
@ -309,7 +312,7 @@ impl<Stream: Read + Write> WebSocket<Stream> {
} }
OpCode::Data(data) => { OpCode::Data(data) => {
let fin = frame.is_final(); let fin = frame.header().is_final;
match data { match data {
OpData::Continue => { OpData::Continue => {
if let Some(ref mut msg) = self.incomplete { if let Some(ref mut msg) = self.incomplete {
@ -423,7 +426,7 @@ impl<Stream: Read + Write> WebSocket<Stream> {
Role::Client => { Role::Client => {
// 5. If the data is being sent by the client, the frame(s) MUST be // 5. If the data is being sent by the client, the frame(s) MUST be
// masked as defined in Section 5.3. (RFC 6455) // masked as defined in Section 5.3. (RFC 6455)
frame.set_mask(); frame.set_random_mask();
} }
} }
let res = self.socket.write_frame(frame); let res = self.socket.write_frame(frame);

Loading…
Cancel
Save