2019-01-07 11:33:07 +01:00
// Copyright 2015-2019 Parity Technologies (UK) Ltd.
// This file is part of Parity Ethereum.
2016-02-05 13:40:41 +01:00
2019-01-07 11:33:07 +01:00
// Parity Ethereum is free software: you can redistribute it and/or modify
2016-02-05 13:40:41 +01:00
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
2019-01-07 11:33:07 +01:00
// Parity Ethereum is distributed in the hope that it will be useful,
2016-02-05 13:40:41 +01:00
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
2019-01-07 11:33:07 +01:00
// along with Parity Ethereum. If not, see <http://www.gnu.org/licenses/>.
2016-02-05 13:40:41 +01:00
2019-03-02 13:18:18 +01:00
// Silence: `use of deprecated item 'std::error::Error::cause': replaced by Error::source, which can support downcasting`
// https://github.com/paritytech/parity-ethereum/issues/10302
#![ allow(deprecated) ]
2017-11-13 14:37:08 +01:00
use std ::{ io , net , fmt } ;
2018-06-01 13:37:43 +02:00
use libc ::{ ENFILE , EMFILE } ;
2016-01-15 16:36:08 +01:00
use io ::IoError ;
2017-11-13 14:37:08 +01:00
use { rlp , ethkey , crypto , snappy } ;
2016-01-15 16:36:08 +01:00
2016-02-17 14:07:26 +01:00
#[ derive(Debug, Copy, Clone, PartialEq, Eq) ]
2016-01-15 16:36:08 +01:00
pub enum DisconnectReason
{
DisconnectRequested ,
2016-02-17 14:07:26 +01:00
TCPError ,
BadProtocol ,
2016-01-15 16:36:08 +01:00
UselessPeer ,
2016-02-17 14:07:26 +01:00
TooManyPeers ,
DuplicatePeer ,
IncompatibleProtocol ,
NullIdentity ,
ClientQuit ,
UnexpectedIdentity ,
LocalIdentity ,
2016-02-02 20:58:12 +01:00
PingTimeout ,
2016-02-17 14:07:26 +01:00
Unknown ,
}
impl DisconnectReason {
pub fn from_u8 ( n : u8 ) -> DisconnectReason {
match n {
0 = > DisconnectReason ::DisconnectRequested ,
1 = > DisconnectReason ::TCPError ,
2 = > DisconnectReason ::BadProtocol ,
3 = > DisconnectReason ::UselessPeer ,
4 = > DisconnectReason ::TooManyPeers ,
5 = > DisconnectReason ::DuplicatePeer ,
6 = > DisconnectReason ::IncompatibleProtocol ,
7 = > DisconnectReason ::NullIdentity ,
8 = > DisconnectReason ::ClientQuit ,
9 = > DisconnectReason ::UnexpectedIdentity ,
10 = > DisconnectReason ::LocalIdentity ,
11 = > DisconnectReason ::PingTimeout ,
_ = > DisconnectReason ::Unknown ,
}
}
2016-01-15 16:36:08 +01:00
}
2016-05-21 00:12:51 +02:00
impl fmt ::Display for DisconnectReason {
fn fmt ( & self , f : & mut fmt ::Formatter ) -> fmt ::Result {
use self ::DisconnectReason ::* ;
let msg = match * self {
DisconnectRequested = > " disconnect requested " ,
TCPError = > " TCP error " ,
BadProtocol = > " bad protocol " ,
UselessPeer = > " useless peer " ,
TooManyPeers = > " too many peers " ,
DuplicatePeer = > " duplicate peer " ,
IncompatibleProtocol = > " incompatible protocol " ,
NullIdentity = > " null identity " ,
ClientQuit = > " client quit " ,
UnexpectedIdentity = > " unexpected identity " ,
LocalIdentity = > " local identity " ,
PingTimeout = > " ping timeout " ,
Unknown = > " unknown " ,
} ;
f . write_str ( msg )
}
}
2017-11-13 14:37:08 +01:00
error_chain! {
foreign_links {
SocketIo ( IoError ) #[ doc = " Socket IO error. " ] ;
Decompression ( snappy ::InvalidInput ) #[ doc = " Decompression error. " ] ;
2018-10-09 22:07:25 +02:00
Rlp ( rlp ::DecoderError ) #[ doc = " Rlp decoder error. " ] ;
2017-11-13 14:37:08 +01:00
}
2016-01-15 16:36:08 +01:00
2017-11-13 14:37:08 +01:00
errors {
2018-05-09 15:58:02 +02:00
#[ doc = " Error concerning the network address parsing subsystem. " ]
AddressParse {
description ( " Failed to parse network address " ) ,
display ( " Failed to parse network address " ) ,
}
2017-11-13 14:37:08 +01:00
#[ doc = " Error concerning the network address resolution subsystem. " ]
AddressResolve ( err : Option < io ::Error > ) {
description ( " Failed to resolve network address " ) ,
display ( " Failed to resolve network address {} " , err . as_ref ( ) . map_or ( " " . to_string ( ) , | e | e . to_string ( ) ) ) ,
}
2016-05-21 00:12:51 +02:00
2017-11-13 14:37:08 +01:00
#[ doc = " Authentication failure " ]
Auth {
description ( " Authentication failure " ) ,
display ( " Authentication failure " ) ,
}
2016-05-21 00:12:51 +02:00
2017-11-13 14:37:08 +01:00
#[ doc = " Unrecognised protocol " ]
BadProtocol {
description ( " Bad protocol " ) ,
display ( " Bad protocol " ) ,
}
2016-05-21 00:12:51 +02:00
2017-11-13 14:37:08 +01:00
#[ doc = " Expired message " ]
Expired {
description ( " Expired message " ) ,
display ( " Expired message " ) ,
}
2016-01-15 16:36:08 +01:00
2017-11-13 14:37:08 +01:00
#[ doc = " Peer not found " ]
PeerNotFound {
description ( " Peer not found " ) ,
display ( " Peer not found " ) ,
}
2016-08-05 10:32:04 +02:00
2017-11-13 14:37:08 +01:00
#[ doc = " Peer is disconnected " ]
Disconnect ( reason : DisconnectReason ) {
description ( " Peer disconnected " ) ,
display ( " Peer disconnected: {} " , reason ) ,
}
2016-01-15 16:36:08 +01:00
2017-11-13 14:37:08 +01:00
#[ doc = " Invalid node id " ]
InvalidNodeId {
description ( " Invalid node id " ) ,
display ( " Invalid node id " ) ,
}
#[ doc = " Packet size is over the protocol limit " ]
OversizedPacket {
description ( " Packet is too large " ) ,
display ( " Packet is too large " ) ,
}
2018-06-01 13:37:43 +02:00
#[ doc = " Reached system resource limits for this process " ]
ProcessTooManyFiles {
description ( " Too many open files in process. " ) ,
display ( " Too many open files in this process. Check your resource limits and restart parity " ) ,
}
#[ doc = " Reached system wide resource limits " ]
SystemTooManyFiles {
description ( " Too many open files on system. " ) ,
display ( " Too many open files on system. Consider closing some processes/release some file handlers or increas the system-wide resource limits and restart parity. " ) ,
}
#[ doc = " An unknown IO error occurred. " ]
Io ( err : io ::Error ) {
description ( " IO Error " ) ,
display ( " Unexpected IO error: {} " , err ) ,
}
}
}
impl From < io ::Error > for Error {
fn from ( err : io ::Error ) -> Self {
match err . raw_os_error ( ) {
Some ( ENFILE ) = > ErrorKind ::ProcessTooManyFiles . into ( ) ,
Some ( EMFILE ) = > ErrorKind ::SystemTooManyFiles . into ( ) ,
_ = > Error ::from_kind ( ErrorKind ::Io ( err ) )
}
2016-08-24 18:35:21 +02:00
}
}
2017-11-13 14:37:08 +01:00
impl From < ethkey ::Error > for Error {
fn from ( _err : ethkey ::Error ) -> Self {
ErrorKind ::Auth . into ( )
2017-10-19 14:41:11 +02:00
}
}
2018-05-05 11:02:33 +02:00
impl From < ethkey ::crypto ::Error > for Error {
fn from ( _err : ethkey ::crypto ::Error ) -> Self {
ErrorKind ::Auth . into ( )
}
}
impl From < crypto ::error ::SymmError > for Error {
fn from ( _err : crypto ::error ::SymmError ) -> Self {
2017-11-13 14:37:08 +01:00
ErrorKind ::Auth . into ( )
2016-08-05 10:32:04 +02:00
}
}
2018-05-09 15:58:02 +02:00
impl From < net ::AddrParseError > for Error {
fn from ( _err : net ::AddrParseError ) -> Self { ErrorKind ::AddressParse . into ( ) }
}
2016-02-17 14:07:26 +01:00
#[ test ]
fn test_errors ( ) {
assert_eq! ( DisconnectReason ::ClientQuit , DisconnectReason ::from_u8 ( 8 ) ) ;
let mut r = DisconnectReason ::DisconnectRequested ;
for i in 0 .. 20 {
r = DisconnectReason ::from_u8 ( i ) ;
}
assert_eq! ( DisconnectReason ::Unknown , r ) ;
2017-11-13 14:37:08 +01:00
match * < Error as From < rlp ::DecoderError > > ::from ( rlp ::DecoderError ::RlpIsTooBig ) . kind ( ) {
2018-10-09 22:07:25 +02:00
ErrorKind ::Rlp ( _ ) = > { } ,
2018-05-05 11:02:33 +02:00
_ = > panic! ( " Unexpected error " ) ,
2016-02-17 14:07:26 +01:00
}
2018-05-05 11:02:33 +02:00
match * < Error as From < ethkey ::crypto ::Error > > ::from ( ethkey ::crypto ::Error ::InvalidMessage ) . kind ( ) {
2017-11-13 14:37:08 +01:00
ErrorKind ::Auth = > { } ,
2018-05-05 11:02:33 +02:00
_ = > panic! ( " Unexpected error " ) ,
2016-02-17 14:07:26 +01:00
}
}
2018-06-01 13:37:43 +02:00
#[ test ]
fn test_io_errors ( ) {
use libc ::{ EMFILE , ENFILE } ;
assert_matches! (
< Error as From < io ::Error > > ::from (
io ::Error ::from_raw_os_error ( ENFILE )
) . kind ( ) ,
ErrorKind ::ProcessTooManyFiles ) ;
assert_matches! (
< Error as From < io ::Error > > ::from (
io ::Error ::from_raw_os_error ( EMFILE )
) . kind ( ) ,
ErrorKind ::SystemTooManyFiles ) ;
assert_matches! (
< Error as From < io ::Error > > ::from (
io ::Error ::from_raw_os_error ( 0 )
) . kind ( ) ,
ErrorKind ::Io ( _ ) ) ;
}