ip-echo-server: Don't use framed decoder, it can't be read-limited
This commit is contained in:
committed by
mergify[bot]
parent
328f59ebef
commit
6dc735e996
1
Cargo.lock
generated
1
Cargo.lock
generated
@ -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",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -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]
|
||||||
|
@ -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");
|
||||||
|
Reference in New Issue
Block a user