2017-10-30 10:22:38 +01:00
|
|
|
// Copyright 2017 Parity Technologies (UK) Ltd.
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
|
|
// to deal in the Software without restriction, including without limitation
|
|
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
|
|
|
|
//! Defines the `SecioError` enum that groups all possible errors in SECIO.
|
|
|
|
|
2018-09-17 10:05:37 +02:00
|
|
|
use aes_ctr::stream_cipher::LoopError;
|
2018-10-08 14:37:36 +02:00
|
|
|
use protobuf::error::ProtobufError;
|
2017-10-30 10:22:38 +01:00
|
|
|
use std::error;
|
|
|
|
use std::fmt;
|
|
|
|
use std::io::Error as IoError;
|
|
|
|
|
|
|
|
/// Error at the SECIO layer communication.
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum SecioError {
|
2018-03-07 16:20:55 +01:00
|
|
|
/// I/O error.
|
|
|
|
IoError(IoError),
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-10-08 14:37:36 +02:00
|
|
|
/// Protocol buffer error.
|
|
|
|
Protobuf(ProtobufError),
|
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// Failed to parse one of the handshake protobuf messages.
|
|
|
|
HandshakeParsingFailure,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// There is no protocol supported by both the local and remote hosts.
|
2018-09-12 09:10:05 +02:00
|
|
|
NoSupportIntersection,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// Failed to generate nonce.
|
|
|
|
NonceGenerationFailed,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// Failed to generate ephemeral key.
|
|
|
|
EphemeralKeyGenerationFailed,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// Failed to sign a message with our local private key.
|
|
|
|
SigningFailure,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// The signature of the exchange packet doesn't verify the remote public key.
|
|
|
|
SignatureVerificationFailed,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// Failed to generate the secret shared key from the ephemeral key.
|
|
|
|
SecretGenerationFailed,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// The final check of the handshake failed.
|
|
|
|
NonceVerificationFailed,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-09-17 10:05:37 +02:00
|
|
|
/// Error with block cipher.
|
|
|
|
CipherError(LoopError),
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// The received frame was of invalid length.
|
|
|
|
FrameTooShort,
|
2017-10-30 10:22:38 +01:00
|
|
|
|
2018-03-07 16:20:55 +01:00
|
|
|
/// The hashes of the message didn't match.
|
|
|
|
HmacNotMatching,
|
2018-09-14 11:37:40 +02:00
|
|
|
|
|
|
|
/// We received an invalid proposition from remote.
|
|
|
|
InvalidProposition(&'static str),
|
2018-10-08 14:37:36 +02:00
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
__Nonexhaustive
|
2017-10-30 10:22:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl error::Error for SecioError {
|
2018-03-07 16:20:55 +01:00
|
|
|
fn cause(&self) -> Option<&error::Error> {
|
|
|
|
match *self {
|
|
|
|
SecioError::IoError(ref err) => Some(err),
|
2018-10-08 14:37:36 +02:00
|
|
|
SecioError::Protobuf(ref err) => Some(err),
|
2018-03-07 16:20:55 +01:00
|
|
|
// TODO: The type doesn't implement `Error`
|
|
|
|
/*SecioError::CipherError(ref err) => {
|
|
|
|
Some(err)
|
|
|
|
},*/
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
2017-10-30 10:22:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for SecioError {
|
2018-03-07 16:20:55 +01:00
|
|
|
#[inline]
|
2018-09-14 11:37:40 +02:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
|
|
match self {
|
|
|
|
SecioError::IoError(e) =>
|
|
|
|
write!(f, "I/O error: {}", e),
|
2018-10-08 14:37:36 +02:00
|
|
|
SecioError::Protobuf(e) =>
|
|
|
|
write!(f, "protobuf error: {}", e),
|
2018-09-14 11:37:40 +02:00
|
|
|
SecioError::HandshakeParsingFailure =>
|
|
|
|
f.write_str("Failed to parse one of the handshake protobuf messages"),
|
|
|
|
SecioError::NoSupportIntersection =>
|
|
|
|
f.write_str("There is no protocol supported by both the local and remote hosts"),
|
|
|
|
SecioError::NonceGenerationFailed =>
|
|
|
|
f.write_str("Failed to generate nonce"),
|
|
|
|
SecioError::EphemeralKeyGenerationFailed =>
|
|
|
|
f.write_str("Failed to generate ephemeral key"),
|
|
|
|
SecioError::SigningFailure =>
|
|
|
|
f.write_str("Failed to sign a message with our local private key"),
|
|
|
|
SecioError::SignatureVerificationFailed =>
|
|
|
|
f.write_str("The signature of the exchange packet doesn't verify the remote public key"),
|
|
|
|
SecioError::SecretGenerationFailed =>
|
|
|
|
f.write_str("Failed to generate the secret shared key from the ephemeral key"),
|
|
|
|
SecioError::NonceVerificationFailed =>
|
|
|
|
f.write_str("The final check of the handshake failed"),
|
|
|
|
SecioError::CipherError(e) =>
|
|
|
|
write!(f, "Error while decoding/encoding data: {:?}", e),
|
|
|
|
SecioError::FrameTooShort =>
|
|
|
|
f.write_str("The received frame was of invalid length"),
|
|
|
|
SecioError::HmacNotMatching =>
|
|
|
|
f.write_str("The hashes of the message didn't match"),
|
|
|
|
SecioError::InvalidProposition(msg) =>
|
2018-10-08 14:37:36 +02:00
|
|
|
write!(f, "invalid proposition: {}", msg),
|
|
|
|
SecioError::__Nonexhaustive =>
|
|
|
|
f.write_str("__Nonexhaustive")
|
2018-09-14 11:37:40 +02:00
|
|
|
}
|
2018-03-07 16:20:55 +01:00
|
|
|
}
|
2017-10-30 10:22:38 +01:00
|
|
|
}
|
|
|
|
|
2018-09-17 10:05:37 +02:00
|
|
|
impl From<LoopError> for SecioError {
|
2018-03-07 16:20:55 +01:00
|
|
|
#[inline]
|
2018-09-17 10:05:37 +02:00
|
|
|
fn from(err: LoopError) -> SecioError {
|
2018-03-07 16:20:55 +01:00
|
|
|
SecioError::CipherError(err)
|
|
|
|
}
|
2017-10-30 10:22:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl From<IoError> for SecioError {
|
2018-03-07 16:20:55 +01:00
|
|
|
#[inline]
|
|
|
|
fn from(err: IoError) -> SecioError {
|
|
|
|
SecioError::IoError(err)
|
|
|
|
}
|
2017-10-30 10:22:38 +01:00
|
|
|
}
|
2018-10-08 14:37:36 +02:00
|
|
|
|
|
|
|
impl From<ProtobufError> for SecioError {
|
|
|
|
#[inline]
|
|
|
|
fn from(err: ProtobufError) -> SecioError {
|
|
|
|
SecioError::Protobuf(err)
|
|
|
|
}
|
|
|
|
}
|