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
use crate::types::{ScriptGroup, ScriptGroupType};
use ckb_error::{prelude::*, Error, ErrorKind};
use ckb_types::core::Cycle;
use std::{error::Error as StdError, fmt};
#[derive(Error, Debug, PartialEq, Eq, Clone)]
pub enum ScriptError {
#[error("InvalidCodeHash")]
InvalidCodeHash,
#[error("ExceededMaximumCycles: expect cycles <= {0}")]
ExceededMaximumCycles(Cycle),
#[error("MultipleMatches")]
MultipleMatches,
#[error("ValidationFailure({0}): the exit code is per script specific, for system scripts, please check https://github.com/nervosnetwork/ckb-system-scripts/wiki/Error-codes")]
ValidationFailure(i8),
#[error("Known bugs encountered in output {1}: {0}")]
EncounteredKnownBugs(String, usize),
#[error("VM Internal Error: {0}")]
VMInternalError(String),
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum TransactionScriptErrorSource {
Inputs(usize, ScriptGroupType),
Outputs(usize, ScriptGroupType),
Unknown,
}
impl TransactionScriptErrorSource {
fn from_script_group(script_group: &ScriptGroup) -> Self {
if let Some(n) = script_group.input_indices.first() {
TransactionScriptErrorSource::Inputs(*n, script_group.group_type)
} else if let Some(n) = script_group.output_indices.first() {
TransactionScriptErrorSource::Outputs(*n, script_group.group_type)
} else {
TransactionScriptErrorSource::Unknown
}
}
}
impl fmt::Display for TransactionScriptErrorSource {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
TransactionScriptErrorSource::Inputs(n, field) => write!(f, "Inputs[{}].{}", n, field),
TransactionScriptErrorSource::Outputs(n, field) => {
write!(f, "Outputs[{}].{}", n, field)
}
TransactionScriptErrorSource::Unknown => write!(f, "Unknown"),
}
}
}
#[derive(Debug, PartialEq, Eq, Clone)]
pub struct TransactionScriptError {
source: TransactionScriptErrorSource,
cause: ScriptError,
}
impl StdError for TransactionScriptError {}
impl fmt::Display for TransactionScriptError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(
f,
"TransactionScriptError {{ source: {}, cause: {} }}",
self.source, self.cause
)
}
}
impl ScriptError {
pub(crate) fn source(self, script_group: &ScriptGroup) -> TransactionScriptError {
TransactionScriptError {
source: TransactionScriptErrorSource::from_script_group(script_group),
cause: self,
}
}
pub fn input_lock_script(self, index: usize) -> TransactionScriptError {
TransactionScriptError {
source: TransactionScriptErrorSource::Inputs(index, ScriptGroupType::Lock),
cause: self,
}
}
pub fn input_type_script(self, index: usize) -> TransactionScriptError {
TransactionScriptError {
source: TransactionScriptErrorSource::Inputs(index, ScriptGroupType::Type),
cause: self,
}
}
pub fn output_type_script(self, index: usize) -> TransactionScriptError {
TransactionScriptError {
source: TransactionScriptErrorSource::Outputs(index, ScriptGroupType::Type),
cause: self,
}
}
}
impl From<TransactionScriptError> for Error {
fn from(error: TransactionScriptError) -> Self {
ErrorKind::Script.because(error)
}
}