lib.rs 8.31 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
// Copyright 2017-2020 Parity Technologies (UK) Ltd.
// This file is part of Polkadot.

// Polkadot 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.

// Polkadot 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 Polkadot.  If not, see <http://www.gnu.org/licenses/>.

//! Primitive types used on the node-side.
//!
//! Unlike the `polkadot-primitives` crate, these primitives are only used on the node-side,
//! not shared between the node and the runtime. This crate builds on top of the primitives defined
//! there.

23
use parity_scale_codec::{Decode, Encode};
asynchronous rob's avatar
asynchronous rob committed
24
25
26
27
28
use polkadot_primitives::v1::{
	Hash, CommittedCandidateReceipt, CandidateReceipt, CompactStatement,
	EncodeAs, Signed, SigningContext, ValidatorIndex, ValidatorId,
	UpwardMessage, Balance, ValidationCode, GlobalValidationSchedule, LocalValidationData,
	HeadData,
29
};
30
31
32
33
34
use polkadot_statement_table::{
	generic::{
		ValidityDoubleVote as TableValidityDoubleVote,
		MultipleCandidates as TableMultipleCandidates,
	},
asynchronous rob's avatar
asynchronous rob committed
35
	v1::Misbehavior as TableMisbehavior,
36
};
37
38

/// A statement, where the candidate receipt is included in the `Seconded` variant.
asynchronous rob's avatar
asynchronous rob committed
39
40
41
42
43
///
/// This is the committed candidate receipt instead of the bare candidate receipt. As such,
/// it gives access to the commitments to validators who have not executed the candidate. This
/// is necessary to allow a block-producing validator to include candidates from outside of the para
/// it is assigned to.
44
#[derive(Debug, Clone, PartialEq, Eq, Encode, Decode)]
45
46
47
pub enum Statement {
	/// A statement that a validator seconds a candidate.
	#[codec(index = "1")]
asynchronous rob's avatar
asynchronous rob committed
48
	Seconded(CommittedCandidateReceipt),
49
50
51
52
53
54
55
56
	/// A statement that a validator has deemed a candidate valid.
	#[codec(index = "2")]
	Valid(Hash),
	/// A statement that a validator has deemed a candidate invalid.
	#[codec(index = "3")]
	Invalid(Hash),
}

57
impl Statement {
asynchronous rob's avatar
asynchronous rob committed
58
59
	/// Transform this statement into its compact version, which references only the hash
	/// of the candidate.
60
61
62
63
64
65
66
67
68
	pub fn to_compact(&self) -> CompactStatement {
		match *self {
			Statement::Seconded(ref c) => CompactStatement::Candidate(c.hash()),
			Statement::Valid(hash) => CompactStatement::Valid(hash),
			Statement::Invalid(hash) => CompactStatement::Invalid(hash),
		}
	}
}

69
70
impl EncodeAs<CompactStatement> for Statement {
	fn encode_as(&self) -> Vec<u8> {
71
		self.to_compact().encode()
72
73
74
75
76
77
	}
}

/// A statement, the corresponding signature, and the index of the sender.
///
/// Signing context and validator set should be apparent from context.
78
79
80
81
///
/// This statement is "full" in the sense that the `Seconded` variant includes the candidate receipt.
/// Only the compact `SignedStatement` is suitable for submission to the chain.
pub type SignedFullStatement = Signed<Statement, CompactStatement>;
82
83

/// A misbehaviour report.
84
#[derive(Debug)]
85
86
87
88
89
90
91
92
93
pub enum MisbehaviorReport {
	/// These validator nodes disagree on this candidate's validity, please figure it out
	///
	/// Most likely, the list of statments all agree except for the final one. That's not
	/// guaranteed, though; if somehow we become aware of lots of
	/// statements disagreeing about the validity of a candidate before taking action,
	/// this message should be dispatched with all of them, in arbitrary order.
	///
	/// This variant is also used when our own validity checks disagree with others'.
asynchronous rob's avatar
asynchronous rob committed
94
	CandidateValidityDisagreement(CandidateReceipt, Vec<SignedFullStatement>),
95
	/// I've noticed a peer contradicting itself about a particular candidate
asynchronous rob's avatar
asynchronous rob committed
96
	SelfContradiction(CandidateReceipt, SignedFullStatement, SignedFullStatement),
97
	/// This peer has seconded more than one parachain candidate for this relay parent head
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
	DoubleVote(SignedFullStatement, SignedFullStatement),
}

/// A utility struct used to convert `TableMisbehavior` to `MisbehaviorReport`s.
pub struct FromTableMisbehavior {
	/// Index of the validator.
	pub id: ValidatorIndex,
	/// The misbehavior reported by the table.
	pub report: TableMisbehavior,
	/// Signing context.
	pub signing_context: SigningContext,
	/// Misbehaving validator's public key.
	pub key: ValidatorId,
}

