From 5836a5f9e382084a4336d71b0d7cc2e091d85e76 Mon Sep 17 00:00:00 2001 From: Shoaib Ahmed Date: Mon, 11 Oct 2021 12:06:06 +0530 Subject: [PATCH 001/211] Fix default trusting period regression (#1441) --- relayer/src/config.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/relayer/src/config.rs b/relayer/src/config.rs index efe72a097..691afa3d7 100644 --- a/relayer/src/config.rs +++ b/relayer/src/config.rs @@ -318,7 +318,7 @@ pub struct ChainConfig { pub max_tx_size: MaxTxSize, #[serde(default = "default::clock_drift", with = "humantime_serde")] pub clock_drift: Duration, - #[serde(with = "humantime_serde")] + #[serde(default, with = "humantime_serde")] pub trusting_period: Option, // these two need to be last otherwise we run into `ValueAfterTable` error when serializing to TOML From 650574f8520719f5938887f1ddb4309a05b4263d Mon Sep 17 00:00:00 2001 From: Jacob Gadikian Date: Mon, 11 Oct 2021 18:01:51 +0700 Subject: [PATCH 002/211] Update compatibility.rs (#1429) * Update compatibility.rs Update Cosmos-SDK version # for compatiblity so Hermes throws fewer errors with newer chains. * added changelog entry with unclog * Update compatibility.rs bump sdk * unclog * update version number in guide * Changelog nits Co-authored-by: Adi Seredinschi --- .changelog/unreleased/bug-fixes/sdkversion.md | 1 + guide/src/features.md | 2 +- relayer/src/chain/cosmos/compatibility.rs | 2 +- 3 files changed, 3 insertions(+), 2 deletions(-) create mode 100644 .changelog/unreleased/bug-fixes/sdkversion.md diff --git a/.changelog/unreleased/bug-fixes/sdkversion.md b/.changelog/unreleased/bug-fixes/sdkversion.md new file mode 100644 index 000000000..cd7a85ad5 --- /dev/null +++ b/.changelog/unreleased/bug-fixes/sdkversion.md @@ -0,0 +1 @@ +- Broadened the range of supported Cosmos-SDK versions from `0.44.0` to `0.44.1` diff --git a/guide/src/features.md b/guide/src/features.md index 9680e75db..06ae6ca4c 100644 --- a/guide/src/features.md +++ b/guide/src/features.md @@ -5,7 +5,7 @@ A feature matrix and comparison between the Rust and Go relayer implementations > **Cosmos SDK compatibility:** > Hermes supports Cosmos SDK chains implementing the [IBC v1.0][ibcv1] protocol specification. -> Cosmos SDK versions `0.41.3` to `0.44.0` are officially supported. +> Cosmos SDK versions `0.41.3` to `0.44.1` are officially supported. > In case Hermes finds an incompatible SDK version, it will output a log warning. [ibcv1]: https://github.com/cosmos/ibc-go/tree/main/proto/ibc diff --git a/relayer/src/chain/cosmos/compatibility.rs b/relayer/src/chain/cosmos/compatibility.rs index a4ea5d6a7..d4e3cc953 100644 --- a/relayer/src/chain/cosmos/compatibility.rs +++ b/relayer/src/chain/cosmos/compatibility.rs @@ -23,7 +23,7 @@ const SDK_MODULE_NAME: &str = "cosmos/cosmos-sdk"; /// # Note: Should be consistent with [features] guide page. /// /// [features]: https://hermes.informal.systems/features.html -const SDK_MODULE_VERSION_REQ: &str = ">=0.41.3, <=0.44.0"; +const SDK_MODULE_VERSION_REQ: &str = ">=0.41.3, <=0.44.1"; /// Helper struct to capture all the reported information of an /// IBC application, e.g., `gaiad`. From 17b4c7ec9f27876fa5e474f68c91a0e05cb99d29 Mon Sep 17 00:00:00 2001 From: Romain Ruetschi Date: Mon, 11 Oct 2021 17:50:38 +0200 Subject: [PATCH 003/211] Clarify configuration section in the guide (#1443) * Remove outdated config docs from guide * Clarify the role of the chains specified in the config --- guide/src/config.md | 155 +++++++------------------------------------- 1 file changed, 25 insertions(+), 130 deletions(-) diff --git a/guide/src/config.md b/guide/src/config.md index 80e76ec95..6a889f3b8 100644 --- a/guide/src/config.md +++ b/guide/src/config.md @@ -20,145 +20,28 @@ hermes [-c CONFIG_FILE] COMMAND -## Configuration format +## Configuration The configuration file must have one `global` section, and one `chains` section for each chain. > **Note:** As of 0.6.0, the Hermes configuration file is self-documented. -> This section of the guide which discusses each parameter in turn is no -> longer maintained, and we may remove it soon. Please read the configuration -> file [`config.toml`](https://github.com/informalsystems/ibc-rs/blob/v0.7.3/config.toml) itself for the most up-to-date documentation of parameters. +> Please read the configuration file [`config.toml`](https://github.com/informalsystems/ibc-rs/blob/v0.7.3/config.toml) +> itself for the most up-to-date documentation of parameters. -### `[global]` +By default, Hermes will relay on all channels available between all the configured chains. +In this way, every configured chain will act as a source (in the sense that Hermes listens for events) +and as a destination (to relay packets that others chains have sent). -The `global` section has parameters that apply globally to the relayer operation. +For example, if there are only two chains configured, then Hermes will only relay packets between those two, +i.e. the two chains will serve as a source for each other, and likewise as a destination for each other's relevant events. +Hermes will ignore all events that pertain to chains which are unknown (ie. not present in config.toml). -#### Parameters - -* __strategy__: *(string)* Specify the strategy to be used by the relayer. Default: `packets` - Two options are currently supported: - - `all`: Relay packets and perform channel and connection handshakes. - - `packets`: Relay packets only. - -* __log_level__: *(string)* Specify the verbosity for the relayer logging output. Valid options are 'error', 'warn', 'info', 'debug', 'trace'. Default: `info`. - For more information on parametrizing the log output, see the section [help/log-level][log-level]. - -Here is an example for the `global` section: - -```toml -[global] -strategy = 'packets' -log_level = 'info' -``` - -### `[telemetry]` - -The `telemetry` section defines parameters for Hermes' built-in [telemetry](telemetry.md) capabilities. - -#### Parameters - -* __enabled__: *(boolean)* Whether or not to enable the telemetry service. Default: `false`. - -* __host__: *(string)* Specify the IPv4/6 host over which the built-in HTTP server will serve the metrics gathered by the telemetry service. Default: `127.0.0.1` - -* __port__: *(u16)* Specify the port over which the built-in HTTP server will serve the metrics gathered by the telemetry service. Default: `3001` - -Here is an example for the `telemetry` section: - -```toml -[telemetry] -enabled = true -host = '127.0.0.1' -port = 3001 -``` - -### `[rest]` - -The `rest` section defines parameters for Hermes' built-in [REST API](rest-api.md).. - -#### Parameters - -* __enabled__: *(boolean)* Whether or not to enable the built-in REST server. Default: `false`. - -* __host__: *(string)* Specify the IPv4/6 host over which the built-in HTTP server will be listening. Default: `127.0.0.1` - -* __port__: *(u16)* Specify the port over which the built-in HTTP server will be listening. Default: `3000` - -Here is an example for the `rest` section: - -```toml -[rest] -enabled = true -host = '127.0.0.1' -port = 3000 -``` - -### `[[chains]]` - -A `chains` section includes parameters related to a chain and the full node to which the relayer can send transactions and queries. - -#### Parameters - -* __id__: *(string)* Specify the chain ID. For example `ibc-0` - -* __rpc_addr__: *(string)* Specify the RPC address and port where the chain RPC server listens on. For example `http://localhost:26657` - -* __grpc_addr__: *(string)* Specify the GRPC address and port where the chain GRPC server listens on. For example `http://localhost:9090` - -* __websocket_addr__: *(string)* Specify the WebSocket address and port where the chain WebSocket server listens on. For example `ws://localhost:26657/websocket` - -* __rpc_timeout__: *(string)* Specify the maximum amount of time (duration) that the RPC requests should take before timing out. Default: `10s` (10 seconds). - -* __account_prefix__: *(string)* Specify the prefix used by the chain. For example `cosmos` - -* __key_name__: *(string)* Specify the name of the private key to use for signing transactions. See the [Adding Keys](commands/keys/index.md#adding-keys) chapter for for more information about managing signing keys. - -* __store_prefix__: *(string)* Specify the store prefix used by the on-chain IBC modules. For example `ibc`. - -* __max_gas__: *(u64)* Specify the maximum amount of gas to be used as the gas limit for a transaction. Default: `300000` - -* __gas_price__: *(table)* - * __price__: *(f64)* Specify the price per gas used of the fee to submit a transaction. - * __denom__: *(string)* Specify the denomination of the fee. - -* __gas_adjustment__: *(f64)* Specify by what percentage to increase the gas estimate used to compute the fee, to account for potential estimation error. Default: `0.1`, ie. 10%. - -* __max_msg_num__: *(u64)* Specify how many IBC messages at most to include in a single transaction. Default: `30` - -* __max_tx_size__: *(u64)* Specify the maximum size, in bytes, of each transaction that Hermes will submit. Default: `2097152` (2 MiB) - -* __clock_drift__: *(string)* Specify the maximum amount of time to tolerate a clock drift. The clock drift parameter defines how much new (untrusted) header's Time can drift into the future. Default: `5s` - -* __trusting_period__: *(string)* Specify the amount of time to be used as the light client trusting period. It should be significantly less than the unbonding period (e.g. unbonding period = 3 weeks, trusting period = 2 weeks). Default: `14days` (336 hours) - -* __trust_threshold__ (advanced): *(table)* Specify the trust threshold for the light client, ie. the maximum fraction of validators which have changed between two blocks. Default: `{ numerator = '1', denominator = '3' }`, ie. 1/3. - * __numerator__: *(string)* The numerator of the fraction (must parse to a `u64`). - * __denominator__: *(string)* The denominator of the fraction (must parse to a `u64`). - - __Warning__ - _This is an advanced feature! Modify with caution._ - -For example if you want to add a configuration for a chain named `ibc-0`: - -```toml -[[chains]] -id = 'ibc-0' -rpc_addr = 'http://127.0.0.1:26657' -grpc_addr = 'http://127.0.0.1:9090' -websocket_addr = 'ws://localhost:26657/websocket' -rpc_timeout = '10s' -account_prefix = 'cosmos' -key_name = 'testkey' -store_prefix = 'ibc' -max_gas = 2000000 -gas_price = { price = 0.001, denom = 'stake' } -gas_adjustment = 0.1 -clock_drift = '5s' -trusting_period = '14days' -``` +To restrict relaying on specific channels, or uni-directionally, you can use [packet filtering policies](https://github.com/informalsystems/ibc-rs/blob/v0.7.3/config.toml#L156-L173). ## Adding private keys -For each chain configured you need to add a private key for that chain in order to submit [transactions](./commands/raw/index.md), please refer to the [Keys](./commands/keys/index.md) sections in order to learn how to add the private keys that are used by the relayer. +For each chain configured you need to add a private key for that chain in order to submit [transactions](./commands/raw/index.md), +please refer to the [Keys](./commands/keys/index.md) sections in order to learn how to add the private keys that are used by the relayer. ## Example configuration file @@ -166,8 +49,20 @@ Here is a full example of a configuration file with two chains configured: ```toml [global] -strategy = 'packets' +strategy = 'all' log_level = 'info' +filter = false +clear_packets_interval = 100 + +[rest] +enabled = true +host = '127.0.0.1' +port = 3000 + +[telemetry] +enabled = true +host = '127.0.0.1' +port = 3001 [[chains]] id = 'ibc-0' From 271cffda3b8bd047bc3b968e6ec235954e309811 Mon Sep 17 00:00:00 2001 From: Adi Seredinschi Date: Tue, 12 Oct 2021 08:27:44 +0200 Subject: [PATCH 004/211] Support for memo field (#1434) * Added initial support for feature #1433 * Comment for Memo type * Derive Default and add Display instance for Memo * Use Memo type directly * Documented the memo prefix Co-authored-by: Romain Ruetschi --- config.toml | 6 ++++ relayer-cli/src/commands.rs | 9 +++++- relayer/src/chain/cosmos.rs | 14 ++++++--- relayer/src/chain/mock.rs | 1 + relayer/src/config.rs | 4 ++- relayer/src/config/types.rs | 62 +++++++++++++++++++++++++++++++++++++ 6 files changed, 90 insertions(+), 6 deletions(-) diff --git a/config.toml b/config.toml index 09066778e..a32b421cb 100644 --- a/config.toml +++ b/config.toml @@ -153,6 +153,12 @@ trusting_period = '14days' # Warning: This is an advanced feature! Modify with caution. trust_threshold = { numerator = '1', denominator = '3' } +# Specify a string that Hermes will use as a memo for each transaction it submits +# to this chain. The string is limited to 50 characters. Default: '' (empty). +# Note: Hermes will append to the string defined here additional +# operational debugging information, e.g., relayer build version. +memo_prefix = '' + # This section specifies the filters for policy based relaying. # Default: no policy/ filters # The section is ignored if the global 'filter' option is set to 'false'. diff --git a/relayer-cli/src/commands.rs b/relayer-cli/src/commands.rs index ce8cc1623..c2db0bf15 100644 --- a/relayer-cli/src/commands.rs +++ b/relayer-cli/src/commands.rs @@ -132,7 +132,14 @@ impl Configurable for CliCmd { /// /// This can be safely deleted if you don't want to override config /// settings from command-line options. - fn process_config(&self, config: Config) -> Result { + fn process_config(&self, mut config: Config) -> Result { + // Alter the memo for all chains to include a suffix with Hermes build details + let web = "https://hermes.informal.systems"; + let suffix = format!("{} {} ({})", CliCmd::name(), CliCmd::version(), web); + for ccfg in config.chains.iter_mut() { + ccfg.memo_prefix.apply_suffix(&suffix); + } + match self { CliCmd::Tx(cmd) => cmd.override_config(config), // CliCmd::Help(cmd) => cmd.override_config(config), diff --git a/relayer/src/chain/cosmos.rs b/relayer/src/chain/cosmos.rs index 72f03aba3..009f8d793 100644 --- a/relayer/src/chain/cosmos.rs +++ b/relayer/src/chain/cosmos.rs @@ -71,13 +71,13 @@ use ibc_proto::ibc::core::connection::v1::{ QueryClientConnectionsRequest, QueryConnectionsRequest, }; -use crate::error::Error; use crate::event::monitor::{EventMonitor, EventReceiver}; use crate::keyring::{KeyEntry, KeyRing, Store}; use crate::light_client::tendermint::LightClient as TmLightClient; use crate::light_client::LightClient; use crate::light_client::Verified; use crate::{chain::QueryResponse, event::monitor::TxMonitorCmd}; +use crate::{config::types::Memo, error::Error}; use crate::{ config::{AddressType, ChainConfig, GasPrice}, sdk_error::sdk_error_from_tx_sync_error_code, @@ -229,7 +229,7 @@ impl CosmosSdkChain { debug!("[{}] default fee: {}", self.id(), PrettyFee(&default_fee)); - let (body, body_buf) = tx_body_and_bytes(proto_msgs)?; + let (body, body_buf) = tx_body_and_bytes(proto_msgs, self.tx_memo())?; let (auth_info, auth_buf) = auth_info_and_bytes(signer_info.clone(), default_fee.clone())?; let signed_doc = self.signed_doc(body_buf.clone(), auth_buf, account_seq)?; @@ -644,6 +644,11 @@ impl CosmosSdkChain { .trusting_period .unwrap_or(2 * unbonding_period / 3) } + + /// Returns the preconfigured memo to be used for every submitted transaction + fn tx_memo(&self) -> &Memo { + &self.config.memo_prefix + } } fn empty_event_present(events: &[IbcEvent]) -> bool { @@ -1987,15 +1992,16 @@ fn auth_info_and_bytes(signer_info: SignerInfo, fee: Fee) -> Result<(AuthInfo, V Ok((auth_info, auth_buf)) } -fn tx_body_and_bytes(proto_msgs: Vec) -> Result<(TxBody, Vec), Error> { +fn tx_body_and_bytes(proto_msgs: Vec, memo: &Memo) -> Result<(TxBody, Vec), Error> { // Create TxBody let body = TxBody { messages: proto_msgs.to_vec(), - memo: "".to_string(), + memo: memo.to_string(), timeout_height: 0_u64, extension_options: Vec::::new(), non_critical_extension_options: Vec::::new(), }; + // A protobuf serialization of a TxBody let mut body_buf = Vec::new(); prost::Message::encode(&body, &mut body_buf).unwrap(); diff --git a/relayer/src/chain/mock.rs b/relayer/src/chain/mock.rs index a4d2ca6d2..6134ea8e6 100644 --- a/relayer/src/chain/mock.rs +++ b/relayer/src/chain/mock.rs @@ -438,6 +438,7 @@ pub mod test_utils { trust_threshold: Default::default(), packet_filter: PacketFilter::default(), address_type: AddressType::default(), + memo_prefix: Default::default(), } } } diff --git a/relayer/src/config.rs b/relayer/src/config.rs index 691afa3d7..81a51d08e 100644 --- a/relayer/src/config.rs +++ b/relayer/src/config.rs @@ -14,7 +14,7 @@ use tendermint_light_client::types::TrustThreshold; use ibc::ics24_host::identifier::{ChainId, ChannelId, PortId}; use ibc::timestamp::ZERO_DURATION; -use crate::config::types::{MaxMsgNum, MaxTxSize}; +use crate::config::types::{MaxMsgNum, MaxTxSize, Memo}; use crate::error::Error; #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] @@ -320,6 +320,8 @@ pub struct ChainConfig { pub clock_drift: Duration, #[serde(default, with = "humantime_serde")] pub trusting_period: Option, + #[serde(default)] + pub memo_prefix: Memo, // these two need to be last otherwise we run into `ValueAfterTable` error when serializing to TOML #[serde(default)] diff --git a/relayer/src/config/types.rs b/relayer/src/config/types.rs index 0f2b58c6c..95daad93c 100644 --- a/relayer/src/config/types.rs +++ b/relayer/src/config/types.rs @@ -3,6 +3,8 @@ //! Implements defaults, as well as serializing and //! deserializing with upper-bound verification. +use core::fmt; + use serde::de::Unexpected; use serde::{de::Error, Deserialize, Deserializer, Serialize, Serializer}; @@ -99,3 +101,63 @@ impl From for usize { m.0 } } + +/// A memo domain-type. +/// +/// Hermes uses this type to populate the `tx.memo` field for +/// each transaction it submits. +/// The memo can be configured on a per-chain basis. +/// +#[derive(Clone, Debug, Default)] +pub struct Memo(String); + +impl Memo { + const MAX_LEN: usize = 50; + + pub fn apply_suffix(&mut self, suffix: &str) { + // Add a separator if the memo + // is pre-populated with some content already. + if !self.0.is_empty() { + self.0.push_str(" | "); + } + + self.0.push_str(suffix); + } + + pub fn as_str(&self) -> &str { + &self.0 + } +} + +impl<'de> Deserialize<'de> for Memo { + fn deserialize(deserializer: D) -> Result + where + D: Deserializer<'de>, + { + let m = String::deserialize(deserializer)?; + + if m.len() > Self::MAX_LEN { + return Err(D::Error::invalid_length( + m.len(), + &format!("a string length of at most {}", Self::MAX_LEN).as_str(), + )); + } + + Ok(Memo(m)) + } +} + +impl Serialize for Memo { + fn serialize(&self, serializer: S) -> Result + where + S: Serializer, + { + self.0.serialize(serializer) + } +} + +impl fmt::Display for Memo { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + write!(f, "{}", self.as_str()) + } +} From 369e39e8cd0a9ab1e9a76e8894655f9f4e766520 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 12 Oct 2021 08:29:16 +0200 Subject: [PATCH 005/211] Bump tracing from 0.1.28 to 0.1.29 (#1449) Bumps [tracing](https://github.com/tokio-rs/tracing) from 0.1.28 to 0.1.29. - [Release notes](https://github.com/tokio-rs/tracing/releases) - [Commits](https://github.com/tokio-rs/tracing/compare/tracing-0.1.28...tracing-0.1.29) --- updated-dependencies: - dependency-name: tracing dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- Cargo.lock | 8 ++++---- modules/Cargo.toml | 2 +- relayer-cli/Cargo.toml | 2 +- relayer/Cargo.toml | 2 +- 4 files changed, 7 insertions(+), 7 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 48bd04083..209a67a6f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -3346,9 +3346,9 @@ checksum = "360dfd1d6d30e05fda32ace2c8c70e9c0a9da713275777f5a4dbb8a1893930c6" [[package]] name = "tracing" -version = "0.1.28" +version = "0.1.29" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "84f96e095c0c82419687c20ddf5cb3eadb61f4e1405923c9dc8e53a1adacbda8" +checksum = "375a639232caf30edfc78e8d89b2d4c375515393e7af7e16f01cd96917fb2105" dependencies = [ "cfg-if 1.0.0", "log", @@ -3359,9 +3359,9 @@ dependencies = [ [[package]] name = "tracing-attributes" -version = "0.1.16" +version = "0.1.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "98863d0dd09fa59a1b79c6750ad80dbda6b75f4e71c437a6a1a8cb91a8bcbd77" +checksum = "f4f480b8f81512e825f337ad51e94c1eb5d3bbdf2b363dcd01e2b19a9ffe3f8e" dependencies = [ "proc-macro2", "quote", diff --git a/modules/Cargo.toml b/modules/Cargo.toml index 4abcbb378..76753360d 100644 --- a/modules/Cargo.toml +++ b/modules/Cargo.toml @@ -30,7 +30,7 @@ thiserror = { version = "1.0.29", default-features = false } serde_derive = { version = "1.0.104", default-features = false } serde = { version = "1.0.130", default-features = false } serde_json = { version = "1", default-features = false } -tracing = { version = "0.1.28", default-features = false } +tracing = { version = "0.1.29", default-features = false } prost = { version = "0.7", default-features = false } prost-types = { version = "0.7", default-features = false } bytes = { version = "1.1.0", default-features = false } diff --git a/relayer-cli/Cargo.toml b/relayer-cli/Cargo.toml index b5e03c387..954ef1350 100644 --- a/relayer-cli/Cargo.toml +++ b/relayer-cli/Cargo.toml @@ -34,7 +34,7 @@ ibc-relayer-rest = { version = "0.7.3", path = "../relayer-rest", optional = tru gumdrop = { version = "0.7", features = ["default_expr"] } serde = { version = "1", features = ["serde_derive"] } tokio = { version = "1.0", features = ["full"] } -tracing = "0.1.28" +tracing = "0.1.29" tracing-subscriber = "0.2.24" eyre = "0.6.5" color-eyre = "0.5" diff --git a/relayer/Cargo.toml b/relayer/Cargo.toml index 49eb60f5a..0c000507c 100644 --- a/relayer/Cargo.toml +++ b/relayer/Cargo.toml @@ -35,7 +35,7 @@ serde_cbor = "0.11.2" serde_derive = "1.0" thiserror = "1.0.29" toml = "0.5" -tracing = "0.1.28" +tracing = "0.1.29" tokio = { version = "1.0", features = ["rt-multi-thread", "time", "sync"] } serde_json = { version = "1" } bytes = "1.1.0" From 688c27d8b4523bb909bfae2c5bf989b931491ce1 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 12 Oct 2021 08:42:47 +0200 Subject: [PATCH 006/211] Bump thiserror from 1.0.29 to 1.0.30 (#1448) Bumps [thiserror](https://github.com/dtolnay/thiserror) from 1.0.29 to 1.0.30. - [Release notes](https://github.com/dtolnay/thiserror/releases) - [Commits](https://github.com/dtolnay/thiserror/compare/1.0.29...1.0.30) --- updated-dependencies: - dependency-name: thiserror dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- Cargo.lock | 8 ++++---- modules/Cargo.toml | 2 +- relayer/Cargo.toml | 2 +- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 209a67a6f..572f0fcc4 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -3083,18 +3083,18 @@ dependencies = [ [[package]] name = "thiserror" -version = "1.0.29" +version = "1.0.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "602eca064b2d83369e2b2f34b09c70b605402801927c65c11071ac911d299b88" +checksum = "854babe52e4df1653706b98fcfc05843010039b406875930a70e4d9644e5c417" dependencies = [ "thiserror-impl", ] [[package]] name = "thiserror-impl" -version = "1.0.29" +version = "1.0.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bad553cc2c78e8de258400763a647e80e6d1b31ee237275d756f6836d204494c" +checksum = "aa32fd3f627f367fe16f893e2597ae3c05020f8bba2666a4e6ea73d377e5714b" dependencies = [ "proc-macro2", "quote", diff --git a/modules/Cargo.toml b/modules/Cargo.toml index 76753360d..0cf01d476 100644 --- a/modules/Cargo.toml +++ b/modules/Cargo.toml @@ -26,7 +26,7 @@ mocks = [ "tendermint-testgen", "sha2" ] ibc-proto = { version = "0.11.0", path = "../proto" } ics23 = { version = "0.6.5", default-features = false } chrono = { version = "0.4.19", default-features = false } -thiserror = { version = "1.0.29", default-features = false } +thiserror = { version = "1.0.30", default-features = false } serde_derive = { version = "1.0.104", default-features = false } serde = { version = "1.0.130", default-features = false } serde_json = { version = "1", default-features = false } diff --git a/relayer/Cargo.toml b/relayer/Cargo.toml index 0c000507c..17938d53a 100644 --- a/relayer/Cargo.toml +++ b/relayer/Cargo.toml @@ -33,7 +33,7 @@ humantime-serde = "1.0.0" serde = "1.0.130" serde_cbor = "0.11.2" serde_derive = "1.0" -thiserror = "1.0.29" +thiserror = "1.0.30" toml = "0.5" tracing = "0.1.29" tokio = { version = "1.0", features = ["rt-multi-thread", "time", "sync"] } From 8944f94815c35b2d3297971aa437900d259dc75a Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 12 Oct 2021 08:43:14 +0200 Subject: [PATCH 007/211] Bump tracing-subscriber from 0.2.24 to 0.2.25 (#1446) Bumps [tracing-subscriber](https://github.com/tokio-rs/tracing) from 0.2.24 to 0.2.25. - [Release notes](https://github.com/tokio-rs/tracing/releases) - [Commits](https://github.com/tokio-rs/tracing/compare/tracing-subscriber-0.2.24...tracing-subscriber-0.2.25) --- updated-dependencies: - dependency-name: tracing-subscriber dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- Cargo.lock | 14 +++++++------- modules/Cargo.toml | 2 +- relayer-cli/Cargo.toml | 2 +- relayer/Cargo.toml | 2 +- 4 files changed, 10 insertions(+), 10 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 572f0fcc4..63a2fd144 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1306,7 +1306,7 @@ dependencies = [ "test-env-log", "thiserror", "tracing", - "tracing-subscriber 0.2.24", + "tracing-subscriber 0.2.25", ] [[package]] @@ -1373,7 +1373,7 @@ dependencies = [ "toml", "tonic", "tracing", - "tracing-subscriber 0.2.24", + "tracing-subscriber 0.2.25", "uint", ] @@ -1413,7 +1413,7 @@ dependencies = [ "tokio", "toml", "tracing", - "tracing-subscriber 0.2.24", + "tracing-subscriber 0.2.25", ] [[package]] @@ -1694,7 +1694,7 @@ dependencies = [ "tempfile", "thiserror", "tracing", - "tracing-subscriber 0.2.24", + "tracing-subscriber 0.2.25", "ureq", ] @@ -3384,7 +3384,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b4d7c0b83d4a500748fa5879461652b361edf5c9d51ede2a2ac03875ca185e24" dependencies = [ "tracing", - "tracing-subscriber 0.2.24", + "tracing-subscriber 0.2.25", ] [[package]] @@ -3437,9 +3437,9 @@ dependencies = [ [[package]] name = "tracing-subscriber" -version = "0.2.24" +version = "0.2.25" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fdd0568dbfe3baf7048b7908d2b32bca0d81cd56bec6d2a8f894b01d74f86be3" +checksum = "0e0d2eaa99c3c2e41547cfa109e910a68ea03823cccad4a0525dcbc9b01e8c71" dependencies = [ "ansi_term 0.12.1", "chrono", diff --git a/modules/Cargo.toml b/modules/Cargo.toml index 0cf01d476..b0f1c742b 100644 --- a/modules/Cargo.toml +++ b/modules/Cargo.toml @@ -51,7 +51,7 @@ optional = true [dev-dependencies] env_logger = "0.9.0" -tracing-subscriber = "0.2.24" +tracing-subscriber = "0.2.25" test-env-log = { version = "0.2.7", features = ["trace"] } modelator = "0.2.1" tendermint-rpc = { version = "=0.22.0", features = ["http-client", "websocket-client"] } diff --git a/relayer-cli/Cargo.toml b/relayer-cli/Cargo.toml index 954ef1350..610755948 100644 --- a/relayer-cli/Cargo.toml +++ b/relayer-cli/Cargo.toml @@ -35,7 +35,7 @@ gumdrop = { version = "0.7", features = ["default_expr"] } serde = { version = "1", features = ["serde_derive"] } tokio = { version = "1.0", features = ["full"] } tracing = "0.1.29" -tracing-subscriber = "0.2.24" +tracing-subscriber = "0.2.25" eyre = "0.6.5" color-eyre = "0.5" futures = "0.3.17" diff --git a/relayer/Cargo.toml b/relayer/Cargo.toml index 17938d53a..731253e67 100644 --- a/relayer/Cargo.toml +++ b/relayer/Cargo.toml @@ -85,7 +85,7 @@ version = "=0.22.0" ibc = { version = "0.7.3", path = "../modules", features = ["mocks"] } serial_test = "0.5.0" env_logger = "0.9.0" -tracing-subscriber = "0.2.24" +tracing-subscriber = "0.2.25" test-env-log = { version = "0.2.7", features = ["trace"] } # Needed for generating (synthetic) light blocks. From 14aeb87880be14be10a6684ff169ff77038dd1ee Mon Sep 17 00:00:00 2001 From: Romain Ruetschi Date: Wed, 13 Oct 2021 17:07:23 +0200 Subject: [PATCH 008/211] Improve error messages when `create client` fails (#1450) * Improve error message when failing to build a Tendermint client state * Fix `ForeignClientError::client_create` description * Always mention underlying reason in error messages * Formatting * Add health check for trusting period * Add .changelog entry * Apply suggestions from code review Co-authored-by: Anca Zamfir * Formatting and further fixes * Add config validaton check for the trusting period to be greater than zero Co-authored-by: Anca Zamfir --- .../1440-improve-error-msg-create-client.md | 3 + Cargo.lock | 1 + modules/src/ics07_tendermint/client_state.rs | 26 +++++---- modules/src/ics07_tendermint/error.rs | 57 +++++++++---------- relayer/Cargo.toml | 1 + relayer/src/chain/cosmos.rs | 21 +++++++ relayer/src/error.rs | 28 ++++++++- relayer/src/foreign_client.rs | 2 +- 8 files changed, 96 insertions(+), 43 deletions(-) create mode 100644 .changelog/unreleased/improvements/ibc-relayer/1440-improve-error-msg-create-client.md diff --git a/.changelog/unreleased/improvements/ibc-relayer/1440-improve-error-msg-create-client.md b/.changelog/unreleased/improvements/ibc-relayer/1440-improve-error-msg-create-client.md new file mode 100644 index 000000000..ba90a1f79 --- /dev/null +++ b/.changelog/unreleased/improvements/ibc-relayer/1440-improve-error-msg-create-client.md @@ -0,0 +1,3 @@ +- Improve error message when `create client` fails and add a health + check for the trusting period being smaller than the unbonding period + ([#1440](https://github.com/informalsystems/ibc-rs/issues/1440)) \ No newline at end of file diff --git a/Cargo.lock b/Cargo.lock index 63a2fd144..724ceb6fe 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1341,6 +1341,7 @@ dependencies = [ "hdpath", "hex", "http", + "humantime", "humantime-serde", "ibc", "ibc-proto", diff --git a/modules/src/ics07_tendermint/client_state.rs b/modules/src/ics07_tendermint/client_state.rs index eff7d5e5d..7fe85a0d7 100644 --- a/modules/src/ics07_tendermint/client_state.rs +++ b/modules/src/ics07_tendermint/client_state.rs @@ -55,19 +55,24 @@ impl ClientState { ) -> Result { // Basic validation of trusting period and unbonding period: each should be non-zero. if trusting_period <= Duration::new(0, 0) { - return Err(Error::invalid_trusting_period( - "ClientState trusting period must be greater than zero".to_string(), - )); + return Err(Error::invalid_trusting_period(format!( + "ClientState trusting period ({:?}) must be greater than zero", + trusting_period + ))); } + if unbonding_period <= Duration::new(0, 0) { - return Err(Error::invalid_unbounding_period( - "ClientState unbonding period must be greater than zero".to_string(), - )); + return Err(Error::invalid_unbonding_period(format!( + "ClientState unbonding period ({:?}) must be greater than zero", + unbonding_period + ))); } + if trusting_period >= unbonding_period { - return Err(Error::invalid_unbounding_period( - "ClientState trusting period must be smaller than unbonding period".to_string(), - )); + return Err(Error::invalid_trusting_period(format!( + "ClientState trusting period ({:?}) must be smaller than unbonding period ({:?})", + trusting_period, unbonding_period, + ))); } // Basic validation for the frozen_height parameter. @@ -76,10 +81,11 @@ impl ClientState { "ClientState cannot be frozen at creation time".to_string(), )); } + // Basic validation for the latest_height parameter. if latest_height <= Height::zero() { return Err(Error::validation( - "ClientState latest height cannot be smaller or equal than zero".to_string(), + "ClientState latest height must be greater than zero".to_string(), )); } diff --git a/modules/src/ics07_tendermint/error.rs b/modules/src/ics07_tendermint/error.rs index e453a0516..a67b69e74 100644 --- a/modules/src/ics07_tendermint/error.rs +++ b/modules/src/ics07_tendermint/error.rs @@ -9,11 +9,11 @@ define_error! { Error { InvalidTrustingPeriod { reason: String } - | _ | { "invalid trusting period" }, + |e| { format_args!("invalid trusting period: {}", e.reason) }, - InvalidUnboundingPeriod + InvalidUnbondingPeriod { reason: String } - | _ | { "invalid unbonding period" }, + |e| { format_args!("invalid unbonding period: {}", e.reason) }, InvalidAddress | _ | { "invalid address" }, @@ -21,86 +21,83 @@ define_error! { InvalidHeader { reason: String } [ tendermint::Error ] - | _ | { "invalid header, failed basic validation" }, + |e| { format_args!("invalid header, failed basic validation: {}", e.reason) }, InvalidTrustThreshold { reason: String } - | e | { - format_args!("invalid client state trust threshold: {}", - e.reason) - }, + |e| { format_args!("invalid client state trust threshold: {}", e.reason) }, MissingSignedHeader - | _ | { "missing signed header" }, + |_| { "missing signed header" }, Validation { reason: String } - | _ | { "invalid header, failed basic validation" }, + |e| { format_args!("invalid header, failed basic validation: {}", e.reason) }, InvalidRawClientState { reason: String } - | _ | { "invalid raw client state" }, + |e| { format_args!("invalid raw client state: {}", e.reason) }, MissingValidatorSet - | _ | { "missing validator set" }, + |_| { "missing validator set" }, MissingTrustedValidatorSet - | _ | { "missing trusted validator set" }, + |_| { "missing trusted validator set" }, MissingTrustedHeight - | _ | { "missing trusted height" }, + |_| { "missing trusted height" }, MissingTrustingPeriod - | _ | { "missing trusting period" }, + |_| { "missing trusting period" }, MissingUnbondingPeriod - | _ | { "missing unbonding period" }, + |_| { "missing unbonding period" }, InvalidChainIdentifier [ ValidationError ] - | _ | { "Invalid chain identifier" }, + |_| { "invalid chain identifier" }, NegativeTrustingPeriod - | _ | { "negative trusting period" }, + |_| { "negative trusting period" }, NegativeUnbondingPeriod - | _ | { "negative unbonding period" }, + |_| { "negative unbonding period" }, MissingMaxClockDrift - | _ | { "missing max clock drift" }, + |_| { "missing max clock drift" }, NegativeMaxClockDrift - | _ | { "negative max clock drift" }, + |_| { "negative max clock drift" }, MissingLatestHeight - | _ | { "missing latest height" }, + |_| { "missing latest height" }, MissingFrozenHeight - | _ | { "missing frozen height" }, + |_| { "missing frozen height" }, InvalidChainId { raw_value: String } [ ValidationError ] - | e | { format_args!("invalid chain identifier: raw value {0}", e.raw_value) }, + |e| { format_args!("invalid chain identifier: {}", e.raw_value) }, InvalidRawHeight - | _ | { "invalid raw height" }, + { raw_height: u64 } + |e| { format_args!("invalid raw height: {}", e.raw_height) }, InvalidRawConsensusState { reason: String } - | _ | { "invalid raw client consensus state" }, + |e| { format_args!("invalid raw client consensus state: {}", e.reason) }, InvalidRawHeader [ tendermint::Error ] - | _ | { "invalid raw header" }, + |_| { "invalid raw header" }, InvalidRawMisbehaviour { reason: String } - | _ | { "invalid raw misbehaviour" }, + |e| { format_args!("invalid raw misbehaviour: {}", e.reason) }, Decode [ TraceError ] - | _ | { "decode error" }, - + |_| { "decode error" }, } } diff --git a/relayer/Cargo.toml b/relayer/Cargo.toml index 731253e67..3f22e3c12 100644 --- a/relayer/Cargo.toml +++ b/relayer/Cargo.toml @@ -64,6 +64,7 @@ anyhow = "1.0.44" fraction = {version = "0.9.0", default-features = false } semver = "1.0" uint = "0.9" +humantime = "2.1.0" [dependencies.tendermint] version = "=0.22.0" diff --git a/relayer/src/chain/cosmos.rs b/relayer/src/chain/cosmos.rs index 009f8d793..374fd975b 100644 --- a/relayer/src/chain/cosmos.rs +++ b/relayer/src/chain/cosmos.rs @@ -128,6 +128,27 @@ impl CosmosSdkChain { /// Emits a log warning in case any error is encountered and /// exits early without doing subsequent validations. pub fn validate_params(&self) -> Result<(), Error> { + // Check that the trusting period is smaller than the unbounding period + let unbonding_period = self.unbonding_period()?; + let trusting_period = self.trusting_period(unbonding_period); + + if trusting_period <= Duration::ZERO { + return Err(Error::config_validation_trusting_period_smaller_than_zero( + self.id().clone(), + trusting_period, + )); + } + + if trusting_period >= unbonding_period { + return Err( + Error::config_validation_trusting_period_greater_than_unbonding_period( + self.id().clone(), + trusting_period, + unbonding_period, + ), + ); + } + // Get the latest height and convert to tendermint Height let latest_height = Height::try_from(self.query_latest_height()?.revision_height) .map_err(Error::invalid_height)?; diff --git a/relayer/src/error.rs b/relayer/src/error.rs index 7d80811d6..361df8061 100644 --- a/relayer/src/error.rs +++ b/relayer/src/error.rs @@ -1,9 +1,10 @@ //! This module defines the various errors that be raised in the relayer. -use crate::keyring::errors::Error as KeyringError; -use crate::sdk_error::SdkError; +use core::time::Duration; + use flex_error::{define_error, DisplayOnly, TraceClone, TraceError}; use http::uri::InvalidUri; +use humantime::format_duration; use prost::DecodeError; use tendermint_light_client::{ components::io::IoError as LightClientIoError, errors::Error as LightClientError, @@ -29,6 +30,8 @@ use ibc::{ use crate::chain::cosmos::GENESIS_MAX_BYTES_MAX_FRACTION; use crate::event::monitor; +use crate::keyring::errors::Error as KeyringError; +use crate::sdk_error::SdkError; define_error! { Error { @@ -408,6 +411,27 @@ define_error! { e.chain_id, e.configured_bound, GENESIS_MAX_BYTES_MAX_FRACTION * 100.0, e.genesis_bound) }, + ConfigValidationTrustingPeriodSmallerThanZero + { + chain_id: ChainId, + trusting_period: Duration, + } + |e| { + format!("semantic config validation failed for option `trusting_period` of chain '{}', reason: trusting period ({}) must be greater than zero", + e.chain_id, format_duration(e.trusting_period)) + }, + + ConfigValidationTrustingPeriodGreaterThanUnbondingPeriod + { + chain_id: ChainId, + trusting_period: Duration, + unbonding_period: Duration, + } + |e| { + format!("semantic config validation failed for option `trusting_period` of chain '{}', reason: trusting period ({}) must be smaller than the unbonding period ({})", + e.chain_id, format_duration(e.trusting_period), format_duration(e.unbonding_period)) + }, + SdkModuleVersion { chain_id: ChainId, diff --git a/relayer/src/foreign_client.rs b/relayer/src/foreign_client.rs index a18de701d..9d2f0a7cd 100644 --- a/relayer/src/foreign_client.rs +++ b/relayer/src/foreign_client.rs @@ -447,7 +447,7 @@ impl ForeignClient Date: Tue, 19 Oct 2021 09:14:32 -0400 Subject: [PATCH 009/211] ICS07: Add verification method for client update handler (#1321) * bla * on going * ongoing * on going * on going * timestamp default changed from none to now * failed ping pong - signs * Update context.rs * bla * blabla * fixed test client_update_ping_pong * fmt + clippy * Update context.rs * Update context.rs * remove comments * Update host.rs * Update to tendermint-rs v0.20.0 * Update changelog * Fix tendermint-rs version in changelog * Update predicates.rs * Update context.rs * tests * tendermint stuff in * Update Cargo.toml * clippy + fmt * moved predicates into ics07 header.rs * Adapted to latest TM changes * Fixed MockHeader test * Fmt & clippy * Removed irrelevant file * Bit more cleanup * fixed tests * upgraded to new error model * fmt * errors and timestamp changes * Fix error notation and formatting Signed-off-by: Thane Thomson * Upgrade to tendermint-rs master Signed-off-by: Thane Thomson * Use tendermint-rs from branch thane/ibc-1252 Signed-off-by: Thane Thomson * Refactor ICS07 update handler to reuse light client verifier This commit makes use of the latest code from https://github.com/informalsystems/tendermint-rs/pull/960 in order to reuse the light client's verification predicates instead of reimplementing them in the `ibc` crate. Signed-off-by: Thane Thomson * Update Cargo.lock to address zeroize issue Signed-off-by: Thane Thomson * Bump tendermint-light-client dep to v0.22.0 for ibc module Signed-off-by: Thane Thomson * Refactor to accommodate new context API Signed-off-by: Thane Thomson * Fix missing import Signed-off-by: Thane Thomson * Fix imports Signed-off-by: Thane Thomson * Fix error check in test Signed-off-by: Thane Thomson * Output debug version of error Signed-off-by: Thane Thomson * Remove test as per https://github.com/informalsystems/ibc-rs/pull/1321#discussion_r713613488 Signed-off-by: Thane Thomson * Address comments from Adi Signed-off-by: Thane Thomson * Cosmetic tweaks Signed-off-by: Thane Thomson * Add revision number check Signed-off-by: Thane Thomson * Fix broken test Signed-off-by: Thane Thomson * Check incoming header height against chain ID version from client state Signed-off-by: Thane Thomson * Add revision_number consistency check when deserializing header Signed-off-by: Thane Thomson * Clarify MismatchedRevisions error message Signed-off-by: Thane Thomson * Add changelog entries Signed-off-by: Thane Thomson * Commented import no longer necessary Signed-off-by: Thane Thomson * Add in-the-middle monotonicity checks Signed-off-by: Thane Thomson * Fix broken dep tree relating to Prometheus Signed-off-by: Thane Thomson * Move next/prev consensus state search functionality to ClientReader trait Signed-off-by: Thane Thomson * Move impl of prev/next to the specific implementation and simplify signatures * Disable `tendermint-light-client` default features, ie. RPC client, std and color-eyre * Apply suggestions from code review * Fix compilation * Cleanup BTreeMap import * Always show underlying reason in ics07_tendermint errors * Update modules/src/ics02_client/handler/update_client.rs * Fix compilation Co-authored-by: cezarad Co-authored-by: Romain Ruetschi Co-authored-by: cezarad <9439384+cezarad@users.noreply.github.com> Co-authored-by: Adi Seredinschi Co-authored-by: Anca Zamfir --- .../breaking-changes/ibc/1214-ics07.md | 3 + .../unreleased/features/ibc/1214-ics07.md | 2 + Cargo.lock | 1 + modules/Cargo.toml | 7 +- .../msgs/transfer.rs | 7 +- modules/src/ics02_client/client_def.rs | 13 +- modules/src/ics02_client/context.rs | 37 +- modules/src/ics02_client/error.rs | 51 ++- .../src/ics02_client/handler/update_client.rs | 349 +++++++++++++++++- modules/src/ics02_client/header.rs | 13 + .../src/ics04_channel/handler/send_packet.rs | 9 +- modules/src/ics07_tendermint/client_def.rs | 182 ++++++++- modules/src/ics07_tendermint/client_state.rs | 25 +- modules/src/ics07_tendermint/error.rs | 134 ++++++- modules/src/ics07_tendermint/header.rs | 20 +- modules/src/ics18_relayer/utils.rs | 17 +- modules/src/ics26_routing/handler.rs | 12 +- modules/src/mock/client_def.rs | 3 + modules/src/mock/context.rs | 169 ++++++++- modules/src/mock/header.rs | 14 +- modules/src/mock/host.rs | 22 +- modules/src/timestamp.rs | 16 + relayer-cli/src/error.rs | 3 +- relayer/src/error.rs | 9 +- 24 files changed, 1029 insertions(+), 89 deletions(-) create mode 100644 .changelog/unreleased/breaking-changes/ibc/1214-ics07.md create mode 100644 .changelog/unreleased/features/ibc/1214-ics07.md diff --git a/.changelog/unreleased/breaking-changes/ibc/1214-ics07.md b/.changelog/unreleased/breaking-changes/ibc/1214-ics07.md new file mode 100644 index 000000000..42080cdef --- /dev/null +++ b/.changelog/unreleased/breaking-changes/ibc/1214-ics07.md @@ -0,0 +1,3 @@ +- The `check_header_and_update_state` method of the `ClientDef` + trait (ICS02) has been expanded to facilitate ICS07 + ([#1214](https://github.com/informalsystems/ibc-rs/issues/1214)) \ No newline at end of file diff --git a/.changelog/unreleased/features/ibc/1214-ics07.md b/.changelog/unreleased/features/ibc/1214-ics07.md new file mode 100644 index 000000000..460b04b45 --- /dev/null +++ b/.changelog/unreleased/features/ibc/1214-ics07.md @@ -0,0 +1,2 @@ +- Add ICS07 verification functionality by using `tendermint-light-client` + ([#1214](https://github.com/informalsystems/ibc-rs/issues/1214)) diff --git a/Cargo.lock b/Cargo.lock index 724ceb6fe..53de095be 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1300,6 +1300,7 @@ dependencies = [ "sha2", "subtle-encoding", "tendermint", + "tendermint-light-client", "tendermint-proto", "tendermint-rpc", "tendermint-testgen", diff --git a/modules/Cargo.toml b/modules/Cargo.toml index b0f1c742b..7dc559213 100644 --- a/modules/Cargo.toml +++ b/modules/Cargo.toml @@ -17,9 +17,10 @@ description = """ default = ["std", "eyre_tracer"] std = ["flex-error/std"] eyre_tracer = ["flex-error/eyre_tracer"] + # This feature grants access to development-time mocking libraries, such as `MockContext` or `MockHeader`. # Depends on the `testgen` suite for generating Tendermint light blocks. -mocks = [ "tendermint-testgen", "sha2" ] +mocks = ["tendermint-testgen", "sha2"] [dependencies] # Proto definitions for all IBC-related interfaces, e.g., connections or channels. @@ -45,6 +46,10 @@ version = "=0.22.0" [dependencies.tendermint-proto] version = "=0.22.0" +[dependencies.tendermint-light-client] +version = "=0.22.0" +default-features = false + [dependencies.tendermint-testgen] version = "=0.22.0" optional = true diff --git a/modules/src/application/ics20_fungible_token_transfer/msgs/transfer.rs b/modules/src/application/ics20_fungible_token_transfer/msgs/transfer.rs index d5eec5c87..336b5c258 100644 --- a/modules/src/application/ics20_fungible_token_transfer/msgs/transfer.rs +++ b/modules/src/application/ics20_fungible_token_transfer/msgs/transfer.rs @@ -101,14 +101,17 @@ impl From for RawMsgTransfer { #[cfg(test)] pub mod test_util { + use std::ops::Add; + use std::time::Duration; + use crate::{ ics24_host::identifier::{ChannelId, PortId}, test_utils::get_dummy_account_id, + timestamp::Timestamp, Height, }; use super::MsgTransfer; - use crate::timestamp::Timestamp; // Returns a dummy `RawMsgTransfer`, for testing only! pub fn get_dummy_msg_transfer(height: u64) -> MsgTransfer { @@ -120,7 +123,7 @@ pub mod test_util { token: None, sender: id.clone(), receiver: id, - timeout_timestamp: Timestamp::from_nanoseconds(1).unwrap(), + timeout_timestamp: Timestamp::now().add(Duration::from_secs(10)).unwrap(), timeout_height: Height { revision_number: 0, revision_height: height, diff --git a/modules/src/ics02_client/client_def.rs b/modules/src/ics02_client/client_def.rs index d19adc2f7..d97f8fa6b 100644 --- a/modules/src/ics02_client/client_def.rs +++ b/modules/src/ics02_client/client_def.rs @@ -4,6 +4,7 @@ use crate::downcast; use crate::ics02_client::client_consensus::{AnyConsensusState, ConsensusState}; use crate::ics02_client::client_state::{AnyClientState, ClientState}; use crate::ics02_client::client_type::ClientType; +use crate::ics02_client::context::ClientReader; use crate::ics02_client::error::Error; use crate::ics02_client::header::{AnyHeader, Header}; use crate::ics03_connection::connection::ConnectionEnd; @@ -23,13 +24,15 @@ pub trait ClientDef: Clone { type ClientState: ClientState; type ConsensusState: ConsensusState; - /// TODO fn check_header_and_update_state( &self, + ctx: &dyn ClientReader, + client_id: ClientId, client_state: Self::ClientState, header: Self::Header, ) -> Result<(Self::ClientState, Self::ConsensusState), Error>; + /// TODO fn verify_upgrade_and_update_state( &self, client_state: &Self::ClientState, @@ -156,7 +159,7 @@ pub enum AnyClient { impl AnyClient { pub fn from_client_type(client_type: ClientType) -> AnyClient { match client_type { - ClientType::Tendermint => Self::Tendermint(TendermintClient), + ClientType::Tendermint => Self::Tendermint(TendermintClient::default()), #[cfg(any(test, feature = "mocks"))] ClientType::Mock => Self::Mock(MockClient), @@ -173,6 +176,8 @@ impl ClientDef for AnyClient { /// Validates an incoming `header` against the latest consensus state of this client. fn check_header_and_update_state( &self, + ctx: &dyn ClientReader, + client_id: ClientId, client_state: AnyClientState, header: AnyHeader, ) -> Result<(AnyClientState, AnyConsensusState), Error> { @@ -185,7 +190,7 @@ impl ClientDef for AnyClient { .ok_or_else(|| Error::client_args_type_mismatch(ClientType::Tendermint))?; let (new_state, new_consensus) = - client.check_header_and_update_state(client_state, header)?; + client.check_header_and_update_state(ctx, client_id, client_state, header)?; Ok(( AnyClientState::Tendermint(new_state), @@ -202,7 +207,7 @@ impl ClientDef for AnyClient { .ok_or_else(|| Error::client_args_type_mismatch(ClientType::Mock))?; let (new_state, new_consensus) = - client.check_header_and_update_state(client_state, header)?; + client.check_header_and_update_state(ctx, client_id, client_state, header)?; Ok(( AnyClientState::Mock(new_state), diff --git a/modules/src/ics02_client/context.rs b/modules/src/ics02_client/context.rs index 611d4ae8d..aaf08ee1d 100644 --- a/modules/src/ics02_client/context.rs +++ b/modules/src/ics02_client/context.rs @@ -5,7 +5,7 @@ use crate::ics02_client::client_consensus::AnyConsensusState; use crate::ics02_client::client_state::AnyClientState; use crate::ics02_client::client_type::ClientType; -use crate::ics02_client::error::Error; +use crate::ics02_client::error::{Error, ErrorDetail}; use crate::ics02_client::handler::ClientResult::{self, Create, Update, Upgrade}; use crate::ics24_host::identifier::ClientId; use crate::Height; @@ -14,12 +14,47 @@ use crate::Height; pub trait ClientReader { fn client_type(&self, client_id: &ClientId) -> Result; fn client_state(&self, client_id: &ClientId) -> Result; + + /// Retrieve the consensus state for the given client ID at the specified + /// height. + /// + /// Returns an error if no such state exists. fn consensus_state( &self, client_id: &ClientId, height: Height, ) -> Result; + /// Similar to `consensus_state`, attempt to retrieve the consensus state, + /// but return `None` if no state exists at the given height. + fn maybe_consensus_state( + &self, + client_id: &ClientId, + height: Height, + ) -> Result, Error> { + match self.consensus_state(client_id, height) { + Ok(cs) => Ok(Some(cs)), + Err(e) => match e.detail() { + ErrorDetail::ConsensusStateNotFound(_) => Ok(None), + _ => Err(e), + }, + } + } + + /// Search for the lowest consensus state higher than `height`. + fn next_consensus_state( + &self, + client_id: &ClientId, + height: Height, + ) -> Result, Error>; + + /// Search for the highest consensus state lower than `height`. + fn prev_consensus_state( + &self, + client_id: &ClientId, + height: Height, + ) -> Result, Error>; + /// Returns a natural number, counting how many clients have been created thus far. /// The value of this counter should increase only via method `ClientKeeper::increase_client_counter`. fn client_counter(&self) -> Result; diff --git a/modules/src/ics02_client/error.rs b/modules/src/ics02_client/error.rs index 4eb0c5495..5839a44c7 100644 --- a/modules/src/ics02_client/error.rs +++ b/modules/src/ics02_client/error.rs @@ -9,8 +9,12 @@ use crate::ics07_tendermint::error::Error as Ics07Error; use crate::ics23_commitment::error::Error as Ics23Error; use crate::ics24_host::error::ValidationError; use crate::ics24_host::identifier::ClientId; +use crate::timestamp::Timestamp; use crate::Height; +use tendermint::Error as TendermintError; +use tendermint_proto::Error as TendermintProtoError; + define_error! { #[derive(Debug, PartialEq, Eq)] Error { @@ -58,7 +62,7 @@ define_error! { FailedTrustThresholdConversion { numerator: u64, denominator: u64 } - [ tendermint::Error ] + [ TendermintError ] | e | { format_args!("failed to build Tendermint domain type trust threshold from fraction: {}/{}", e.numerator, e.denominator) }, UnknownClientStateType @@ -105,14 +109,14 @@ define_error! { }, DecodeRawClientState - [ TraceError ] + [ TraceError ] | _ | { "error decoding raw client state" }, MissingRawClientState | _ | { "missing raw client state" }, InvalidRawConsensusState - [ TraceError ] + [ TraceError ] | _ | { "invalid raw client consensus state" }, MissingRawConsensusState @@ -134,14 +138,14 @@ define_error! { | _ | { "invalid client identifier" }, InvalidRawHeader - [ TraceError ] + [ TraceError ] | _ | { "invalid raw header" }, MissingRawHeader | _ | { "missing raw header" }, DecodeRawMisbehaviour - [ TraceError ] + [ TraceError ] | _ | { "invalid raw misbehaviour" }, InvalidRawMisbehaviour @@ -180,6 +184,12 @@ define_error! { e.client_type) }, + InsufficientVotingPower + { reason: String } + | e | { + format_args!("Insufficient overlap {}", e.reason) + }, + RawClientAndConsensusStateTypesMismatch { state_type: ClientType, @@ -206,8 +216,37 @@ define_error! { client_height: Height, } | e | { - format_args!("upgraded client height {0} must be at greater than current client height {1}", + format_args!("upgraded client height {} must be at greater than current client height {}", e.upgraded_height, e.client_height) }, + + InvalidConsensusStateTimestamp + { + time1: Timestamp, + time2: Timestamp, + } + | e | { + format_args!("timestamp is invalid or missing, timestamp={0}, now={1}", e.time1, e.time2) + }, + + HeaderNotWithinTrustPeriod + { + latest_time:Timestamp, + update_time: Timestamp, + } + | e | { + format_args!("header not withing trusting period: expires_at={0} now={1}", e.latest_time, e.update_time) + }, + + TendermintHandlerError + [ Ics07Error ] + | _ | { format_args!("Tendermint-specific handler error") }, + + } +} + +impl From for Error { + fn from(e: Ics07Error) -> Error { + Error::tendermint_handler_error(e) } } diff --git a/modules/src/ics02_client/handler/update_client.rs b/modules/src/ics02_client/handler/update_client.rs index 0c993d28f..94aa2fcb8 100644 --- a/modules/src/ics02_client/handler/update_client.rs +++ b/modules/src/ics02_client/handler/update_client.rs @@ -1,17 +1,21 @@ //! Protocol logic specific to processing ICS2 messages of type `MsgUpdateAnyClient`. +use tracing::debug; + use crate::events::IbcEvent; use crate::handler::{HandlerOutput, HandlerResult}; use crate::ics02_client::client_consensus::AnyConsensusState; use crate::ics02_client::client_def::{AnyClient, ClientDef}; -use crate::ics02_client::client_state::AnyClientState; +use crate::ics02_client::client_state::{AnyClientState, ClientState}; use crate::ics02_client::context::ClientReader; use crate::ics02_client::error::Error; use crate::ics02_client::events::Attributes; use crate::ics02_client::handler::ClientResult; +use crate::ics02_client::header::Header; use crate::ics02_client::msgs::update_client::MsgUpdateAnyClient; use crate::ics24_host::identifier::ClientId; use crate::prelude::*; +use crate::timestamp::Timestamp; /// The result following the successful processing of a `MsgUpdateAnyClient` message. Preferably /// this data type should be used with a qualified name `update_client::Result` to avoid ambiguity. @@ -42,14 +46,40 @@ pub fn process( // Read client state from the host chain store. let client_state = ctx.client_state(&client_id)?; - let latest_height = client_state.latest_height(); - ctx.consensus_state(&client_id, latest_height)?; + if client_state.is_frozen() { + return Err(Error::client_frozen(client_id)); + } + + // Read consensus state from the host chain store. + let latest_consensus_state = ctx + .consensus_state(&client_id, client_state.latest_height()) + .map_err(|_| { + Error::consensus_state_not_found(client_id.clone(), client_state.latest_height()) + })?; + + debug!("latest consensus state: {:?}", latest_consensus_state); + + let duration = Timestamp::now() + .duration_since(&latest_consensus_state.timestamp()) + .ok_or_else(|| { + Error::invalid_consensus_state_timestamp( + latest_consensus_state.timestamp(), + header.timestamp(), + ) + })?; + + if client_state.expired(duration) { + return Err(Error::header_not_within_trust_period( + latest_consensus_state.timestamp(), + header.timestamp(), + )); + } // Use client_state to validate the new header against the latest consensus_state. // This function will return the new client_state (its latest_height changed) and a // consensus_state obtained from header. These will be later persisted by the keeper. let (new_client_state, new_consensus_state) = client_def - .check_header_and_update_state(client_state, header) + .check_header_and_update_state(ctx, client_id.clone(), client_state, header) .map_err(|e| Error::header_verification_failure(e.to_string()))?; let result = ClientResult::Update(Result { @@ -74,19 +104,23 @@ mod tests { use crate::events::IbcEvent; use crate::handler::HandlerOutput; - use crate::ics02_client::client_state::AnyClientState; + use crate::ics02_client::client_consensus::AnyConsensusState; + use crate::ics02_client::client_state::{AnyClientState, ClientState}; + use crate::ics02_client::client_type::ClientType; use crate::ics02_client::error::{Error, ErrorDetail}; use crate::ics02_client::handler::dispatch; use crate::ics02_client::handler::ClientResult::Update; - use crate::ics02_client::header::Header; + use crate::ics02_client::header::{AnyHeader, Header}; use crate::ics02_client::msgs::update_client::MsgUpdateAnyClient; use crate::ics02_client::msgs::ClientMsg; - use crate::ics24_host::identifier::ClientId; + use crate::ics24_host::identifier::{ChainId, ClientId}; use crate::mock::client_state::MockClientState; use crate::mock::context::MockContext; use crate::mock::header::MockHeader; + use crate::mock::host::HostType; use crate::prelude::*; use crate::test_utils::get_dummy_account_id; + use crate::timestamp::Timestamp; use crate::Height; #[test] @@ -94,10 +128,14 @@ mod tests { let client_id = ClientId::default(); let signer = get_dummy_account_id(); + let timestamp = Timestamp::now(); + let ctx = MockContext::default().with_client(&client_id, Height::new(0, 42)); let msg = MsgUpdateAnyClient { client_id: client_id.clone(), - header: MockHeader::new(Height::new(0, 46)).into(), + header: MockHeader::new(Height::new(0, 46)) + .with_timestamp(timestamp) + .into(), signer, }; @@ -121,9 +159,9 @@ mod tests { assert_eq!(upd_res.client_id, client_id); assert_eq!( upd_res.client_state, - AnyClientState::Mock(MockClientState(MockHeader::new( - msg.header.height() - ))) + AnyClientState::Mock(MockClientState( + MockHeader::new(msg.header.height()).with_timestamp(timestamp) + )) ) } _ => panic!("update handler result has incorrect type"), @@ -205,4 +243,293 @@ mod tests { } } } + + #[test] + fn test_update_synthetic_tendermint_client_adjacent_ok() { + let client_id = ClientId::new(ClientType::Tendermint, 0).unwrap(); + let client_height = Height::new(1, 20); + let update_height = Height::new(1, 21); + + let ctx = MockContext::new( + ChainId::new("mockgaiaA".to_string(), 1), + HostType::Mock, + 5, + Height::new(1, 1), + ) + .with_client_parametrized( + &client_id, + client_height, + Some(ClientType::Tendermint), // The target host chain (B) is synthetic TM. + Some(client_height), + ); + + let ctx_b = MockContext::new( + ChainId::new("mockgaiaB".to_string(), 1), + HostType::SyntheticTendermint, + 5, + update_height, + ); + + let signer = get_dummy_account_id(); + + let block_ref = ctx_b.host_block(update_height); + let mut latest_header: AnyHeader = block_ref.cloned().map(Into::into).unwrap(); + + latest_header = match latest_header { + AnyHeader::Tendermint(mut theader) => { + theader.trusted_height = client_height; + AnyHeader::Tendermint(theader) + } + AnyHeader::Mock(m) => AnyHeader::Mock(m), + }; + + let msg = MsgUpdateAnyClient { + client_id: client_id.clone(), + header: latest_header, + signer, + }; + + let output = dispatch(&ctx, ClientMsg::UpdateClient(msg.clone())); + + match output { + Ok(HandlerOutput { + result, + mut events, + log, + }) => { + assert_eq!(events.len(), 1); + let event = events.pop().unwrap(); + assert!( + matches!(event, IbcEvent::UpdateClient(e) if e.client_id() == &msg.client_id) + ); + assert!(log.is_empty()); + // Check the result + match result { + Update(upd_res) => { + assert_eq!(upd_res.client_id, client_id); + assert!(!upd_res.client_state.is_frozen()); + assert_eq!(upd_res.client_state.latest_height(), msg.header.height(),) + } + _ => panic!("update handler result has incorrect type"), + } + } + Err(err) => { + panic!("unexpected error: {}", err); + } + } + } + + #[test] + fn test_update_synthetic_tendermint_client_non_adjacent_ok() { + let client_id = ClientId::new(ClientType::Tendermint, 0).unwrap(); + let client_height = Height::new(1, 20); + let update_height = Height::new(1, 21); + + let ctx = MockContext::new( + ChainId::new("mockgaiaA".to_string(), 1), + HostType::Mock, + 5, + Height::new(1, 1), + ) + .with_client_parametrized_history( + &client_id, + client_height, + Some(ClientType::Tendermint), // The target host chain (B) is synthetic TM. + Some(client_height), + ); + + let ctx_b = MockContext::new( + ChainId::new("mockgaiaB".to_string(), 1), + HostType::SyntheticTendermint, + 5, + update_height, + ); + + let signer = get_dummy_account_id(); + + let block_ref = ctx_b.host_block(update_height); + let mut latest_header: AnyHeader = block_ref.cloned().map(Into::into).unwrap(); + + let trusted_height = client_height.clone().sub(1).unwrap_or_default(); + + latest_header = match latest_header { + AnyHeader::Tendermint(mut theader) => { + theader.trusted_height = trusted_height; + AnyHeader::Tendermint(theader) + } + AnyHeader::Mock(m) => AnyHeader::Mock(m), + }; + + let msg = MsgUpdateAnyClient { + client_id: client_id.clone(), + header: latest_header, + signer, + }; + + let output = dispatch(&ctx, ClientMsg::UpdateClient(msg.clone())); + + match output { + Ok(HandlerOutput { + result, + mut events, + log, + }) => { + assert_eq!(events.len(), 1); + let event = events.pop().unwrap(); + assert!( + matches!(event, IbcEvent::UpdateClient(e) if e.client_id() == &msg.client_id) + ); + assert!(log.is_empty()); + // Check the result + match result { + Update(upd_res) => { + assert_eq!(upd_res.client_id, client_id); + assert!(!upd_res.client_state.is_frozen()); + assert_eq!(upd_res.client_state.latest_height(), msg.header.height(),) + } + _ => panic!("update handler result has incorrect type"), + } + } + Err(err) => { + panic!("unexpected error: {}", err); + } + } + } + + #[test] + fn test_update_synthetic_tendermint_client_duplicate_ok() { + let client_id = ClientId::new(ClientType::Tendermint, 0).unwrap(); + let client_height = Height::new(1, 20); + + let chain_start_height = Height::new(1, 11); + + let ctx = MockContext::new( + ChainId::new("mockgaiaA".to_string(), 1), + HostType::Mock, + 5, + chain_start_height, + ) + .with_client_parametrized( + &client_id, + client_height, + Some(ClientType::Tendermint), // The target host chain (B) is synthetic TM. + Some(client_height), + ); + + let ctx_b = MockContext::new( + ChainId::new("mockgaiaB".to_string(), 1), + HostType::SyntheticTendermint, + 5, + client_height, + ); + + let signer = get_dummy_account_id(); + + let block_ref = ctx_b.host_block(client_height); + let latest_header: AnyHeader = match block_ref.cloned().map(Into::into).unwrap() { + AnyHeader::Tendermint(mut theader) => { + let cons_state = ctx + .latest_consensus_states(&client_id, &client_height) + .clone(); + if let AnyConsensusState::Tendermint(tcs) = cons_state { + theader.signed_header.header.time = tcs.timestamp; + theader.trusted_height = Height::new(1, 11) + } + AnyHeader::Tendermint(theader) + } + AnyHeader::Mock(header) => AnyHeader::Mock(header), + }; + + let msg = MsgUpdateAnyClient { + client_id: client_id.clone(), + header: latest_header, + signer, + }; + + let output = dispatch(&ctx, ClientMsg::UpdateClient(msg.clone())); + + match output { + Ok(HandlerOutput { + result, + mut events, + log, + }) => { + assert_eq!(events.len(), 1); + let event = events.pop().unwrap(); + assert!( + matches!(event, IbcEvent::UpdateClient(e) if e.client_id() == &msg.client_id) + ); + assert!(log.is_empty()); + // Check the result + match result { + Update(upd_res) => { + assert_eq!(upd_res.client_id, client_id); + assert!(!upd_res.client_state.is_frozen()); + assert_eq!( + upd_res.client_state, + ctx.latest_client_states(&client_id).clone() + ); + assert_eq!(upd_res.client_state.latest_height(), msg.header.height(),) + } + _ => panic!("update handler result has incorrect type"), + } + } + Err(err) => { + panic!("unexpected error: {:?}", err); + } + } + } + + #[test] + fn test_update_synthetic_tendermint_client_lower_height() { + let client_id = ClientId::new(ClientType::Tendermint, 0).unwrap(); + let client_height = Height::new(1, 20); + + let client_update_height = Height::new(1, 19); + + let chain_start_height = Height::new(1, 11); + + let ctx = MockContext::new( + ChainId::new("mockgaiaA".to_string(), 1), + HostType::Mock, + 5, + chain_start_height, + ) + .with_client_parametrized( + &client_id, + client_height, + Some(ClientType::Tendermint), // The target host chain (B) is synthetic TM. + Some(client_height), + ); + + let ctx_b = MockContext::new( + ChainId::new("mockgaiaB".to_string(), 1), + HostType::SyntheticTendermint, + 5, + client_height, + ); + + let signer = get_dummy_account_id(); + + let block_ref = ctx_b.host_block(client_update_height); + let latest_header: AnyHeader = block_ref.cloned().map(Into::into).unwrap(); + + let msg = MsgUpdateAnyClient { + client_id, + header: latest_header, + signer, + }; + + let output = dispatch(&ctx, ClientMsg::UpdateClient(msg)); + + match output { + Ok(_) => { + panic!("update handler result has incorrect type"); + } + Err(err) => match err.detail() { + ErrorDetail::HeaderVerificationFailure(_) => {} + _ => panic!("unexpected error: {:?}", err), + }, + } + } } diff --git a/modules/src/ics02_client/header.rs b/modules/src/ics02_client/header.rs index caf23d598..675b375a2 100644 --- a/modules/src/ics02_client/header.rs +++ b/modules/src/ics02_client/header.rs @@ -10,6 +10,7 @@ use crate::ics02_client::error::Error; use crate::ics07_tendermint::header::{decode_header, Header as TendermintHeader}; #[cfg(any(test, feature = "mocks"))] use crate::mock::header::MockHeader; +use crate::timestamp::Timestamp; use crate::Height; pub const TENDERMINT_HEADER_TYPE_URL: &str = "/ibc.lightclients.tendermint.v1.Header"; @@ -23,6 +24,9 @@ pub trait Header: Clone + core::fmt::Debug + Send + Sync { /// The height of the consensus state fn height(&self) -> Height; + /// The timestamp of the consensus state + fn timestamp(&self) -> Timestamp; + /// Wrap into an `AnyHeader` fn wrap_any(self) -> AnyHeader; } @@ -55,6 +59,15 @@ impl Header for AnyHeader { } } + fn timestamp(&self) -> Timestamp { + match self { + Self::Tendermint(header) => header.timestamp(), + + #[cfg(any(test, feature = "mocks"))] + Self::Mock(header) => header.timestamp, + } + } + fn wrap_any(self) -> AnyHeader { self } diff --git a/modules/src/ics04_channel/handler/send_packet.rs b/modules/src/ics04_channel/handler/send_packet.rs index 603b4a928..0b118a21d 100644 --- a/modules/src/ics04_channel/handler/send_packet.rs +++ b/modules/src/ics04_channel/handler/send_packet.rs @@ -127,7 +127,10 @@ mod tests { use crate::ics04_channel::packet::Packet; use crate::ics24_host::identifier::{ChannelId, ClientId, ConnectionId, PortId}; use crate::mock::context::MockContext; + use crate::timestamp::Timestamp; use crate::timestamp::ZERO_DURATION; + use std::ops::Add; + use std::time::Duration; #[test] fn send_packet_processing() { @@ -140,7 +143,11 @@ mod tests { let context = MockContext::default(); - let mut packet: Packet = get_dummy_raw_packet(1, 6).try_into().unwrap(); + let timestamp = Timestamp::now().add(Duration::from_secs(10)); + //CD:TODO remove unwrap + let mut packet: Packet = get_dummy_raw_packet(1, timestamp.unwrap().as_nanoseconds()) + .try_into() + .unwrap(); packet.sequence = 1.into(); packet.data = vec![0]; diff --git a/modules/src/ics07_tendermint/client_def.rs b/modules/src/ics07_tendermint/client_def.rs index 2cb6c9d75..a3b47bc97 100644 --- a/modules/src/ics07_tendermint/client_def.rs +++ b/modules/src/ics07_tendermint/client_def.rs @@ -1,23 +1,44 @@ +use std::convert::TryInto; + use ibc_proto::ibc::core::commitment::v1::MerkleProof; +use tendermint::Time; +use tendermint_light_client::components::verifier::{ProdVerifier, Verdict, Verifier}; +use tendermint_light_client::types::{TrustedBlockState, UntrustedBlockState}; use crate::ics02_client::client_consensus::AnyConsensusState; use crate::ics02_client::client_def::ClientDef; use crate::ics02_client::client_state::AnyClientState; -use crate::ics02_client::error::Error; +use crate::ics02_client::client_type::ClientType; +use crate::ics02_client::context::ClientReader; +use crate::ics02_client::error::Error as Ics02Error; use crate::ics03_connection::connection::ConnectionEnd; use crate::ics04_channel::channel::ChannelEnd; use crate::ics04_channel::packet::Sequence; use crate::ics07_tendermint::client_state::ClientState; use crate::ics07_tendermint::consensus_state::ConsensusState; +use crate::ics07_tendermint::error::Error; use crate::ics07_tendermint::header::Header; + use crate::ics23_commitment::commitment::{CommitmentPrefix, CommitmentProofBytes, CommitmentRoot}; use crate::ics24_host::identifier::ConnectionId; use crate::ics24_host::identifier::{ChannelId, ClientId, PortId}; use crate::prelude::*; use crate::Height; +use crate::downcast; + #[derive(Clone, Debug, PartialEq, Eq)] -pub struct TendermintClient; +pub struct TendermintClient { + verifier: ProdVerifier, +} + +impl Default for TendermintClient { + fn default() -> Self { + Self { + verifier: ProdVerifier::default(), + } + } +} impl ClientDef for TendermintClient { type Header = Header; @@ -26,17 +47,137 @@ impl ClientDef for TendermintClient { fn check_header_and_update_state( &self, + ctx: &dyn ClientReader, + client_id: ClientId, client_state: Self::ClientState, header: Self::Header, - ) -> Result<(Self::ClientState, Self::ConsensusState), Error> { - if client_state.latest_height() >= header.height() { - return Err(Error::low_header_height( - header.height(), - client_state.latest_height(), + ) -> Result<(Self::ClientState, Self::ConsensusState), Ics02Error> { + if header.height().revision_number != client_state.chain_id.version() { + return Err(Ics02Error::tendermint_handler_error( + Error::mismatched_revisions( + client_state.chain_id.version(), + header.height().revision_number, + ), )); } - // TODO: Additional verifications should be implemented here. + // Check if a consensus state is already installed; if so it should + // match the untrusted header. + let header_consensus_state = ConsensusState::from(header.clone()); + let existing_consensus_state = + match ctx.maybe_consensus_state(&client_id, header.height())? { + Some(cs) => { + let cs = downcast_consensus_state(cs)?; + // If this consensus state matches, skip verification + // (optimization) + if cs == header_consensus_state { + // Header is already installed and matches the incoming + // header (already verified) + return Ok((client_state, cs)); + } + Some(cs) + } + None => None, + }; + + let trusted_consensus_state = + downcast_consensus_state(ctx.consensus_state(&client_id, header.trusted_height)?)?; + + let trusted_state = TrustedBlockState { + header_time: trusted_consensus_state.timestamp, + height: header + .trusted_height + .revision_height + .try_into() + .map_err(|_| { + Ics02Error::tendermint_handler_error(Error::invalid_header_height( + header.trusted_height, + )) + })?, + next_validators: &header.trusted_validator_set, + next_validators_hash: trusted_consensus_state.next_validators_hash, + }; + + let untrusted_state = UntrustedBlockState { + signed_header: &header.signed_header, + validators: &header.validator_set, + // NB: This will skip the + // VerificationPredicates::next_validators_match check for the + // untrusted state. + next_validators: None, + }; + + let options = client_state.as_light_client_options()?; + + match self + .verifier + .verify(untrusted_state, trusted_state, &options, Time::now()) + { + Verdict::Success => {} + Verdict::NotEnoughTrust(voting_power_tally) => { + return Err(Error::not_enough_trusted_vals_signed(format!( + "voting power tally: {}", + voting_power_tally + )) + .into()) + } + Verdict::Invalid(detail) => { + return Err(Ics02Error::tendermint_handler_error( + Error::verification_error(detail), + )) + } + } + + // If the header has verified, but its corresponding consensus state + // differs from the existing consensus state for that height, freeze the + // client and return the installed consensus state. + if let Some(cs) = existing_consensus_state { + if cs != header_consensus_state { + return Ok((client_state.with_set_frozen(header.height()), cs)); + } + } + + // Monotonicity checks for timestamps for in-the-middle updates + // (cs-new, cs-next, cs-latest) + if header.height() < client_state.latest_height() { + let maybe_next_cs = ctx + .next_consensus_state(&client_id, header.height())? + .map(downcast_consensus_state) + .transpose()?; + + if let Some(next_cs) = maybe_next_cs { + // New (untrusted) header timestamp cannot occur after next + // consensus state's height + if header.signed_header.header().time > next_cs.timestamp { + return Err(Ics02Error::tendermint_handler_error( + Error::header_timestamp_too_high( + header.signed_header.header().time.to_string(), + next_cs.timestamp.to_string(), + ), + )); + } + } + } + // (cs-trusted, cs-prev, cs-new) + if header.trusted_height < header.height() { + let maybe_prev_cs = ctx + .prev_consensus_state(&client_id, header.height())? + .map(downcast_consensus_state) + .transpose()?; + + if let Some(prev_cs) = maybe_prev_cs { + // New (untrusted) header timestamp cannot occur before the + // previous consensus state's height + if header.signed_header.header().time < prev_cs.timestamp { + return Err(Ics02Error::tendermint_handler_error( + Error::header_timestamp_too_low( + header.signed_header.header().time.to_string(), + prev_cs.timestamp.to_string(), + ), + )); + } + } + } Ok(( client_state.with_header(header.clone()), @@ -53,7 +194,7 @@ impl ClientDef for TendermintClient { _client_id: &ClientId, _consensus_height: Height, _expected_consensus_state: &AnyConsensusState, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { todo!() } @@ -65,7 +206,7 @@ impl ClientDef for TendermintClient { _proof: &CommitmentProofBytes, _connection_id: Option<&ConnectionId>, _expected_connection_end: &ConnectionEnd, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { todo!() } @@ -78,7 +219,7 @@ impl ClientDef for TendermintClient { _port_id: &PortId, _channel_id: &ChannelId, _expected_channel_end: &ChannelEnd, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { todo!() } @@ -91,7 +232,7 @@ impl ClientDef for TendermintClient { _client_id: &ClientId, _proof: &CommitmentProofBytes, _expected_client_state: &AnyClientState, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { unimplemented!() } @@ -104,7 +245,7 @@ impl ClientDef for TendermintClient { _channel_id: &ChannelId, _seq: &Sequence, _data: String, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { todo!() } @@ -117,7 +258,7 @@ impl ClientDef for TendermintClient { _channel_id: &ChannelId, _seq: &Sequence, _data: Vec, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { todo!() } @@ -129,7 +270,7 @@ impl ClientDef for TendermintClient { _port_id: &PortId, _channel_id: &ChannelId, _seq: &Sequence, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { todo!() } @@ -141,7 +282,7 @@ impl ClientDef for TendermintClient { _port_id: &PortId, _channel_id: &ChannelId, _seq: &Sequence, - ) -> Result<(), Error> { + ) -> Result<(), Ics02Error> { todo!() } @@ -151,7 +292,14 @@ impl ClientDef for TendermintClient { _consensus_state: &Self::ConsensusState, _proof_upgrade_client: MerkleProof, _proof_upgrade_consensus_state: MerkleProof, - ) -> Result<(Self::ClientState, Self::ConsensusState), Error> { + ) -> Result<(Self::ClientState, Self::ConsensusState), Ics02Error> { todo!() } } + +fn downcast_consensus_state(cs: AnyConsensusState) -> Result { + downcast!( + cs => AnyConsensusState::Tendermint + ) + .ok_or_else(|| Ics02Error::client_args_type_mismatch(ClientType::Tendermint)) +} diff --git a/modules/src/ics07_tendermint/client_state.rs b/modules/src/ics07_tendermint/client_state.rs index 7fe85a0d7..ea2bcbadd 100644 --- a/modules/src/ics07_tendermint/client_state.rs +++ b/modules/src/ics07_tendermint/client_state.rs @@ -4,12 +4,14 @@ use core::str::FromStr; use core::time::Duration; use serde::{Deserialize, Serialize}; +use tendermint_light_client::light_client::Options; use tendermint_proto::Protobuf; use ibc_proto::ibc::lightclients::tendermint::v1::ClientState as RawClientState; use crate::ics02_client::client_state::AnyClientState; use crate::ics02_client::client_type::ClientType; +use crate::ics02_client::error::Error as Ics02Error; use crate::ics02_client::trust_threshold::TrustThreshold; use crate::ics07_tendermint::error::Error; use crate::ics07_tendermint::header::Header; @@ -116,7 +118,14 @@ impl ClientState { } } - /// Helper function for the upgrade chain & client procedures. + pub fn with_set_frozen(self, h: Height) -> Self { + Self { + frozen_height: h, + ..self + } + } + + /// Helper function to verify the upgrade client procedure. /// Resets all fields except the blockchain-specific ones. pub fn zero_custom_fields(mut client_state: Self) -> Self { client_state.trusting_period = ZERO_DURATION; @@ -138,6 +147,20 @@ impl ClientState { pub fn expired(&self, elapsed: Duration) -> bool { elapsed > self.trusting_period } + + /// Helper method to produce a + /// [`tendermint_light_client::light_client::Options`] struct for use in + /// Tendermint-specific light client verification. + pub fn as_light_client_options(&self) -> Result { + Ok(Options { + trust_threshold: self + .trust_level + .try_into() + .map_err(|e: Ics02Error| Error::invalid_trust_threshold(e.to_string()))?, + trusting_period: self.trusting_period, + clock_drift: self.max_clock_drift, + }) + } } impl crate::ics02_client::client_state::ClientState for ClientState { diff --git a/modules/src/ics07_tendermint/error.rs b/modules/src/ics07_tendermint/error.rs index a67b69e74..83029f617 100644 --- a/modules/src/ics07_tendermint/error.rs +++ b/modules/src/ics07_tendermint/error.rs @@ -3,6 +3,10 @@ use crate::prelude::*; use flex_error::{define_error, TraceError}; use crate::ics24_host::error::ValidationError; +use crate::Height; +use tendermint::account::Id; +use tendermint::hash::Hash; +use tendermint::Error as TendermintError; define_error! { #[derive(Debug, PartialEq, Eq)] @@ -16,11 +20,11 @@ define_error! { |e| { format_args!("invalid unbonding period: {}", e.reason) }, InvalidAddress - | _ | { "invalid address" }, + |_| { "invalid address" }, InvalidHeader { reason: String } - [ tendermint::Error ] + [ TendermintError ] |e| { format_args!("invalid header, failed basic validation: {}", e.reason) }, InvalidTrustThreshold @@ -86,18 +90,134 @@ define_error! { InvalidRawConsensusState { reason: String } - |e| { format_args!("invalid raw client consensus state: {}", e.reason) }, + | e | { format_args!("invalid raw client consensus state: {}", e.reason) }, InvalidRawHeader - [ tendermint::Error ] - |_| { "invalid raw header" }, + [ TendermintError ] + | _ | { "invalid raw header" }, InvalidRawMisbehaviour { reason: String } - |e| { format_args!("invalid raw misbehaviour: {}", e.reason) }, + | e | { format_args!("invalid raw misbehaviour: {}", e.reason) }, Decode [ TraceError ] - |_| { "decode error" }, + | _ | { "decode error" }, + + InsufficientVotingPower + { reason: String } + | e | { + format_args!("insufficient overlap: {}", e.reason) + }, + + LowUpdateTimestamp + { + low: String, + high: String + } + | e | { + format_args!("header timestamp {0} must be greater than current client consensus state timestamp {1}", e.low, e.high) + }, + + HeaderTimestampOutsideTrustingTime + { + low: String, + high: String + } + | e | { + format_args!("header timestamp {0} is outside the trusting period w.r.t. consensus state timestamp {1}", e.low, e.high) + }, + + HeaderTimestampTooHigh + { + actual: String, + max: String, + } + | e | { + format_args!("given other previous updates, header timestamp should be at most {0}, but was {1}", e.max, e.actual) + }, + + HeaderTimestampTooLow + { + actual: String, + min: String, + } + | e | { + format_args!("given other previous updates, header timestamp should be at least {0}, but was {1}", e.min, e.actual) + }, + + InvalidHeaderHeight + { height: Height } + | e | { + format_args!("header height = {0} is invalid", e.height) + }, + + InvalidTrustedHeaderHeight + { + trusted_header_height: Height, + height_header: Height + } + | e | { + format_args!("header height is {0} and is lower than the trusted header height, which is {1} ", e.height_header, e.trusted_header_height) + }, + + LowUpdateHeight + { + low: Height, + high: Height + } + | e | { + format_args!("header height is {0} but it must be greater than the current client height which is {1}", e.low, e.high) + }, + + MismatchedRevisions + { + current_revision: u64, + update_revision: u64, + } + | e | { + format_args!("the header's current/trusted revision number ({0}) and the update's revision number ({1}) should be the same", e.current_revision, e.update_revision) + }, + + InvalidValidatorSet + { + hash1: Hash, + hash2: Hash, + } + | e | { + format_args!("invalid validator set: header_validators_hash={} and validators_hash={}", e.hash1, e.hash2) + }, + + NotEnoughTrustedValsSigned + { reason: String } + | e | { + format_args!("not enough trust because insufficient validators overlap: {}", e.reason) + }, + + VerificationError + { detail: tendermint_light_client::predicates::errors::VerificationErrorDetail } + | e | { + format_args!("verification failed: {}", e.detail) + } + } +} + +define_error! { + #[derive(Debug, PartialEq, Eq)] + VerificationError { + InvalidSignature + | _ | { "couldn't verify validator signature" }, + + DuplicateValidator + { id: Id } + | e | { + format_args!("duplicate validator in commit signatures with address {}", e.id) + }, + + InsufficientOverlap + { q1: u64, q2: u64 } + | e | { + format_args!("insufficient signers overlap between {0} and {1}", e.q1, e.q2) + }, } } diff --git a/modules/src/ics07_tendermint/header.rs b/modules/src/ics07_tendermint/header.rs index abe865405..bd16911ce 100644 --- a/modules/src/ics07_tendermint/header.rs +++ b/modules/src/ics07_tendermint/header.rs @@ -9,6 +9,8 @@ use tendermint::validator::Set as ValidatorSet; use tendermint::Time; use tendermint_proto::Protobuf; +use crate::timestamp::Timestamp; + use ibc_proto::ibc::lightclients::tendermint::v1::Header as RawHeader; use crate::ics02_client::client_type::ClientType; @@ -24,6 +26,7 @@ pub struct Header { pub signed_header: SignedHeader, // contains the commitment root pub validator_set: ValidatorSet, // the validator set that signed Header pub trusted_height: Height, // the height of a trusted header seen by client less than or equal to Header + // TODO(thane): Rename this to trusted_next_validator_set? pub trusted_validator_set: ValidatorSet, // the last trusted validator set at trusted height } @@ -79,6 +82,10 @@ impl crate::ics02_client::header::Header for Header { self.height() } + fn timestamp(&self) -> Timestamp { + self.time().into() + } + fn wrap_any(self) -> AnyHeader { AnyHeader::Tendermint(self) } @@ -90,7 +97,7 @@ impl TryFrom for Header { type Error = Error; fn try_from(raw: RawHeader) -> Result { - Ok(Self { + let header = Self { signed_header: raw .signed_header .ok_or_else(Error::missing_signed_header)? @@ -110,7 +117,16 @@ impl TryFrom for Header { .ok_or_else(Error::missing_trusted_validator_set)? .try_into() .map_err(Error::invalid_raw_header)?, - }) + }; + + if header.height().revision_number != header.trusted_height.revision_number { + return Err(Error::mismatched_revisions( + header.trusted_height.revision_number, + header.height().revision_number, + )); + } + + Ok(header) } } diff --git a/modules/src/ics18_relayer/utils.rs b/modules/src/ics18_relayer/utils.rs index 4bf7671ed..05166e0fc 100644 --- a/modules/src/ics18_relayer/utils.rs +++ b/modules/src/ics18_relayer/utils.rs @@ -50,7 +50,7 @@ where #[cfg(test)] mod tests { use crate::ics02_client::client_type::ClientType; - use crate::ics02_client::header::Header; + use crate::ics02_client::header::{AnyHeader, Header}; use crate::ics18_relayer::context::Ics18Context; use crate::ics18_relayer::utils::build_client_update_datagram; use crate::ics24_host::identifier::{ChainId, ClientId}; @@ -60,6 +60,7 @@ mod tests { use crate::prelude::*; use crate::Height; use test_env_log::test; + use tracing::debug; #[test] /// Serves to test both ICS 26 `dispatch` & `build_client_update_datagram` functions. @@ -150,7 +151,17 @@ mod tests { // Update client on chain B to latest height of B. // - create the client update message with the latest header from B - let b_latest_header = ctx_b.query_latest_header().unwrap(); + // The test uses LightClientBlock that does not store the trusted height + let b_latest_header = match ctx_b.query_latest_header().unwrap() { + AnyHeader::Tendermint(header) => { + let th = header.height(); + let mut hheader = header.clone(); + hheader.trusted_height = th.decrement().unwrap(); + hheader.wrap_any() + } + AnyHeader::Mock(header) => header.wrap_any(), + }; + assert_eq!( b_latest_header.client_type(), ClientType::Tendermint, @@ -171,6 +182,8 @@ mod tests { let client_msg_a = client_msg_a_res.unwrap(); + debug!("client_msg_a = {:?}", client_msg_a); + // - send the message to A let dispatch_res_a = ctx_a.deliver(Ics26Envelope::Ics2Msg(client_msg_a)); let validation_res = ctx_a.validate(); diff --git a/modules/src/ics26_routing/handler.rs b/modules/src/ics26_routing/handler.rs index d83b86c6f..f6d35e788 100644 --- a/modules/src/ics26_routing/handler.rs +++ b/modules/src/ics26_routing/handler.rs @@ -169,6 +169,7 @@ mod tests { use crate::mock::context::MockContext; use crate::mock::header::MockHeader; use crate::test_utils::get_dummy_account_id; + use crate::timestamp::Timestamp; use crate::Height; #[test] @@ -255,7 +256,6 @@ mod tests { .unwrap(); let msg_transfer = get_dummy_msg_transfer(35); - let msg_transfer_two = get_dummy_msg_transfer(36); let mut msg_to_on_close = @@ -299,7 +299,9 @@ mod tests { name: "Client update successful".to_string(), msg: Ics26Envelope::Ics2Msg(ClientMsg::UpdateClient(MsgUpdateAnyClient { client_id: client_id.clone(), - header: MockHeader::new(update_client_height).into(), + header: MockHeader::new(update_client_height) + .with_timestamp(Timestamp::now()) + .into(), signer: default_signer.clone(), })), want_pass: true, @@ -374,10 +376,12 @@ mod tests { // The client update is required in this test, because the proof associated with // msg_recv_packet has the same height as the packet TO height (see get_dummy_raw_msg_recv_packet) Test { - name: "Client update successful".to_string(), + name: "Client update successful #2".to_string(), msg: Ics26Envelope::Ics2Msg(ClientMsg::UpdateClient(MsgUpdateAnyClient { client_id: client_id.clone(), - header: MockHeader::new(update_client_height_after_send).into(), + header: MockHeader::new(update_client_height_after_send) + .with_timestamp(Timestamp::now()) + .into(), signer: default_signer.clone(), })), want_pass: true, diff --git a/modules/src/mock/client_def.rs b/modules/src/mock/client_def.rs index 4b520f1da..de4e01948 100644 --- a/modules/src/mock/client_def.rs +++ b/modules/src/mock/client_def.rs @@ -3,6 +3,7 @@ use ibc_proto::ibc::core::commitment::v1::MerkleProof; use crate::ics02_client::client_consensus::AnyConsensusState; use crate::ics02_client::client_def::ClientDef; use crate::ics02_client::client_state::AnyClientState; +use crate::ics02_client::context::ClientReader; use crate::ics02_client::error::Error; use crate::ics03_connection::connection::ConnectionEnd; use crate::ics04_channel::channel::ChannelEnd; @@ -26,6 +27,8 @@ impl ClientDef for MockClient { fn check_header_and_update_state( &self, + _ctx: &dyn ClientReader, + _client_id: ClientId, client_state: Self::ClientState, header: Self::Header, ) -> Result<(Self::ClientState, Self::ConsensusState), Error> { diff --git a/modules/src/mock/context.rs b/modules/src/mock/context.rs index 76828496b..c6631b6f3 100644 --- a/modules/src/mock/context.rs +++ b/modules/src/mock/context.rs @@ -1,9 +1,12 @@ //! Implementation of a global context mock. Used in testing handlers of all IBC modules. use crate::prelude::*; -use alloc::collections::btree_map::BTreeMap as HashMap; + +use alloc::collections::btree_map::BTreeMap; use core::cmp::min; +use tracing::debug; + use prost_types::Any; use sha2::Digest; @@ -63,49 +66,49 @@ pub struct MockContext { history: Vec, /// The set of all clients, indexed by their id. - clients: HashMap, + clients: BTreeMap, /// Counter for the client identifiers, necessary for `increase_client_counter` and the /// `client_counter` methods. client_ids_counter: u64, /// Association between client ids and connection ids. - client_connections: HashMap, + client_connections: BTreeMap, /// All the connections in the store. - connections: HashMap, + connections: BTreeMap, /// Counter for connection identifiers (see `increase_connection_counter`). connection_ids_counter: u64, /// Association between connection ids and channel ids. - connection_channels: HashMap>, + connection_channels: BTreeMap>, /// Counter for channel identifiers (see `increase_channel_counter`). channel_ids_counter: u64, /// All the channels in the store. TODO Make new key PortId X ChanneId - channels: HashMap<(PortId, ChannelId), ChannelEnd>, + channels: BTreeMap<(PortId, ChannelId), ChannelEnd>, /// Tracks the sequence number for the next packet to be sent. - next_sequence_send: HashMap<(PortId, ChannelId), Sequence>, + next_sequence_send: BTreeMap<(PortId, ChannelId), Sequence>, /// Tracks the sequence number for the next packet to be received. - next_sequence_recv: HashMap<(PortId, ChannelId), Sequence>, + next_sequence_recv: BTreeMap<(PortId, ChannelId), Sequence>, /// Tracks the sequence number for the next packet to be acknowledged. - next_sequence_ack: HashMap<(PortId, ChannelId), Sequence>, + next_sequence_ack: BTreeMap<(PortId, ChannelId), Sequence>, - packet_acknowledgement: HashMap<(PortId, ChannelId, Sequence), String>, + packet_acknowledgement: BTreeMap<(PortId, ChannelId, Sequence), String>, /// Maps ports to their capabilities - port_capabilities: HashMap, + port_capabilities: BTreeMap, /// Constant-size commitments to packets data fields - packet_commitment: HashMap<(PortId, ChannelId, Sequence), String>, + packet_commitment: BTreeMap<(PortId, ChannelId, Sequence), String>, // Used by unordered channel - packet_receipt: HashMap<(PortId, ChannelId, Sequence), Receipt>, + packet_receipt: BTreeMap<(PortId, ChannelId, Sequence), Receipt>, } /// Returns a MockContext with bare minimum initialization: no clients, no connections and no channels are @@ -218,6 +221,7 @@ impl MockContext { self.host_chain_id.clone(), cs_height.revision_height, ); + let consensus_state = AnyConsensusState::from(light_block.clone()); let client_state = get_dummy_tendermint_client_state(light_block.signed_header.header); @@ -228,6 +232,8 @@ impl MockContext { }; let consensus_states = vec![(cs_height, consensus_state)].into_iter().collect(); + debug!("consensus states: {:?}", consensus_states); + let client_record = MockClientRecord { client_type, client_state, @@ -237,6 +243,72 @@ impl MockContext { self } + pub fn with_client_parametrized_history( + mut self, + client_id: &ClientId, + client_state_height: Height, + client_type: Option, + consensus_state_height: Option, + ) -> Self { + let cs_height = consensus_state_height.unwrap_or(client_state_height); + let prev_cs_height = cs_height.clone().sub(1).unwrap_or(client_state_height); + + let client_type = client_type.unwrap_or(ClientType::Mock); + + let (client_state, consensus_state) = match client_type { + // If it's a mock client, create the corresponding mock states. + ClientType::Mock => ( + Some(MockClientState(MockHeader::new(client_state_height)).into()), + MockConsensusState::new(MockHeader::new(cs_height)).into(), + ), + // If it's a Tendermint client, we need TM states. + ClientType::Tendermint => { + let light_block = HostBlock::generate_tm_block( + self.host_chain_id.clone(), + cs_height.revision_height, + ); + + let consensus_state = AnyConsensusState::from(light_block.clone()); + let client_state = + get_dummy_tendermint_client_state(light_block.signed_header.header); + + // Return the tuple. + (Some(client_state), consensus_state) + } + }; + + let prev_consensus_state = match client_type { + // If it's a mock client, create the corresponding mock states. + ClientType::Mock => MockConsensusState::new(MockHeader::new(prev_cs_height)).into(), + // If it's a Tendermint client, we need TM states. + ClientType::Tendermint => { + let light_block = HostBlock::generate_tm_block( + self.host_chain_id.clone(), + prev_cs_height.revision_height, + ); + AnyConsensusState::from(light_block) + } + }; + + let consensus_states = vec![ + (prev_cs_height, prev_consensus_state), + (cs_height, consensus_state), + ] + .into_iter() + .collect(); + + debug!("consensus states: {:?}", consensus_states); + + let client_record = MockClientRecord { + client_type, + client_state, + consensus_states, + }; + + self.clients.insert(client_id.clone(), client_record); + self + } + /// Associates a connection to this context. pub fn with_connection( mut self, @@ -347,7 +419,7 @@ impl MockContext { /// Accessor for a block of the local (host) chain from this context. /// Returns `None` if the block at the requested height does not exist. - fn host_block(&self, target_height: Height) -> Option<&HostBlock> { + pub fn host_block(&self, target_height: Height) -> Option<&HostBlock> { let target = target_height.revision_height as usize; let latest = self.latest_height.revision_height as usize; @@ -431,6 +503,21 @@ impl MockContext { }) .collect() } + + pub fn latest_client_states(&self, client_id: &ClientId) -> &AnyClientState { + self.clients[client_id].client_state.as_ref().unwrap() + } + + pub fn latest_consensus_states( + &self, + client_id: &ClientId, + height: &Height, + ) -> &AnyConsensusState { + self.clients[client_id] + .consensus_states + .get(height) + .unwrap() + } } impl Ics26Context for MockContext {} @@ -798,6 +885,60 @@ impl ClientReader for MockContext { } } + /// Search for the lowest consensus state higher than `height`. + fn next_consensus_state( + &self, + client_id: &ClientId, + height: Height, + ) -> Result, Ics02Error> { + let client_record = self + .clients + .get(client_id) + .ok_or_else(|| Ics02Error::client_not_found(client_id.clone()))?; + + // Get the consensus state heights and sort them in ascending order. + let mut heights: Vec = client_record.consensus_states.keys().cloned().collect(); + heights.sort(); + + // Search for next state. + for h in heights { + if h > height { + // unwrap should never happen, as the consensus state for h must exist + return Ok(Some( + client_record.consensus_states.get(&h).unwrap().clone(), + )); + } + } + Ok(None) + } + + /// Search for the highest consensus state lower than `height`. + fn prev_consensus_state( + &self, + client_id: &ClientId, + height: Height, + ) -> Result, Ics02Error> { + let client_record = self + .clients + .get(client_id) + .ok_or_else(|| Ics02Error::client_not_found(client_id.clone()))?; + + // Get the consensus state heights and sort them in descending order. + let mut heights: Vec = client_record.consensus_states.keys().cloned().collect(); + heights.sort_by(|a, b| b.cmp(a)); + + // Search for previous state. + for h in heights { + if h < height { + // unwrap should never happen, as the consensus state for h must exist + return Ok(Some( + client_record.consensus_states.get(&h).unwrap().clone(), + )); + } + } + Ok(None) + } + fn client_counter(&self) -> Result { Ok(self.client_ids_counter) } diff --git a/modules/src/mock/header.rs b/modules/src/mock/header.rs index 13af52f11..3031fc486 100644 --- a/modules/src/mock/header.rs +++ b/modules/src/mock/header.rs @@ -52,9 +52,13 @@ impl MockHeader { pub fn new(height: Height) -> Self { Self { height, - timestamp: Default::default(), + timestamp: Timestamp::now(), } } + + pub fn with_timestamp(self, timestamp: Timestamp) -> Self { + Self { timestamp, ..self } + } } impl From for AnyHeader { @@ -72,6 +76,10 @@ impl Header for MockHeader { self.height } + fn timestamp(&self) -> Timestamp { + self.timestamp + } + fn wrap_any(self) -> AnyHeader { AnyHeader::Mock(self) } @@ -89,14 +97,14 @@ mod tests { #[test] fn encode_any() { - let header = MockHeader::new(Height::new(1, 10)); + let header = MockHeader::new(Height::new(1, 10)).with_timestamp(Timestamp::none()); let bytes = header.wrap_any().encode_vec().unwrap(); assert_eq!( &bytes, &[ 10, 16, 47, 105, 98, 99, 46, 109, 111, 99, 107, 46, 72, 101, 97, 100, 101, 114, 18, - 6, 10, 4, 8, 1, 16, 10, + 6, 10, 4, 8, 1, 16, 10 ] ); } diff --git a/modules/src/mock/host.rs b/modules/src/mock/host.rs index 5409c1c56..6bcba721e 100644 --- a/modules/src/mock/host.rs +++ b/modules/src/mock/host.rs @@ -1,9 +1,6 @@ //! Host chain types and methods, used by context mock. -use crate::prelude::*; -use core::convert::TryFrom; - -use tendermint::chain::Id as TMChainId; +use tendermint::time::Time; use tendermint_testgen::light_block::TmLightBlock; use tendermint_testgen::{Generator, LightBlock as TestgenLightBlock}; @@ -13,6 +10,7 @@ use crate::ics07_tendermint::consensus_state::ConsensusState as TMConsensusState use crate::ics07_tendermint::header::Header as TMHeader; use crate::ics24_host::identifier::ChainId; use crate::mock::header::MockHeader; +use crate::prelude::*; use crate::timestamp::Timestamp; use crate::Height; @@ -52,7 +50,7 @@ impl HostBlock { match chain_type { HostType::Mock => HostBlock::Mock(MockHeader { height: Height::new(chain_id.version(), height), - timestamp: Timestamp::from_nanoseconds(1).unwrap(), + timestamp: Timestamp::now(), }), HostType::SyntheticTendermint => { HostBlock::SyntheticTendermint(Box::new(Self::generate_tm_block(chain_id, height))) @@ -61,10 +59,18 @@ impl HostBlock { } pub fn generate_tm_block(chain_id: ChainId, height: u64) -> TmLightBlock { - let mut block = TestgenLightBlock::new_default(height).generate().unwrap(); - block.signed_header.header.chain_id = TMChainId::try_from(chain_id.to_string()).unwrap(); + // Sleep is required otherwise the generator produces blocks with the + // same timestamp as two block can be generated per second. + let ten_millis = core::time::Duration::from_millis(1000); + std::thread::sleep(ten_millis); + let time = Time::now() + .duration_since(Time::unix_epoch()) + .unwrap() + .as_secs(); - block + TestgenLightBlock::new_default_with_time_and_chain_id(chain_id.to_string(), time, height) + .generate() + .unwrap() } } diff --git a/modules/src/timestamp.rs b/modules/src/timestamp.rs index fe08476b0..a527200bf 100644 --- a/modules/src/timestamp.rs +++ b/modules/src/timestamp.rs @@ -9,6 +9,7 @@ use core::time::Duration; use chrono::{offset::Utc, DateTime, TimeZone}; use flex_error::{define_error, TraceError}; use serde_derive::{Deserialize, Serialize}; +use tendermint::Time; pub const ZERO_DURATION: Duration = Duration::from_secs(0); @@ -59,6 +60,13 @@ impl Timestamp { } } + /// Returns a `Timestamp` representation of the current time. + pub fn now() -> Timestamp { + Timestamp { + time: Some(Utc::now()), + } + } + /// Returns a `Timestamp` representation of a timestamp not being set. pub fn none() -> Self { Timestamp { time: None } @@ -178,6 +186,14 @@ impl FromStr for Timestamp { } } +impl From