|
|
@ -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] |
|
|
|
#[inline] |
|
|
|
pub fn has_rsv1(&self) -> bool { |
|
|
|
pub fn header(&self) -> &FrameHeader { |
|
|
|
self.header.rsv1 |
|
|
|
&self.header |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/// Test whether the second reserved bit is set.
|
|
|
|
/// Get a mutable reference to the frame's header.
|
|
|
|
#[inline] |
|
|
|
#[inline] |
|
|
|
pub fn has_rsv2(&self) -> bool { |
|
|
|
pub fn header_mut(&mut self) -> &mut FrameHeader { |
|
|
|
self.header.rsv2 |
|
|
|
&mut self.header |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// Test whether the third reserved bit is set.
|
|
|
|
|
|
|
|
#[inline] |
|
|
|
|
|
|
|
pub fn has_rsv3(&self) -> bool { |
|
|
|
|
|
|
|
self.header.rsv3 |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// Get the OpCode of the frame.
|
|
|
|
|
|
|
|
#[inline] |
|
|
|
|
|
|
|
pub fn opcode(&self) -> OpCode { |
|
|
|
|
|
|
|
self.header.opcode |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/// 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.
|
|
|
|
/// Test whether the frame is masked.
|
|
|
|
#[allow(dead_code)] |
|
|
|
|
|
|
|
#[inline] |
|
|
|
#[inline] |
|
|
|
pub fn set_opcode(&mut self, opcode: OpCode) -> &mut Frame { |
|
|
|
pub(crate) fn is_masked(&self) -> bool { |
|
|
|
self.header.opcode = opcode; |
|
|
|
self.header.mask.is_some() |
|
|
|
self |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/// Edit the frame's payload.
|
|
|
|
/// Generate a random mask for the frame.
|
|
|
|
#[allow(dead_code)] |
|
|
|
///
|
|
|
|
|
|
|
|
/// This just generates a mask, payload is not changed. The actual masking is performed
|
|
|
|
|
|
|
|
/// either on `format()` or on `apply_mask()` call.
|
|
|
|
#[inline] |
|
|
|
#[inline] |
|
|
|
pub fn payload_mut(&mut self) -> &mut Vec<u8> { |
|
|
|
pub(crate) fn set_random_mask(&mut self) { |
|
|
|
&mut self.payload |
|
|
|
self.header.set_random_mask() |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Generate a new mask for this frame.
|
|
|
|
/// 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.
|
|
|
|
// This method simply generates and stores the mask. It does not change the payload data.
|
|
|
|
|
|
|
|
// Instead, the payload data will be masked with the generated mask when the frame is sent
|
|
|
|
|
|
|
|
// to the other endpoint.
|
|
|
|
|
|
|
|
#[doc(hidden)] |
|
|
|
|
|
|
|
#[inline] |
|
|
|
#[inline] |
|
|
|
pub fn set_mask(&mut self) -> &mut Frame { |
|
|
|
pub(crate) fn apply_mask(&mut self) { |
|
|
|
self.header.mask = Some(generate_mask()); |
|
|
|
if let Some(mask) = self.header.mask.take() { |
|
|
|
self |
|
|
|
apply_mask(&mut self.payload, &mask) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
|
|
|
|
#[doc(hidden)] |
|
|
|
|
|
|
|
#[inline] |
|
|
|
|
|
|
|
pub fn remove_mask(&mut self) { |
|
|
|
|
|
|
|
self.header.mask.and_then(|mask| { |
|
|
|
|
|
|
|
Some(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(()) |
|
|
|
} |
|
|
|
} |
|
|
|