From 3f5c64bec0810b540741161c8c64bb710e324262 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marcel=20M=C3=A4rtens?= Date: Mon, 22 Feb 2021 00:48:30 +0100 Subject: [PATCH] Client::new can now resolve DNS requests, better networking error messages --- Cargo.lock | 2 +- assets/voxygen/i18n/en/main.ron | 1 + client/examples/chat-cli/main.rs | 9 +- client/src/addr.rs | 169 +++++++++++++++++++++++++++ client/src/error.rs | 4 +- client/src/lib.rs | 60 +++++++--- network/protocol/Cargo.toml | 2 +- network/protocol/src/error.rs | 54 +++++++++ network/protocol/src/handshake.rs | 3 +- network/protocol/src/lib.rs | 26 +---- network/protocol/src/mpsc.rs | 3 +- network/protocol/src/tcp.rs | 6 +- network/src/api.rs | 37 ++++-- network/src/lib.rs | 5 +- network/src/scheduler.rs | 33 ++---- server/src/chunk_generator.rs | 2 +- server/src/events/player.rs | 2 +- server/src/lib.rs | 2 +- voxygen/src/menu/main/client_init.rs | 132 +++++++-------------- voxygen/src/menu/main/mod.rs | 29 +++-- 20 files changed, 394 insertions(+), 187 deletions(-) create mode 100644 client/src/addr.rs create mode 100644 network/protocol/src/error.rs diff --git a/Cargo.lock b/Cargo.lock index c556d660ef..d394200d4c 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -5681,7 +5681,7 @@ dependencies = [ [[package]] name = "veloren-network-protocol" -version = "0.5.0" +version = "0.6.0" dependencies = [ "async-channel", "async-trait", diff --git a/assets/voxygen/i18n/en/main.ron b/assets/voxygen/i18n/en/main.ron index 4f0c73ef12..85ebd34e9e 100644 --- a/assets/voxygen/i18n/en/main.ron +++ b/assets/voxygen/i18n/en/main.ron @@ -48,6 +48,7 @@ https://veloren.net/account/."#, "main.login.server_shut_down": "Server shut down", "main.login.already_logged_in": "You are already logged into the server.", "main.login.network_error": "Network error", + "main.login.network_wrong_version": "The server is running a different version than you are. Check your version and update your game.", "main.login.failed_sending_request": "Request to Auth server failed", "main.login.invalid_character": "The selected character is invalid", "main.login.client_crashed": "Client crashed", diff --git a/client/examples/chat-cli/main.rs b/client/examples/chat-cli/main.rs index a89c0dab5f..a7dc7bb93c 100644 --- a/client/examples/chat-cli/main.rs +++ b/client/examples/chat-cli/main.rs @@ -5,14 +5,13 @@ use common::{clock::Clock, comp}; use std::{ io, - net::ToSocketAddrs, sync::{mpsc, Arc}, thread, time::Duration, }; use tokio::runtime::Runtime; use tracing::{error, info}; -use veloren_client::{Client, Event}; +use veloren_client::{addr::ConnectionArgs, Client, Event}; const TPS: u64 = 10; // Low value is okay, just reading messages. @@ -50,11 +49,7 @@ fn main() { // Create a client. let mut client = runtime .block_on(Client::new( - server_addr - .to_socket_addrs() - .expect("Invalid server address") - .next() - .unwrap(), + ConnectionArgs::HostnameAndOptionalPort(server_addr, false), None, runtime2, )) diff --git a/client/src/addr.rs b/client/src/addr.rs new file mode 100644 index 0000000000..a880658ea0 --- /dev/null +++ b/client/src/addr.rs @@ -0,0 +1,169 @@ +use std::net::SocketAddr; +use tokio::net::lookup_host; +use tracing::trace; + +#[derive(Clone, Debug)] +pub enum ConnectionArgs { + /// :[] + preferIpv6 flag + HostnameAndOptionalPort(String, bool), + IpAndPort(Vec), + Mpsc(u64), +} + +impl ConnectionArgs { + const DEFAULT_PORT: u16 = 14004; + + /// Parse ip address or resolves hostname, moves HostnameAndOptionalPort to + /// IpAndPort state. + /// Note: If you use an ipv6 address, the number after + /// the last colon will be used as the port unless you use [] around the + /// address. + pub async fn resolve(&mut self) -> Result<(), std::io::Error> { + if let ConnectionArgs::HostnameAndOptionalPort(server_address, prefer_ipv6) = self { + // 1. Try if server_address already contains a port + if let Ok(addr) = server_address.parse::() { + trace!("Server address with port found"); + *self = ConnectionArgs::IpAndPort(vec![addr]); + return Ok(()); + } + + // 2, Try server_address and port + if let Ok(addr) = + format!("{}:{}", server_address, Self::DEFAULT_PORT).parse::() + { + trace!("Server address without port found"); + *self = ConnectionArgs::IpAndPort(vec![addr]); + return Ok(()); + } + + // 3. Assert it's a hostname + port + let new = match lookup_host(server_address.to_string()).await { + Ok(s) => { + trace!("Host lookup succeeded"); + Ok(Self::sort_ipv6(s, *prefer_ipv6)) + }, + Err(e) => { + // 4. Assume its a hostname without port + match lookup_host((server_address.to_string(), Self::DEFAULT_PORT)).await { + Ok(s) => { + trace!("Host lookup without ports succeeded"); + Ok(Self::sort_ipv6(s, *prefer_ipv6)) + }, + Err(_) => Err(e), + } + }, + }?; + + *self = new; + } + Ok(()) + } + + fn sort_ipv6(s: impl Iterator, prefer_ipv6: bool) -> Self { + let (mut first_addrs, mut second_addrs) = + s.partition::, _>(|a| a.is_ipv6() == prefer_ipv6); + let addr = std::iter::Iterator::chain(first_addrs.drain(..), second_addrs.drain(..)) + .collect::>(); + ConnectionArgs::IpAndPort(addr) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use std::net::{IpAddr, Ipv4Addr, Ipv6Addr}; + + #[tokio::test] + async fn keep_mpcs() { + let mut args = ConnectionArgs::Mpsc(1337); + assert!(args.resolve().await.is_ok()); + assert!(matches!(args, ConnectionArgs::Mpsc(1337))); + } + + #[tokio::test] + async fn keep_ip() { + let mut args = ConnectionArgs::IpAndPort(vec![SocketAddr::new( + IpAddr::V4(Ipv4Addr::LOCALHOST), + ConnectionArgs::DEFAULT_PORT, + )]); + assert!(args.resolve().await.is_ok()); + assert!(matches!(args, ConnectionArgs::IpAndPort(..))); + } + + #[tokio::test] + async fn resolve_localhost() { + let mut args = ConnectionArgs::HostnameAndOptionalPort("localhost".to_string(), false); + assert!(args.resolve().await.is_ok()); + if let ConnectionArgs::IpAndPort(args) = args { + assert!(args.len() == 1 || args.len() == 2); + assert_eq!(args[0].ip(), IpAddr::V4(Ipv4Addr::LOCALHOST)); + assert_eq!(args[0].port(), 14004); + } else { + panic!("wrong resolution"); + } + + let mut args = ConnectionArgs::HostnameAndOptionalPort("localhost:666".to_string(), false); + assert!(args.resolve().await.is_ok()); + if let ConnectionArgs::IpAndPort(args) = args { + assert!(args.len() == 1 || args.len() == 2); + assert_eq!(args[0].port(), 666); + } else { + panic!("wrong resolution"); + } + } + + #[tokio::test] + async fn resolve_ipv6() { + let mut args = ConnectionArgs::HostnameAndOptionalPort("localhost".to_string(), true); + assert!(args.resolve().await.is_ok()); + if let ConnectionArgs::IpAndPort(args) = args { + assert!(args.len() == 1 || args.len() == 2); + assert_eq!(args[0].ip(), Ipv6Addr::LOCALHOST); + assert_eq!(args[0].port(), 14004); + } else { + panic!("wrong resolution"); + } + } + + #[tokio::test] + async fn resolve() { + let mut args = ConnectionArgs::HostnameAndOptionalPort("google.com".to_string(), false); + assert!(args.resolve().await.is_ok()); + if let ConnectionArgs::IpAndPort(args) = args { + assert!(args.len() == 1 || args.len() == 2); + assert_eq!(args[0].port(), 14004); + } else { + panic!("wrong resolution"); + } + + let mut args = ConnectionArgs::HostnameAndOptionalPort("127.0.0.1".to_string(), false); + assert!(args.resolve().await.is_ok()); + if let ConnectionArgs::IpAndPort(args) = args { + assert_eq!(args.len(), 1); + assert_eq!(args[0].port(), 14004); + assert_eq!(args[0].ip(), IpAddr::V4(Ipv4Addr::LOCALHOST)); + } else { + panic!("wrong resolution"); + } + + let mut args = ConnectionArgs::HostnameAndOptionalPort("55.66.77.88".to_string(), false); + assert!(args.resolve().await.is_ok()); + if let ConnectionArgs::IpAndPort(args) = args { + assert_eq!(args.len(), 1); + assert_eq!(args[0].port(), 14004); + assert_eq!(args[0].ip(), IpAddr::V4(Ipv4Addr::new(55, 66, 77, 88))); + } else { + panic!("wrong resolution"); + } + + let mut args = ConnectionArgs::HostnameAndOptionalPort("127.0.0.1:776".to_string(), false); + assert!(args.resolve().await.is_ok()); + if let ConnectionArgs::IpAndPort(args) = args { + assert_eq!(args.len(), 1); + assert_eq!(args[0].port(), 776); + assert_eq!(args[0].ip(), IpAddr::V4(Ipv4Addr::LOCALHOST)); + } else { + panic!("wrong resolution"); + } + } +} diff --git a/client/src/error.rs b/client/src/error.rs index 1b1cf0d435..346dcbcbd2 100644 --- a/client/src/error.rs +++ b/client/src/error.rs @@ -1,5 +1,5 @@ use authc::AuthClientError; -pub use network::NetworkError; +pub use network::{InitProtocolError, NetworkConnectError, NetworkError}; use network::{ParticipantError, StreamError}; #[derive(Debug)] @@ -19,6 +19,8 @@ pub enum Error { Banned(String), /// Persisted character data is invalid or missing InvalidCharacter, + /// is thrown when parsing the address or resolving Dns fails + DnsResolveFailed(String), //TODO: InvalidAlias, Other(String), } diff --git a/client/src/lib.rs b/client/src/lib.rs index 74d51b0d81..160e79eae6 100644 --- a/client/src/lib.rs +++ b/client/src/lib.rs @@ -1,7 +1,8 @@ #![deny(unsafe_code)] #![deny(clippy::clone_on_ref_ptr)] -#![feature(label_break_value, option_zip)] +#![feature(label_break_value, option_zip, str_split_once)] +pub mod addr; pub mod cmd; pub mod error; @@ -14,6 +15,7 @@ pub use specs::{ Builder, DispatcherBuilder, Entity as EcsEntity, ReadStorage, WorldExt, }; +use crate::addr::ConnectionArgs; use byteorder::{ByteOrder, LittleEndian}; use common::{ character::{CharacterId, CharacterItem}, @@ -57,7 +59,6 @@ use rayon::prelude::*; use specs::Component; use std::{ collections::VecDeque, - net::SocketAddr, sync::Arc, time::{Duration, Instant}, }; @@ -183,14 +184,40 @@ pub struct CharacterList { impl Client { /// Create a new `Client`. - pub async fn new>( - addr: A, + pub async fn new( + mut addr: ConnectionArgs, view_distance: Option, runtime: Arc, ) -> Result { let network = Network::new(Pid::new(), Arc::clone(&runtime)); - let participant = network.connect(ProtocolAddr::Tcp(addr.into())).await?; + if let Err(e) = addr.resolve().await { + error!(?e, "Dns resolve failed"); + return Err(Error::DnsResolveFailed(e.to_string())); + } + + let participant = match addr { + ConnectionArgs::HostnameAndOptionalPort(..) => { + unreachable!(".resolve() should have switched that state") + }, + ConnectionArgs::IpAndPort(addrs) => { + // Try to connect to all IP's and return the first that works + let mut participant = None; + for addr in addrs { + match network.connect(ProtocolAddr::Tcp(addr)).await { + Ok(p) => { + participant = Some(Ok(p)); + break; + }, + Err(e) => participant = Some(Err(Error::NetworkErr(e))), + } + } + participant + .unwrap_or_else(|| Err(Error::Other("No Ip Addr provided".to_string())))? + }, + ConnectionArgs::Mpsc(id) => network.connect(ProtocolAddr::Mpsc(id)).await?, + }; + let stream = participant.opened().await?; let mut ping_stream = participant.opened().await?; let mut register_stream = participant.opened().await?; @@ -2019,18 +2046,22 @@ impl Drop for Client { } else { trace!("no disconnect msg necessary as client wasn't registered") } - if let Err(e) = self - .runtime - .block_on(self.participant.take().unwrap().disconnect()) - { - warn!(?e, "error when disconnecting, couldn't send all data"); - } + + tokio::task::block_in_place(|| { + if let Err(e) = self + .runtime + .block_on(self.participant.take().unwrap().disconnect()) + { + warn!(?e, "error when disconnecting, couldn't send all data"); + } + }); } } #[cfg(test)] mod tests { use super::*; + use std::net::SocketAddr; #[test] /// THIS TEST VERIFIES THE CONSTANT API. @@ -2048,8 +2079,11 @@ mod tests { let view_distance: Option = None; let runtime = Arc::new(Runtime::new().unwrap()); let runtime2 = Arc::clone(&runtime); - let veloren_client: Result = - runtime.block_on(Client::new(socket, view_distance, runtime2)); + let veloren_client: Result = runtime.block_on(Client::new( + ConnectionArgs::IpAndPort(vec![socket]), + view_distance, + runtime2, + )); let _ = veloren_client.map(|mut client| { //register diff --git a/network/protocol/Cargo.toml b/network/protocol/Cargo.toml index 954c217fe0..6ab6e72496 100644 --- a/network/protocol/Cargo.toml +++ b/network/protocol/Cargo.toml @@ -1,7 +1,7 @@ [package] name = "veloren-network-protocol" description = "pure Protocol without any I/O itself" -version = "0.5.0" +version = "0.6.0" authors = ["Marcel Märtens "] edition = "2018" diff --git a/network/protocol/src/error.rs b/network/protocol/src/error.rs new file mode 100644 index 0000000000..089208b24f --- /dev/null +++ b/network/protocol/src/error.rs @@ -0,0 +1,54 @@ +/// All possible Errors that can happen during Handshake [`InitProtocol`] +/// +/// [`InitProtocol`]: crate::InitProtocol +#[derive(Debug, PartialEq)] +pub enum InitProtocolError { + Closed, + WrongMagicNumber([u8; 7]), + WrongVersion([u32; 3]), +} + +/// When you return closed you must stay closed! +#[derive(Debug, PartialEq)] +pub enum ProtocolError { + Closed, +} + +impl From for InitProtocolError { + fn from(err: ProtocolError) -> Self { + match err { + ProtocolError::Closed => InitProtocolError::Closed, + } + } +} + +impl core::fmt::Display for InitProtocolError { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + match self { + InitProtocolError::Closed => write!(f, "Channel closed"), + InitProtocolError::WrongMagicNumber(r) => write!( + f, + "Magic Number doesn't match, remote side send '{:?}' instead of '{:?}'", + &r, + &crate::types::VELOREN_MAGIC_NUMBER + ), + InitProtocolError::WrongVersion(r) => write!( + f, + "Network doesn't match, remote side send '{:?}' we are on '{:?}'", + &r, + &crate::types::VELOREN_NETWORK_VERSION + ), + } + } +} + +impl core::fmt::Display for ProtocolError { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + match self { + ProtocolError::Closed => write!(f, "Channel closed"), + } + } +} + +impl std::error::Error for InitProtocolError {} +impl std::error::Error for ProtocolError {} diff --git a/network/protocol/src/handshake.rs b/network/protocol/src/handshake.rs index bc672420d3..2c524d5241 100644 --- a/network/protocol/src/handshake.rs +++ b/network/protocol/src/handshake.rs @@ -1,10 +1,11 @@ use crate::{ + error::{InitProtocolError, ProtocolError}, frame::InitFrame, types::{ Pid, Sid, STREAM_ID_OFFSET1, STREAM_ID_OFFSET2, VELOREN_MAGIC_NUMBER, VELOREN_NETWORK_VERSION, }, - InitProtocol, InitProtocolError, ProtocolError, + InitProtocol, }; use async_trait::async_trait; use tracing::{debug, error, info, trace}; diff --git a/network/protocol/src/lib.rs b/network/protocol/src/lib.rs index 8896d7c447..de036cee76 100644 --- a/network/protocol/src/lib.rs +++ b/network/protocol/src/lib.rs @@ -49,6 +49,7 @@ //! [`RecvProtocol`]: crate::RecvProtocol //! [`InitProtocol`]: crate::InitProtocol +mod error; mod event; mod frame; mod handshake; @@ -59,6 +60,7 @@ mod prio; mod tcp; mod types; +pub use error::{InitProtocolError, ProtocolError}; pub use event::ProtocolEvent; pub use metrics::ProtocolMetricCache; #[cfg(feature = "metrics")] @@ -150,27 +152,3 @@ pub trait UnreliableSink: Send { type DataFormat; async fn recv(&mut self) -> Result; } - -/// All possible Errors that can happen during Handshake [`InitProtocol`] -/// -/// [`InitProtocol`]: crate::InitProtocol -#[derive(Debug, PartialEq)] -pub enum InitProtocolError { - Closed, - WrongMagicNumber([u8; 7]), - WrongVersion([u32; 3]), -} - -/// When you return closed you must stay closed! -#[derive(Debug, PartialEq)] -pub enum ProtocolError { - Closed, -} - -impl From for InitProtocolError { - fn from(err: ProtocolError) -> Self { - match err { - ProtocolError::Closed => InitProtocolError::Closed, - } - } -} diff --git a/network/protocol/src/mpsc.rs b/network/protocol/src/mpsc.rs index dca355fec2..5a00f27209 100644 --- a/network/protocol/src/mpsc.rs +++ b/network/protocol/src/mpsc.rs @@ -1,12 +1,13 @@ #[cfg(feature = "metrics")] use crate::metrics::RemoveReason; use crate::{ + error::ProtocolError, event::ProtocolEvent, frame::InitFrame, handshake::{ReliableDrain, ReliableSink}, metrics::ProtocolMetricCache, types::Bandwidth, - ProtocolError, RecvProtocol, SendProtocol, UnreliableDrain, UnreliableSink, + RecvProtocol, SendProtocol, UnreliableDrain, UnreliableSink, }; use async_trait::async_trait; use std::time::{Duration, Instant}; diff --git a/network/protocol/src/tcp.rs b/network/protocol/src/tcp.rs index 95c7a31215..c944674ad6 100644 --- a/network/protocol/src/tcp.rs +++ b/network/protocol/src/tcp.rs @@ -1,4 +1,5 @@ use crate::{ + error::ProtocolError, event::ProtocolEvent, frame::{ITFrame, InitFrame, OTFrame}, handshake::{ReliableDrain, ReliableSink}, @@ -6,7 +7,7 @@ use crate::{ metrics::{ProtocolMetricCache, RemoveReason}, prio::PrioManager, types::{Bandwidth, Mid, Sid}, - ProtocolError, RecvProtocol, SendProtocol, UnreliableDrain, UnreliableSink, + RecvProtocol, SendProtocol, UnreliableDrain, UnreliableSink, }; use async_trait::async_trait; use bytes::BytesMut; @@ -377,11 +378,12 @@ mod test_utils { #[cfg(test)] mod tests { use crate::{ + error::ProtocolError, frame::OTFrame, metrics::{ProtocolMetricCache, ProtocolMetrics, RemoveReason}, tcp::test_utils::*, types::{Pid, Promises, Sid, STREAM_ID_OFFSET1, STREAM_ID_OFFSET2}, - InitProtocol, ProtocolError, ProtocolEvent, RecvProtocol, SendProtocol, + InitProtocol, ProtocolEvent, RecvProtocol, SendProtocol, }; use bytes::{Bytes, BytesMut}; use std::{sync::Arc, time::Duration}; diff --git a/network/src/api.rs b/network/src/api.rs index c1110298b0..c71357a5fd 100644 --- a/network/src/api.rs +++ b/network/src/api.rs @@ -1,12 +1,12 @@ use crate::{ message::{partial_eq_bincode, Message}, participant::{A2bStreamOpen, S2bShutdownBparticipant}, - scheduler::Scheduler, + scheduler::{A2sConnect, Scheduler}, }; use bytes::Bytes; #[cfg(feature = "compression")] use lz_fear::raw::DecodeError; -use network_protocol::{Bandwidth, Pid, Prio, Promises, Sid}; +use network_protocol::{Bandwidth, InitProtocolError, Pid, Prio, Promises, Sid}; #[cfg(feature = "metrics")] use prometheus::Registry; use serde::{de::DeserializeOwned, Serialize}; @@ -83,7 +83,16 @@ pub struct Stream { pub enum NetworkError { NetworkClosed, ListenFailed(std::io::Error), - ConnectFailed(std::io::Error), + ConnectFailed(NetworkConnectError), +} + +/// Error type thrown by [`Networks`](Network) connect +#[derive(Debug)] +pub enum NetworkConnectError { + /// Either a Pid UUID clash or you are trying to hijack a connection + InvalidSecret, + Handshake(InitProtocolError), + Io(std::io::Error), } /// Error type thrown by [`Participants`](Participant) methods @@ -149,10 +158,8 @@ pub struct Network { local_pid: Pid, runtime: Arc, participant_disconnect_sender: Mutex>, - listen_sender: - Mutex>)>>, - connect_sender: - Mutex>)>>, + listen_sender: Mutex>)>>, + connect_sender: Mutex>, connected_receiver: Mutex>, shutdown_sender: Option>, } @@ -348,7 +355,8 @@ impl Network { /// [`ProtocolAddres`]: crate::api::ProtocolAddr #[instrument(name="network", skip(self, address), fields(p = %self.local_pid))] pub async fn connect(&self, address: ProtocolAddr) -> Result { - let (pid_sender, pid_receiver) = oneshot::channel::>(); + let (pid_sender, pid_receiver) = + oneshot::channel::>(); debug!(?address, "Connect to address"); self.connect_sender .lock() @@ -1147,6 +1155,18 @@ impl core::fmt::Display for NetworkError { } } +impl core::fmt::Display for NetworkConnectError { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + match self { + NetworkConnectError::Io(e) => write!(f, "Io error: {}", e), + NetworkConnectError::Handshake(e) => write!(f, "Handshake error: {}", e), + NetworkConnectError::InvalidSecret => { + write!(f, "You specified the wrong secret on your second channel") + }, + } + } +} + /// implementing PartialEq as it's super convenient in tests impl core::cmp::PartialEq for StreamError { fn eq(&self, other: &Self) -> bool { @@ -1177,3 +1197,4 @@ impl core::cmp::PartialEq for StreamError { impl std::error::Error for StreamError {} impl std::error::Error for ParticipantError {} impl std::error::Error for NetworkError {} +impl std::error::Error for NetworkConnectError {} diff --git a/network/src/lib.rs b/network/src/lib.rs index 4dd6b5eef7..e5c1545f32 100644 --- a/network/src/lib.rs +++ b/network/src/lib.rs @@ -107,7 +107,8 @@ mod participant; mod scheduler; pub use api::{ - Network, NetworkError, Participant, ParticipantError, ProtocolAddr, Stream, StreamError, + Network, NetworkConnectError, NetworkError, Participant, ParticipantError, ProtocolAddr, + Stream, StreamError, }; pub use message::Message; -pub use network_protocol::{Pid, Promises}; +pub use network_protocol::{InitProtocolError, Pid, Promises}; diff --git a/network/src/scheduler.rs b/network/src/scheduler.rs index c2ac365b0f..8aed480abf 100644 --- a/network/src/scheduler.rs +++ b/network/src/scheduler.rs @@ -1,5 +1,5 @@ use crate::{ - api::{Participant, ProtocolAddr}, + api::{NetworkConnectError, Participant, ProtocolAddr}, channel::Protocols, metrics::NetworkMetrics, participant::{B2sPrioStatistic, BParticipant, S2bCreateChannel, S2bShutdownBparticipant}, @@ -47,7 +47,10 @@ struct ParticipantInfo { } type A2sListen = (ProtocolAddr, oneshot::Sender>); -type A2sConnect = (ProtocolAddr, oneshot::Sender>); +pub(crate) type A2sConnect = ( + ProtocolAddr, + oneshot::Sender>, +); type A2sDisconnect = (Pid, S2bShutdownBparticipant); type S2sMpscConnect = ( mpsc::Sender, @@ -196,13 +199,7 @@ impl Scheduler { trace!("Stop listen_mgr"); } - async fn connect_mgr( - &self, - mut a2s_connect_r: mpsc::UnboundedReceiver<( - ProtocolAddr, - oneshot::Sender>, - )>, - ) { + async fn connect_mgr(&self, mut a2s_connect_r: mpsc::UnboundedReceiver) { trace!("Start connect_mgr"); while let Some((addr, pid_sender)) = a2s_connect_r.recv().await { let (protocol, cid, handshake) = match addr { @@ -215,7 +212,7 @@ impl Scheduler { let stream = match net::TcpStream::connect(addr).await { Ok(stream) => stream, Err(e) => { - pid_sender.send(Err(e)).unwrap(); + pid_sender.send(Err(NetworkConnectError::Io(e))).unwrap(); continue; }, }; @@ -232,10 +229,10 @@ impl Scheduler { Some(s) => s.clone(), None => { pid_sender - .send(Err(std::io::Error::new( + .send(Err(NetworkConnectError::Io(std::io::Error::new( std::io::ErrorKind::NotConnected, "no mpsc listen on this addr", - ))) + )))) .unwrap(); continue; }, @@ -543,7 +540,7 @@ impl Scheduler { &self, mut protocol: Protocols, cid: Cid, - s2a_return_pid_s: Option>>, + s2a_return_pid_s: Option>>, send_handshake: bool, ) { //channels are unknown till PID is known! @@ -647,10 +644,7 @@ impl Scheduler { if let Some(pid_oneshot) = s2a_return_pid_s { // someone is waiting with `connect`, so give them their Error pid_oneshot - .send(Err(std::io::Error::new( - std::io::ErrorKind::PermissionDenied, - "invalid secret, denying connection", - ))) + .send(Err(NetworkConnectError::InvalidSecret)) .unwrap(); } return; @@ -670,10 +664,7 @@ impl Scheduler { // someone is waiting with `connect`, so give them their Error trace!(?cid, "returning the Err to api who requested the connect"); pid_oneshot - .send(Err(std::io::Error::new( - std::io::ErrorKind::PermissionDenied, - "Handshake failed, denying connection", - ))) + .send(Err(NetworkConnectError::Handshake(e))) .unwrap(); } }, diff --git a/server/src/chunk_generator.rs b/server/src/chunk_generator.rs index 791b81ff81..556ba08bdd 100644 --- a/server/src/chunk_generator.rs +++ b/server/src/chunk_generator.rs @@ -40,7 +40,7 @@ impl ChunkGenerator { &mut self, entity: Option, key: Vec2, - runtime: &mut Arc, + runtime: &Runtime, world: Arc, index: IndexOwned, ) { diff --git a/server/src/events/player.rs b/server/src/events/player.rs index 9900a9fc3e..978309d4ff 100644 --- a/server/src/events/player.rs +++ b/server/src/events/player.rs @@ -12,7 +12,7 @@ use common::{ use common_net::msg::{PlayerListUpdate, PresenceKind, ServerGeneral}; use common_sys::state::State; use specs::{saveload::MarkerAllocator, Builder, Entity as EcsEntity, WorldExt}; -use tracing::*; +use tracing::{debug, error, trace, warn, Instrument}; pub fn handle_exit_ingame(server: &mut Server, entity: EcsEntity) { span!(_guard, "handle_exit_ingame"); diff --git a/server/src/lib.rs b/server/src/lib.rs index c4c2e7e164..da261fab5c 100644 --- a/server/src/lib.rs +++ b/server/src/lib.rs @@ -1008,7 +1008,7 @@ impl Server { .generate_chunk( Some(entity), key, - &mut self.runtime, + &self.runtime, Arc::clone(&self.world), self.index.clone(), ); diff --git a/voxygen/src/menu/main/client_init.rs b/voxygen/src/menu/main/client_init.rs index 4dca251c8c..98259664ed 100644 --- a/voxygen/src/menu/main/client_init.rs +++ b/voxygen/src/menu/main/client_init.rs @@ -1,27 +1,23 @@ use client::{ - error::{Error as ClientError, NetworkError}, + addr::ConnectionArgs, + error::{Error as ClientError, NetworkConnectError, NetworkError}, Client, }; use crossbeam::channel::{unbounded, Receiver, Sender, TryRecvError}; use std::{ - net::SocketAddr, sync::{ atomic::{AtomicBool, AtomicUsize, Ordering}, Arc, }, time::Duration, }; -use tokio::{net::lookup_host, runtime}; +use tokio::runtime; use tracing::{trace, warn}; #[derive(Debug)] pub enum Error { - // Error parsing input string or error resolving host name. - BadAddress(std::io::Error), - // Parsing/host name resolution successful but there was an error within the client. - ClientError(ClientError), - // Parsing yielded an empty iterator (specifically to_socket_addrs()). NoAddress, + ClientError(ClientError), ClientCrashed, } @@ -40,20 +36,17 @@ pub struct ClientInit { rx: Receiver, trust_tx: Sender, cancel: Arc, - _runtime: Arc, } impl ClientInit { #[allow(clippy::op_ref)] // TODO: Pending review in #587 #[allow(clippy::or_fun_call)] // TODO: Pending review in #587 pub fn new( - connection_args: (String, u16, bool), + connection_args: ConnectionArgs, username: String, view_distance: Option, password: String, runtime: Option>, ) -> Self { - let (server_address, port, prefer_ipv6) = connection_args; - let (tx, rx) = unbounded(); let (trust_tx, trust_rx) = unbounded(); let cancel = Arc::new(AtomicBool::new(false)); @@ -77,13 +70,14 @@ impl ClientInit { let runtime2 = Arc::clone(&runtime); runtime.spawn(async move { - let addresses = match Self::resolve(server_address, port, prefer_ipv6).await { - Ok(a) => a, - Err(e) => { - let _ = tx.send(Msg::Done(Err(Error::BadAddress(e)))); - return; - }, + let trust_fn = |auth_server: &str| { + let _ = tx.send(Msg::IsAuthTrusted(auth_server.to_string())); + trust_rx + .recv() + .map(|AuthTrust(server, trust)| trust && &server == auth_server) + .unwrap_or(false) }; + let mut last_err = None; const FOUR_MINUTES_RETRIES: u64 = 48; @@ -91,97 +85,51 @@ impl ClientInit { if cancel2.load(Ordering::Relaxed) { break; } - for socket_addr in &addresses { - match Client::new(*socket_addr, view_distance, Arc::clone(&runtime2)).await { - Ok(mut client) => { - if let Err(e) = client - .register(username, password, |auth_server| { - let _ = tx.send(Msg::IsAuthTrusted(auth_server.to_string())); - trust_rx - .recv() - .map(|AuthTrust(server, trust)| { - trust && &server == auth_server - }) - .unwrap_or(false) - }) - .await - { - last_err = Some(Error::ClientError(e)); - break 'tries; - } - let _ = tx.send(Msg::Done(Ok(client))); - return; - }, - Err(ClientError::NetworkErr(NetworkError::ConnectFailed(e))) => { - if e.kind() == std::io::ErrorKind::PermissionDenied { - warn!(?e, "Cannot connect to server: Incompatible version"); - last_err = Some(Error::ClientError(ClientError::NetworkErr( - NetworkError::ConnectFailed(e), - ))); - break 'tries; - } else { - warn!(?e, "Failed to connect to the server. Retrying..."); - } - }, - Err(e) => { - trace!(?e, "Aborting server connection attempt"); + match Client::new( + connection_args.clone(), + view_distance, + Arc::clone(&runtime2), + ) + .await + { + Ok(mut client) => { + if let Err(e) = client.register(username, password, trust_fn).await { last_err = Some(Error::ClientError(e)); break 'tries; - }, - } + } + let _ = tx.send(Msg::Done(Ok(client))); + return; + }, + Err(ClientError::NetworkErr(NetworkError::ConnectFailed( + NetworkConnectError::Io(e), + ))) => { + warn!(?e, "Failed to connect to the server. Retrying..."); + }, + Err(e) => { + trace!(?e, "Aborting server connection attempt"); + last_err = Some(Error::ClientError(e)); + break 'tries; + }, } tokio::time::sleep(Duration::from_secs(5)).await; } // Parsing/host name resolution successful but no connection succeeded. let _ = tx.send(Msg::Done(Err(last_err.unwrap_or(Error::NoAddress)))); + + //Safe drop runtime + tokio::task::block_in_place(move || { + drop(runtime2); + }); }); ClientInit { rx, trust_tx, cancel, - _runtime: runtime, } } - /// Parse ip address or resolves hostname. - /// Note: if you use an ipv6 address, the number after the last colon will - /// be used as the port unless you use [] around the address. - async fn resolve( - server_address: String, - port: u16, - prefer_ipv6: bool, - ) -> Result, std::io::Error> { - // 1. try if server_address already contains a port - if let Ok(addr) = server_address.parse::() { - warn!("please don't add port directly to server_address"); - return Ok(vec![addr]); - } - - // 2, try server_address and port - if let Ok(addr) = format!("{}:{}", server_address, port).parse::() { - return Ok(vec![addr]); - } - - // 3. do DNS call - let (mut first_addrs, mut second_addrs) = match lookup_host(server_address).await { - Ok(s) => s.partition::, _>(|a| a.is_ipv6() == prefer_ipv6), - Err(e) => { - return Err(e); - }, - }; - - Ok( - std::iter::Iterator::chain(first_addrs.drain(..), second_addrs.drain(..)) - .map(|mut addr| { - addr.set_port(port); - addr - }) - .collect(), - ) - } - /// Poll if the thread is complete. /// Returns None if the thread is still running, otherwise returns the /// Result of client creation. diff --git a/voxygen/src/menu/main/mod.rs b/voxygen/src/menu/main/mod.rs index 341422c137..eb4e076bd9 100644 --- a/voxygen/src/menu/main/mod.rs +++ b/voxygen/src/menu/main/mod.rs @@ -11,6 +11,10 @@ use crate::{ window::Event, Direction, GlobalState, PlayState, PlayStateResult, }; +use client::{ + addr::ConnectionArgs, + error::{InitProtocolError, NetworkConnectError, NetworkError}, +}; use client_init::{ClientInit, Error as InitError, Msg as InitMsg}; use common::{assets::AssetExt, comp, span}; use std::sync::Arc; @@ -34,8 +38,6 @@ impl MainMenuState { } } -const DEFAULT_PORT: u16 = 14004; - impl PlayState for MainMenuState { fn enter(&mut self, global_state: &mut GlobalState, _: Direction) { // Kick off title music @@ -74,8 +76,7 @@ impl PlayState for MainMenuState { &mut global_state.info_message, "singleplayer".to_owned(), "".to_owned(), - server_settings.gameserver_address.ip().to_string(), - server_settings.gameserver_address.port(), + ConnectionArgs::IpAndPort(vec![server_settings.gameserver_address]), &mut self.client_init, Some(runtime), ); @@ -122,10 +123,15 @@ impl PlayState for MainMenuState { self.client_init = None; global_state.info_message = Some({ let err = match err { - InitError::BadAddress(_) | InitError::NoAddress => { + InitError::NoAddress => { localized_strings.get("main.login.server_not_found").into() }, InitError::ClientError(err) => match err { + client::Error::DnsResolveFailed(reason) => format!( + "{}: {}", + localized_strings.get("main.login.server_not_found"), + reason + ), client::Error::AuthErr(e) => format!( "{}: {}", localized_strings.get("main.login.authentication_error"), @@ -160,6 +166,11 @@ impl PlayState for MainMenuState { client::Error::InvalidCharacter => { localized_strings.get("main.login.invalid_character").into() }, + client::Error::NetworkErr(NetworkError::ConnectFailed( + NetworkConnectError::Handshake(InitProtocolError::WrongVersion(_)), + )) => localized_strings + .get("main.login.network_wrong_version") + .into(), client::Error::NetworkErr(e) => format!( "{}: {:?}", localized_strings.get("main.login.network_error"), @@ -247,8 +258,7 @@ impl PlayState for MainMenuState { &mut global_state.info_message, username, password, - server_address, - DEFAULT_PORT, + ConnectionArgs::HostnameAndOptionalPort(server_address, false), &mut self.client_init, None, ); @@ -321,8 +331,7 @@ fn attempt_login( info_message: &mut Option, username: String, password: String, - server_address: String, - server_port: u16, + connection_args: ConnectionArgs, client_init: &mut Option, runtime: Option>, ) { @@ -330,7 +339,7 @@ fn attempt_login( // Don't try to connect if there is already a connection in progress. if client_init.is_none() { *client_init = Some(ClientInit::new( - (server_address, server_port, false), + connection_args, username, Some(settings.graphics.view_distance), password,