ip-echo-server: Don't use framed decoder, it can't be read-limited

This commit is contained in:
Trent Nelson
2020-11-12 23:13:29 -07:00
committed by mergify[bot]
parent 328f59ebef
commit 6dc735e996
3 changed files with 20 additions and 19 deletions

1
Cargo.lock generated
View File

@ -4444,7 +4444,6 @@ dependencies = [
"solana-logger 1.5.0", "solana-logger 1.5.0",
"solana-version", "solana-version",
"tokio 0.1.22", "tokio 0.1.22",
"tokio-codec",
"url 2.1.1", "url 2.1.1",
] ]

View File

@ -22,7 +22,6 @@ solana-clap-utils = { path = "../clap-utils", version = "1.5.0" }
solana-logger = { path = "../logger", version = "1.5.0" } solana-logger = { path = "../logger", version = "1.5.0" }
solana-version = { path = "../version", version = "1.5.0" } solana-version = { path = "../version", version = "1.5.0" }
tokio = "0.1" tokio = "0.1"
tokio-codec = "0.1"
url = "2.1.1" url = "2.1.1"
[lib] [lib]

View File

@ -4,7 +4,6 @@ use log::*;
use serde_derive::{Deserialize, Serialize}; use serde_derive::{Deserialize, Serialize};
use std::{io, net::SocketAddr, time::Duration}; use std::{io, net::SocketAddr, time::Duration};
use tokio::{net::TcpListener, prelude::*, reactor::Handle, runtime::Runtime}; use tokio::{net::TcpListener, prelude::*, reactor::Handle, runtime::Runtime};
use tokio_codec::{BytesCodec, Decoder};
pub type IpEchoServer = Runtime; pub type IpEchoServer = Runtime;
@ -28,6 +27,13 @@ impl IpEchoServerMessage {
} }
} }
pub(crate) fn ip_echo_server_request_length() -> usize {
const REQUEST_TERMINUS_LENGTH: usize = 1;
HEADER_LENGTH
+ bincode::serialized_size(&IpEchoServerMessage::default()).unwrap() as usize
+ REQUEST_TERMINUS_LENGTH
}
/// Starts a simple TCP server on the given port that echos the IP address of any peer that /// Starts a simple TCP server on the given port that echos the IP address of any peer that
/// connects. Used by |get_public_ip_addr| /// connects. Used by |get_public_ip_addr|
pub fn ip_echo_server(tcp: std::net::TcpListener) -> IpEchoServer { pub fn ip_echo_server(tcp: std::net::TcpListener) -> IpEchoServer {
@ -41,18 +47,19 @@ pub fn ip_echo_server(tcp: std::net::TcpListener) -> IpEchoServer {
.filter_map(|socket| match socket.peer_addr() { .filter_map(|socket| match socket.peer_addr() {
Ok(peer_addr) => { Ok(peer_addr) => {
info!("connection from {:?}", peer_addr); info!("connection from {:?}", peer_addr);
Some((peer_addr, BytesCodec::new().framed(socket))) Some((peer_addr, socket))
} }
Err(err) => { Err(err) => {
info!("peer_addr failed for {:?}: {:?}", socket, err); info!("peer_addr failed for {:?}: {:?}", socket, err);
None None
} }
}) })
.for_each(move |(peer_addr, framed)| { .for_each(move |(peer_addr, socket)| {
let (writer, reader) = framed.split(); let data = vec![0u8; ip_echo_server_request_length()];
let (reader, writer) = socket.split();
let processor = reader let processor = tokio::io::read_exact(reader, data)
.and_then(move |data| { .and_then(move |(_, data)| {
if data.len() < HEADER_LENGTH { if data.len() < HEADER_LENGTH {
return Err(io::Error::new( return Err(io::Error::new(
io::ErrorKind::Other, io::ErrorKind::Other,
@ -170,22 +177,18 @@ pub fn ip_echo_server(tcp: std::net::TcpListener) -> IpEchoServer {
} }
}) })
.and_then(move |valid_request| { .and_then(move |valid_request| {
if valid_request.is_none() { let bytes = if valid_request.is_none() {
Ok(Bytes::from( Bytes::from("HTTP/1.1 400 Bad Request\nContent-length: 0\n\n")
"HTTP/1.1 400 Bad Request\nContent-length: 0\n\n",
))
} else { } else {
// "\0\0\0\0" header is added to ensure a valid response will never // "\0\0\0\0" header is added to ensure a valid response will never
// conflict with the first four bytes of a valid HTTP response. // conflict with the first four bytes of a valid HTTP response.
let mut bytes = vec![0u8; ip_echo_server_reply_length()]; let mut bytes = vec![0u8; ip_echo_server_reply_length()];
bincode::serialize_into(&mut bytes[HEADER_LENGTH..], &peer_addr.ip()) bincode::serialize_into(&mut bytes[HEADER_LENGTH..], &peer_addr.ip())
.unwrap(); .unwrap();
Ok(Bytes::from(bytes)) Bytes::from(bytes)
} };
}); tokio::io::write_all(writer, bytes)
})
let connection = writer
.send_all(processor)
.timeout(Duration::from_secs(5)) .timeout(Duration::from_secs(5))
.then(|result| { .then(|result| {
if let Err(err) = result { if let Err(err) = result {
@ -194,7 +197,7 @@ pub fn ip_echo_server(tcp: std::net::TcpListener) -> IpEchoServer {
Ok(()) Ok(())
}); });
tokio::spawn(connection) tokio::spawn(processor)
}); });
let mut rt = Runtime::new().expect("Failed to create Runtime"); let mut rt = Runtime::new().expect("Failed to create Runtime");