1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
// Copyright 2015-2019 Parity Technologies (UK) Ltd.
// This file is part of Parity Ethereum.

// Parity Ethereum is free software: you can redistribute it and/or modify
// 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.

// Parity Ethereum is distributed in the hope that it will be useful,
// 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
// along with Parity Ethereum.  If not, see <http://www.gnu.org/licenses/>.

use crate::crypto::{
    self, publickey::Error as CryptoPublicKeyError, Error as EthCryptoError,
};
use cfxkey::{self, DerivationError, Error as EthKeyError};
use std::{fmt, io::Error as IoError};

/// Account-related errors.
#[derive(Debug)]
pub enum Error {
    /// IO error
    Io(IoError),
    /// Invalid Password
    InvalidPassword,
    /// Account's secret is invalid.
    InvalidSecret,
    /// Invalid Vault Crypto meta.
    InvalidCryptoMeta,
    /// Invalid Account.
    InvalidAccount,
    /// Invalid Message.
    InvalidMessage,
    /// Invalid Key File
    InvalidKeyFile(String),
    /// Vaults are not supported.
    VaultsAreNotSupported,
    /// Unsupported vault
    UnsupportedVault,
    /// Invalid vault name
    InvalidVaultName,
    /// Vault not found
    VaultNotFound,
    /// Account creation failed.
    CreationFailed,
    /// `EthKey` error
    EthKey(EthKeyError),
    /// `cfxkey::crypto::Error`
    EthKeyCrypto(cfxkey::crypto::Error),
    /// `EthCrypto` error
    EthCrypto(EthCryptoError),
    /// Derivation error
    Derivation(DerivationError),
    /// Custom error
    Custom(String),
}

impl fmt::Display for Error {
    fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
        let s = match *self {
            Error::Io(ref err) => err.to_string(),
            Error::InvalidPassword => "Invalid password".into(),
            Error::InvalidSecret => "Invalid secret".into(),
            Error::InvalidCryptoMeta => "Invalid crypted metadata".into(),
            Error::InvalidAccount => "Invalid account".into(),
            Error::InvalidMessage => "Invalid message".into(),
            Error::InvalidKeyFile(ref reason) => {
                format!("Invalid key file: {}", reason)
            }
            Error::VaultsAreNotSupported => "Vaults are not supported".into(),
            Error::UnsupportedVault => {
                "Vault is not supported for this operation".into()
            }
            Error::InvalidVaultName => "Invalid vault name".into(),
            Error::VaultNotFound => "Vault not found".into(),
            Error::CreationFailed => "Account creation failed".into(),
            Error::EthKey(ref err) => err.to_string(),
            Error::EthKeyCrypto(ref err) => err.to_string(),
            Error::EthCrypto(ref err) => err.to_string(),
            Error::Derivation(ref err) => {
                format!("Derivation error: {:?}", err)
            }
            Error::Custom(ref s) => s.clone(),
        };

        write!(f, "{}", s)
    }
}

impl From<IoError> for Error {
    fn from(err: IoError) -> Self { Error::Io(err) }
}

impl From<EthKeyError> for Error {
    fn from(err: EthKeyError) -> Self { Error::EthKey(err) }
}

impl From<cfxkey::crypto::Error> for Error {
    fn from(err: cfxkey::crypto::Error) -> Self { Error::EthKeyCrypto(err) }
}

impl From<EthCryptoError> for Error {
    fn from(err: EthCryptoError) -> Self { Error::EthCrypto(err) }
}

impl From<crypto::error::ScryptError> for Error {
    fn from(err: crypto::error::ScryptError) -> Self {
        Error::EthCrypto(err.into())
    }
}

impl From<crypto::error::SymmError> for Error {
    fn from(err: crypto::error::SymmError) -> Self {
        Error::EthCrypto(err.into())
    }
}

impl From<DerivationError> for Error {
    fn from(err: DerivationError) -> Self { Error::Derivation(err) }
}

impl From<CryptoPublicKeyError> for Error {
    fn from(err: CryptoPublicKeyError) -> Self { Error::Custom(err.into()) }
}