Skip to content

Commit

Permalink
Fix multiple websocket connection issues.
Browse files Browse the repository at this point in the history
- The websocket client sends two Connection headers:

    HTTP/1.1 GET /
    Host: 127.0.0.1
    Origin: foo.com
    Content-Length: 0
    Connection: Upgrade
    Upgrade: websocket
    Sec-WebSocket-Version: 13
    Sec-WebSocket-Key: h57ZiKx+KAwmKbB+mxR8Ag==
    Connection: Keep-Alive

  The last "Connection: Keep-Alive" is added because the ConnectionType
  enum does not have an "Upgrade" variant. Adding the variant prevents
  this issue.

- If the websocket client does not receive a "Content-Length: 0" header
  in the server's response, an error is returned stating "Unknown body
  type in a response with a Keep-Alive connection. This is not allowed."
  This occurs with the example client using the
  "websockets.chilkat.io/" service. This is corrected by assuming a
  "BodyType::ContentLen(0)" for the response when no other BodyType is
  provided and the "Connection: Upgrade" header is present.
  • Loading branch information
bridadan committed Jan 20, 2025
1 parent b429fca commit bcde418
Showing 1 changed file with 23 additions and 0 deletions.
23 changes: 23 additions & 0 deletions edge-http/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -444,6 +444,7 @@ impl<const N: usize> Default for Headers<'_, N> {
pub enum ConnectionType {
KeepAlive,
Close,
Upgrade,
}

impl ConnectionType {
Expand Down Expand Up @@ -499,6 +500,8 @@ impl ConnectionType {
&& value.eq_ignore_ascii_case("Keep-Alive")
{
Some(Self::KeepAlive)
} else if "Connection".eq_ignore_ascii_case(name) && value.eq_ignore_ascii_case("Upgrade") {
Some(Self::Upgrade)
} else {
None
}
Expand Down Expand Up @@ -535,6 +538,7 @@ impl ConnectionType {
let connection = match self {
Self::KeepAlive => "Keep-Alive",
Self::Close => "Close",
Self::Upgrade => "Upgrade",
};

("Connection", connection.as_bytes())
Expand All @@ -546,6 +550,7 @@ impl Display for ConnectionType {
match self {
Self::KeepAlive => write!(f, "Keep-Alive"),
Self::Close => write!(f, "Close"),
Self::Upgrade => write!(f, "Upgrade"),
}
}
}
Expand Down Expand Up @@ -619,6 +624,16 @@ impl BodyType {
}
} else if matches!(connection_type, ConnectionType::Close) {
Ok(BodyType::Raw)
} else if matches!(connection_type, ConnectionType::Upgrade) {
if http11 {
debug!("Unknown body type in response but the Connection is Upgrade. Assuming Content-Length=0.");
Ok(BodyType::ContentLen(0))
} else {
warn!("Connection is set to Upgrade but the HTTP protocol version is not 1.1. This is not allowed.");
Err(HeadersMismatchError::BodyTypeError(
"Connection is set to Upgrade but the HTTP protocol version is not 1.1. This is not allowed.",
))
}
} else if chunked_if_unspecified && http11 {
// With HTTP1.1 we can safely upgrade the body to a chunked one
Ok(BodyType::Chunked)
Expand Down Expand Up @@ -1183,6 +1198,14 @@ mod test {
BodyType::resolve(None, ConnectionType::Close, true, false, false).unwrap(),
BodyType::ContentLen(0)
);
assert_eq!(
BodyType::resolve(None, ConnectionType::Upgrade, false, true, false).unwrap(),
BodyType::ContentLen(0)
);

// Receiving a response with no body type after requesting a connection upgrade with
// HTTP1.0 is no allowed.
assert!(BodyType::resolve(None, ConnectionType::Upgrade, false, false, false).is_err());

// Request or response with a chunked body type is invalid for HTTP1.0
assert!(BodyType::resolve(
Expand Down

0 comments on commit bcde418

Please sign in to comment.