asynchronous rob's avatar
asynchronous rob committed
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/// Outputs of validating a candidate.
#[derive(Debug)]
pub struct ValidationOutputs {
	/// The head-data produced by validation.
	pub head_data: HeadData,
	/// The global validation schedule.
	pub global_validation_schedule: GlobalValidationSchedule,
	/// The local validation data.
	pub local_validation_data: LocalValidationData,
	/// Upward messages to the relay chain.
	pub upward_messages: Vec<UpwardMessage>,
	/// Fees paid to the validators of the relay-chain.
	pub fees: Balance,
	/// The new validation code submitted by the execution, if any.
	pub new_validation_code: Option<ValidationCode>,
}

130
131
132
/// Result of the validation of the candidate.
#[derive(Debug)]
pub enum ValidationResult {
asynchronous rob's avatar
asynchronous rob committed
133
134
	/// Candidate is valid. The validation process yields these outputs.
	Valid(ValidationOutputs),
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
	/// Candidate is invalid.
	Invalid,
}

impl std::convert::TryFrom<FromTableMisbehavior> for MisbehaviorReport {
	type Error = ();

	fn try_from(f: FromTableMisbehavior) -> Result<Self, Self::Error> {
		match f.report {
			TableMisbehavior::ValidityDoubleVote(
				TableValidityDoubleVote::IssuedAndValidity((c, s1), (d, s2))
			) => {
				let receipt = c.clone();
				let signed_1 = SignedFullStatement::new(
					Statement::Seconded(c),
					f.id,
					s1,
					&f.signing_context,
					&f.key,
				).ok_or(())?;
				let signed_2 = SignedFullStatement::new(
					Statement::Valid(d),
					f.id,
					s2,
					&f.signing_context,
					&f.key,
				).ok_or(())?;

asynchronous rob's avatar
asynchronous rob committed
163
				Ok(MisbehaviorReport::SelfContradiction(receipt.to_plain(), signed_1, signed_2))
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
			}
			TableMisbehavior::ValidityDoubleVote(
				TableValidityDoubleVote::IssuedAndInvalidity((c, s1), (d, s2))
			) => {
				let receipt = c.clone();
				let signed_1 = SignedFullStatement::new(
					Statement::Seconded(c),
					f.id,
					s1,
					&f.signing_context,
					&f.key,
				).ok_or(())?;
				let signed_2 = SignedFullStatement::new(
					Statement::Invalid(d),
					f.id,
					s2,
					&f.signing_context,
					&f.key,
				).ok_or(())?;

asynchronous rob's avatar
asynchronous rob committed
184
				Ok(MisbehaviorReport::SelfContradiction(receipt.to_plain(), signed_1, signed_2))
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
			}
			TableMisbehavior::ValidityDoubleVote(
				TableValidityDoubleVote::ValidityAndInvalidity(c, s1, s2)
			) => {
				let signed_1 = SignedFullStatement::new(
					Statement::Valid(c.hash()),
					f.id,
					s1,
					&f.signing_context,
					&f.key,
				).ok_or(())?;
				let signed_2 = SignedFullStatement::new(
					Statement::Invalid(c.hash()),
					f.id,
					s2,
					&f.signing_context,
					&f.key,
				).ok_or(())?;

asynchronous rob's avatar
asynchronous rob committed
204
				Ok(MisbehaviorReport::SelfContradiction(c.to_plain(), signed_1, signed_2))
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
			}
			TableMisbehavior::MultipleCandidates(
				TableMultipleCandidates {
					first,
					second,
				}
			) => {
				let signed_1 = SignedFullStatement::new(
					Statement::Seconded(first.0),
					f.id,
					first.1,
					&f.signing_context,
					&f.key,
				).ok_or(())?;

				let signed_2 = SignedFullStatement::new(
					Statement::Seconded(second.0),
					f.id,
					second.1,
					&f.signing_context,
					&f.key,
				).ok_or(())?;

				Ok(MisbehaviorReport::DoubleVote(signed_1, signed_2))
			}
			_ => Err(()),
		}
	}
233
}
234
235
236
237
238
239
240

/// A unique identifier for a network protocol.
pub type ProtocolId = [u8; 4];

/// A succinct representation of a peer's view. This consists of a bounded amount of chain heads.
///
/// Up to `N` (5?) chain heads.
241
#[derive(Default, Debug, Clone, PartialEq, Eq, Encode, Decode)]
242
pub struct View(pub Vec<Hash>);
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259

impl View {
	/// Returns an iterator of the hashes present in `Self` but not in `other`.
	pub fn difference<'a>(&'a self, other: &'a View) -> impl Iterator<Item = &'a Hash> + 'a {
		self.0.iter().filter(move |h| !other.contains(h))
	}

	/// An iterator containing hashes present in both `Self` and in `other`.
	pub fn intersection<'a>(&'a self, other: &'a View) -> impl Iterator<Item = &'a Hash> + 'a {
		self.0.iter().filter(move |h| other.contains(h))
	}

	/// Whether the view contains a given hash.
	pub fn contains(&self, hash: &Hash) -> bool {
		self.0.contains(hash)
	}
}