1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2025-04-04 17:19:35 +00:00

Improve close condition handling

This commit is contained in:
Heinz Gies 2019-09-10 16:47:37 +02:00 committed by Heinz N. Gies
parent 46ec5b77f9
commit c5dfead7f9
3 changed files with 36 additions and 5 deletions

View file

@ -184,7 +184,7 @@ impl Decoder for Codec {
OpCode::Bad => Err(ProtocolError::BadOpCode),
OpCode::Close => {
if let Some(ref pl) = payload {
let close_reason = Parser::parse_close_payload(pl);
let close_reason = Parser::parse_close_payload(pl)?;
Ok(Some(Frame::Close(close_reason)))
} else {
Ok(Some(Frame::Close(None)))

View file

@ -140,18 +140,23 @@ impl Parser {
}
/// Parse the payload of a close frame.
pub fn parse_close_payload(payload: &[u8]) -> Option<CloseReason> {
pub fn parse_close_payload(payload: &[u8]) -> Result<Option<CloseReason>, ProtocolError> {
if payload.len() >= 2 {
let raw_code = u16::from_be_bytes(TryFrom::try_from(&payload[..2]).unwrap());
let code = CloseCode::from(raw_code);
let description = if payload.len() > 2 {
Some(String::from_utf8_lossy(&payload[2..]).into())
if let Ok(desc) = String::from_utf8(payload[2..].to_vec()) {
Some(desc)
} else {
return Err(ProtocolError::BadEncoding)
}
} else {
None
};
Some(CloseReason { code, description })
Ok(Some(CloseReason { code, description }))
} else {
None
Ok(None)
}
}

View file

@ -128,6 +128,20 @@ pub enum CloseCode {
/// connect to a different IP (when multiple targets exist), or
/// reconnect to the same IP when a user has performed an action.
Again,
/// 3000-3999
///
/// Status codes in the range 3000-3999 are reserved for use by
/// libraries, frameworks, and applications. These status codes are
/// registered directly with IANA. The interpretation of these codes
/// is undefined by this protocol.
Library(u16),
/// 4000-4999
///
/// Status codes in the range 4000-4999 are reserved for private use
/// and thus can't be registered. Such codes can be used by prior
/// agreements between WebSocket applications. The interpretation of
/// these codes is undefined by this protocol.
Private(u16),
#[doc(hidden)]
Tls,
#[doc(hidden)]
@ -150,6 +164,8 @@ impl Into<u16> for CloseCode {
Restart => 1012,
Again => 1013,
Tls => 1015,
Library(code) => code,
Private(code) => code,
Other(code) => code,
}
}
@ -171,6 +187,8 @@ impl From<u16> for CloseCode {
1012 => Restart,
1013 => Again,
1015 => Tls,
3000..=3999 => Library(code),
4000..=4999 => Private(code),
_ => Other(code),
}
}
@ -293,6 +311,10 @@ mod test {
assert_eq!(CloseCode::from(1013u16), CloseCode::Again);
assert_eq!(CloseCode::from(1015u16), CloseCode::Tls);
assert_eq!(CloseCode::from(2000u16), CloseCode::Other(2000));
assert_eq!(CloseCode::from(3000u16), CloseCode::Library(3000));
assert_eq!(CloseCode::from(3999u16), CloseCode::Library(3999));
assert_eq!(CloseCode::from(4000u16), CloseCode::Private(4000));
assert_eq!(CloseCode::from(4999u16), CloseCode::Private(4999));
}
#[test]
@ -311,5 +333,9 @@ mod test {
assert_eq!(1013u16, Into::<u16>::into(CloseCode::Again));
assert_eq!(1015u16, Into::<u16>::into(CloseCode::Tls));
assert_eq!(2000u16, Into::<u16>::into(CloseCode::Other(2000)));
assert_eq!(3000u16, Into::<u16>::into(CloseCode::Library(3000)));
assert_eq!(3999u16, Into::<u16>::into(CloseCode::Library(3999)));
assert_eq!(4000u16, Into::<u16>::into(CloseCode::Private(4000)));
assert_eq!(4999u16, Into::<u16>::into(CloseCode::Private(4999)));
}
}