diff --git a/.changelog/unreleased/breaking-changes/1665-tendermint-0.23.2.md b/.changelog/unreleased/breaking-changes/1665-tendermint-0.23.2.md new file mode 100644 index 0000000000..9ffadf79d6 --- /dev/null +++ b/.changelog/unreleased/breaking-changes/1665-tendermint-0.23.2.md @@ -0,0 +1,2 @@ +- Pin tendermint-rs dependencies to =0.23.2 + ([#1665](https://github.com/informalsystems/ibc-rs/pull/1665)) diff --git a/.changelog/unreleased/breaking-changes/ibc/1665-remove-chrono.md b/.changelog/unreleased/breaking-changes/ibc/1665-remove-chrono.md new file mode 100644 index 0000000000..c58260ee02 --- /dev/null +++ b/.changelog/unreleased/breaking-changes/ibc/1665-remove-chrono.md @@ -0,0 +1,4 @@ +- Remove `Timestamp` API that depended on the `chrono` crate: + ([#1665](https://github.com/informalsystems/ibc-rs/pull/1665)): + - `Timestamp::from_datetime`; use `From` + - `Timestamp::as_datetime`, superseded by `Timestamp::into_datetime` diff --git a/.changelog/unreleased/improvements/ibc/1665-remove-chrono.md b/.changelog/unreleased/improvements/ibc/1665-remove-chrono.md new file mode 100644 index 0000000000..477fa06f2d --- /dev/null +++ b/.changelog/unreleased/improvements/ibc/1665-remove-chrono.md @@ -0,0 +1,4 @@ +- More conventional ad-hoc conversion methods on `Timestamp` + ([#1665](https://github.com/informalsystems/ibc-rs/pull/1665)): + - `Timestamp::nanoseconds` replaces `Timestamp::as_nanoseconds` + - `Timestamp::into_datetime` substitutes `Timestamp::as_datetime` diff --git a/Cargo.lock b/Cargo.lock index f6d822a07c..4779cefca7 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -324,7 +324,6 @@ dependencies = [ "libc", "num-integer", "num-traits", - "serde", "winapi", ] @@ -1281,7 +1280,6 @@ name = "ibc" version = "0.9.0" dependencies = [ "bytes", - "chrono", "env_logger", "flex-error", "ibc-proto", @@ -1302,6 +1300,7 @@ dependencies = [ "tendermint-testgen", "test-log", "thiserror", + "time", "tracing", "tracing-subscriber 0.3.3", ] @@ -1357,7 +1356,6 @@ dependencies = [ "bech32", "bitcoin", "bytes", - "chrono", "crossbeam-channel 0.5.1", "dirs-next", "env_logger", @@ -2917,13 +2915,12 @@ dependencies = [ [[package]] name = "tendermint" -version = "0.23.1" +version = "0.23.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "02880339f0e89dc7de809cb300a7dbe5e066c71937a7a55472438ae02e4a841b" +checksum = "9015fdeab074f9b8f97dcb89c2bb2ec8537c89423e95551e8d7ecdfbab58a329" dependencies = [ "async-trait", "bytes", - "chrono", "ed25519", "ed25519-dalek", "flex-error", @@ -2943,14 +2940,15 @@ dependencies = [ "subtle", "subtle-encoding", "tendermint-proto", + "time", "zeroize", ] [[package]] name = "tendermint-config" -version = "0.23.1" +version = "0.23.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2aff01e0eacdcfb75e8a1bbbc2d71286f7b6db3eda82c52ad7ca4f619ebb90cc" +checksum = "a2b2e6d4442bab49319dbacdfd79c5929bc6e0b35d1e0d959ff5b79fddf3f018" dependencies = [ "flex-error", "serde", @@ -2962,11 +2960,10 @@ dependencies = [ [[package]] name = "tendermint-light-client" -version = "0.23.1" +version = "0.23.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2e7c4fe20856c514b6c37550f2f391183a5d4831820fec2e0703a19fc1523259" +checksum = "e0edd884720ab9e3be8031ca38b929b56cb80082b8d58f40984dd1e194a2f8ec" dependencies = [ - "chrono", "contracts", "crossbeam-channel 0.4.4", "derive_more", @@ -2978,17 +2975,17 @@ dependencies = [ "static_assertions", "tendermint", "tendermint-rpc", + "time", "tokio", ] [[package]] name = "tendermint-proto" -version = "0.23.1" +version = "0.23.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d1b8319fcf0860833eabe3278cf2d94ca20ef04bee6a77d892f986026fb1c252" +checksum = "da86f6e52ced9c2f24c4ae57662ce8a44dd90ee7bc47bae27a710b02d48e193c" dependencies = [ "bytes", - "chrono", "flex-error", "num-derive", "num-traits", @@ -2997,18 +2994,18 @@ dependencies = [ "serde", "serde_bytes", "subtle-encoding", + "time", ] [[package]] name = "tendermint-rpc" -version = "0.23.1" +version = "0.23.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d9c88786eeb1b408530fa7a8151aa9639cc04a6210cd8b5fa27aaafbd1e2a091" +checksum = "50f5f4875c36798e5590894a5176cf8d75e4bc81ec34ced1b9a87609e7d56861" dependencies = [ "async-trait", "async-tungstenite", "bytes", - "chrono", "flex-error", "futures", "getrandom 0.1.16", @@ -3026,6 +3023,7 @@ dependencies = [ "tendermint-config", "tendermint-proto", "thiserror", + "time", "tokio", "tracing", "url", @@ -3035,11 +3033,10 @@ dependencies = [ [[package]] name = "tendermint-testgen" -version = "0.23.1" +version = "0.23.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a6df4b2bc95b5c65ab661c8ab0e0054dd61b4e4c4edd53c2831c317e40042eac" +checksum = "019142fb30c80449e8106e91c5c3c8d846859dd7d5fe713727a0642efc819ab0" dependencies = [ - "chrono", "ed25519-dalek", "gumdrop", "serde", @@ -3047,6 +3044,7 @@ dependencies = [ "simple-error", "tempfile", "tendermint", + "time", ] [[package]] @@ -3123,8 +3121,15 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3e0a10c9a9fb3a5dce8c2239ed670f1a2569fcf42da035f5face1b19860d52b0" dependencies = [ "libc", + "time-macros", ] +[[package]] +name = "time-macros" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72c7ca5e81982d152f0788b131ebf96cc4bbd1b4575a7ed51f0f6fc866a05fdb" + [[package]] name = "tiny-bip39" version = "0.8.2" diff --git a/Cargo.toml b/Cargo.toml index d80af786e4..c1e00932ba 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -18,8 +18,15 @@ exclude = [ ] # [patch.crates-io] -# tendermint = { git = "https://github.com/informalsystems/tendermint-rs", branch = "master" } -# tendermint-rpc = { git = "https://github.com/informalsystems/tendermint-rs", branch = "master" } -# tendermint-proto = { git = "https://github.com/informalsystems/tendermint-rs", branch = "master" } -# tendermint-light-client = { git = "https://github.com/informalsystems/tendermint-rs", branch = "master" } -# tendermint-testgen = { git = "https://github.com/informalsystems/tendermint-rs", branch = "master" } +# tendermint = { git = "https://github.com/informalsystems/tendermint-rs", branch = "v0.23.x" } +# tendermint-rpc = { git = "https://github.com/informalsystems/tendermint-rs", branch = "v0.23.x" } +# tendermint-proto = { git = "https://github.com/informalsystems/tendermint-rs", branch = "v0.23.x" } +# tendermint-light-client = { git = "https://github.com/informalsystems/tendermint-rs", branch = "v0.23.x" } +# tendermint-testgen = { git = "https://github.com/informalsystems/tendermint-rs", branch = "v0.23.x" } + +# [patch.crates-io] +# tendermint = { path = "../tendermint-rs/tendermint" } +# tendermint-rpc = { path = "../tendermint-rs/rpc" } +# tendermint-proto = { path = "../tendermint-rs/proto" } +# tendermint-light-client = { path = "../tendermint-rs/light-client" } +# tendermint-testgen = { path = "../tendermint-rs/testgen" } diff --git a/ci/no-std-check/Cargo.toml b/ci/no-std-check/Cargo.toml index f10061ccef..ae2ebb560b 100644 --- a/ci/no-std-check/Cargo.toml +++ b/ci/no-std-check/Cargo.toml @@ -16,7 +16,6 @@ sp-std = { version = "3.0.0", default-features = false, optional = true } # Dependencies that support no_std bytes = { version = "1.0.1", default-features = false } -chrono = { version = "0.4.19", default-features = false } contracts = { version = "0.4.0", default-features = false } crossbeam-channel = { version = "0.5.1", default-features = false } ed25519 = { version = "1.2.0", default-features = false, features = ["serde"] } diff --git a/ci/no-std-check/src/lib.rs b/ci/no-std-check/src/lib.rs index a90084f0a1..0fe9452904 100644 --- a/ci/no-std-check/src/lib.rs +++ b/ci/no-std-check/src/lib.rs @@ -25,7 +25,6 @@ use sp_std; // Supported Imports use bytes; -use chrono; use contracts; use crossbeam_channel; use ed25519; diff --git a/modules/Cargo.toml b/modules/Cargo.toml index 4c77f03af3..9fc3ca2a1c 100644 --- a/modules/Cargo.toml +++ b/modules/Cargo.toml @@ -27,7 +27,7 @@ mocks = ["tendermint-testgen"] # Proto definitions for all IBC-related interfaces, e.g., connections or channels. ibc-proto = { version = "0.13.0", path = "../proto" } ics23 = { version = "0.6.7", default-features = false } -chrono = { version = "0.4.19", default-features = false } +time = { version = "0.3", default-features = false } thiserror = { version = "1.0.30", default-features = false } serde_derive = { version = "1.0.104", default-features = false } serde = { version = "1.0", default-features = false } @@ -42,17 +42,17 @@ sha2 = { version = "0.10.0", default-features = false } flex-error = { version = "0.4.4", default-features = false } [dependencies.tendermint] -version = "=0.23.1" +version = "=0.23.2" [dependencies.tendermint-proto] -version = "=0.23.1" +version = "=0.23.2" [dependencies.tendermint-light-client] -version = "=0.23.1" +version = "=0.23.2" default-features = false [dependencies.tendermint-testgen] -version = "=0.23.1" +version = "=0.23.2" optional = true [dev-dependencies] @@ -61,8 +61,8 @@ tracing-subscriber = { version = "0.3.3", features = ["fmt", "env-filter", "json test-log = { version = "0.2.8", features = ["trace"] } modelator = "0.4.1" sha2 = { version = "0.10.0" } -tendermint-rpc = { version = "=0.23.1", features = ["http-client", "websocket-client"] } -tendermint-testgen = { version = "=0.23.1" } # Needed for generating (synthetic) light blocks. +tendermint-rpc = { version = "=0.23.2", features = ["http-client", "websocket-client"] } +tendermint-testgen = { version = "=0.23.2" } # Needed for generating (synthetic) light blocks. [[test]] name = "mbt" diff --git a/modules/src/applications/ics20_fungible_token_transfer/msgs/transfer.rs b/modules/src/applications/ics20_fungible_token_transfer/msgs/transfer.rs index 2522376f95..88be8fb54b 100644 --- a/modules/src/applications/ics20_fungible_token_transfer/msgs/transfer.rs +++ b/modules/src/applications/ics20_fungible_token_transfer/msgs/transfer.rs @@ -92,7 +92,7 @@ impl From for RawMsgTransfer { sender: domain_msg.sender.to_string(), receiver: domain_msg.receiver.to_string(), timeout_height: Some(domain_msg.timeout_height.into()), - timeout_timestamp: domain_msg.timeout_timestamp.as_nanoseconds(), + timeout_timestamp: domain_msg.timeout_timestamp.nanoseconds(), } } } diff --git a/modules/src/clients/ics07_tendermint/client_def.rs b/modules/src/clients/ics07_tendermint/client_def.rs index 351ffffa2a..f9b2199514 100644 --- a/modules/src/clients/ics07_tendermint/client_def.rs +++ b/modules/src/clients/ics07_tendermint/client_def.rs @@ -1,9 +1,9 @@ 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 time::OffsetDateTime; use crate::clients::ics07_tendermint::client_state::ClientState; use crate::clients::ics07_tendermint::consensus_state::ConsensusState; @@ -107,7 +107,7 @@ impl ClientDef for TendermintClient { untrusted_state, trusted_state, &options, - Time(chrono::Utc::now()), + OffsetDateTime::now_utc().try_into().unwrap(), ); match verdict { diff --git a/modules/src/clients/ics07_tendermint/consensus_state.rs b/modules/src/clients/ics07_tendermint/consensus_state.rs index d439264bc0..141df337a6 100644 --- a/modules/src/clients/ics07_tendermint/consensus_state.rs +++ b/modules/src/clients/ics07_tendermint/consensus_state.rs @@ -2,9 +2,9 @@ use crate::prelude::*; use core::convert::Infallible; -use chrono::{TimeZone, Utc}; use serde::Serialize; use tendermint::{hash::Algorithm, time::Time, Hash}; +use tendermint_proto::google::protobuf as tpb; use tendermint_proto::Protobuf; use ibc_proto::ibc::lightclients::tendermint::v1::ConsensusState as RawConsensusState; @@ -58,9 +58,15 @@ impl TryFrom for ConsensusState { type Error = Error; fn try_from(raw: RawConsensusState) -> Result { - let proto_timestamp = raw + let prost_types::Timestamp { seconds, nanos } = raw .timestamp .ok_or_else(|| Error::invalid_raw_consensus_state("missing timestamp".into()))?; + // FIXME: shunts like this are necessary due to + // https://github.com/informalsystems/tendermint-rs/issues/1053 + let proto_timestamp = tpb::Timestamp { seconds, nanos }; + let timestamp = proto_timestamp + .try_into() + .map_err(|e| Error::invalid_raw_consensus_state(format!("invalid timestamp: {}", e)))?; Ok(Self { root: raw @@ -70,9 +76,7 @@ impl TryFrom for ConsensusState { })? .hash .into(), - timestamp: Utc - .timestamp(proto_timestamp.seconds, proto_timestamp.nanos as u32) - .into(), + timestamp, next_validators_hash: Hash::from_bytes(Algorithm::Sha256, &raw.next_validators_hash) .map_err(|e| Error::invalid_raw_consensus_state(e.to_string()))?, }) @@ -81,10 +85,10 @@ impl TryFrom for ConsensusState { impl From for RawConsensusState { fn from(value: ConsensusState) -> Self { - let timestamp = prost_types::Timestamp { - seconds: value.timestamp.0.timestamp(), - nanos: value.timestamp.0.timestamp_subsec_nanos() as i32, - }; + // FIXME: shunts like this are necessary due to + // https://github.com/informalsystems/tendermint-rs/issues/1053 + let tpb::Timestamp { seconds, nanos } = value.timestamp.into(); + let timestamp = prost_types::Timestamp { seconds, nanos }; RawConsensusState { timestamp: Some(timestamp), diff --git a/modules/src/clients/ics07_tendermint/header.rs b/modules/src/clients/ics07_tendermint/header.rs index a45734022d..8fe32105a5 100644 --- a/modules/src/clients/ics07_tendermint/header.rs +++ b/modules/src/clients/ics07_tendermint/header.rs @@ -1,7 +1,6 @@ use core::cmp::Ordering; use bytes::Buf; -use chrono::DateTime; use prost::Message; use serde_derive::{Deserialize, Serialize}; use tendermint::block::signed_header::SignedHeader; @@ -78,7 +77,7 @@ impl crate::core::ics02_client::header::Header for Header { } fn timestamp(&self) -> Timestamp { - Timestamp::from_datetime(DateTime::from(self.signed_header.header.time)) + self.signed_header.header.time.into() } fn wrap_any(self) -> AnyHeader { diff --git a/modules/src/core/ics02_client/client_consensus.rs b/modules/src/core/ics02_client/client_consensus.rs index 54c7192db6..de1e54fb80 100644 --- a/modules/src/core/ics02_client/client_consensus.rs +++ b/modules/src/core/ics02_client/client_consensus.rs @@ -3,8 +3,6 @@ use crate::prelude::*; use core::convert::Infallible; use core::marker::{Send, Sync}; -use chrono::{DateTime, Utc}; - use ibc_proto::ibc::core::client::v1::ConsensusStateWithHeight; use prost_types::Any; use serde::Serialize; @@ -55,10 +53,7 @@ pub enum AnyConsensusState { impl AnyConsensusState { pub fn timestamp(&self) -> Timestamp { match self { - Self::Tendermint(cs_state) => { - let date: DateTime = cs_state.timestamp.into(); - Timestamp::from_datetime(date) - } + Self::Tendermint(cs_state) => cs_state.timestamp.into(), #[cfg(any(test, feature = "mocks"))] Self::Mock(mock_state) => mock_state.timestamp(), diff --git a/modules/src/core/ics04_channel/events.rs b/modules/src/core/ics04_channel/events.rs index 972794c8a8..11efb275b6 100644 --- a/modules/src/core/ics04_channel/events.rs +++ b/modules/src/core/ics04_channel/events.rs @@ -294,7 +294,7 @@ impl TryFrom for Vec { key: PKT_TIMEOUT_TIMESTAMP_ATTRIBUTE_KEY.parse().unwrap(), value: p .timeout_timestamp - .as_nanoseconds() + .nanoseconds() .to_string() .parse() .unwrap(), @@ -960,7 +960,7 @@ mod tests { destination_channel: "b_test_channel".parse().unwrap(), data: "test_data".as_bytes().to_vec(), timeout_height: Height::new(1, 10), - timeout_timestamp: Timestamp::from_datetime(chrono::offset::Utc::now()), + timeout_timestamp: Timestamp::now(), }; let mut abci_events = vec![]; let send_packet = SendPacket { diff --git a/modules/src/core/ics04_channel/handler/send_packet.rs b/modules/src/core/ics04_channel/handler/send_packet.rs index 77d6279bf4..28793ff7da 100644 --- a/modules/src/core/ics04_channel/handler/send_packet.rs +++ b/modules/src/core/ics04_channel/handler/send_packet.rs @@ -146,7 +146,7 @@ mod tests { 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()) + let mut packet: Packet = get_dummy_raw_packet(1, timestamp.unwrap().nanoseconds()) .try_into() .unwrap(); packet.sequence = 1.into(); diff --git a/modules/src/core/ics04_channel/packet.rs b/modules/src/core/ics04_channel/packet.rs index 12ec013549..6b4cdd1093 100644 --- a/modules/src/core/ics04_channel/packet.rs +++ b/modules/src/core/ics04_channel/packet.rs @@ -238,7 +238,7 @@ impl From for RawPacket { destination_channel: packet.destination_channel.to_string(), data: packet.data, timeout_height: Some(packet.timeout_height.into()), - timeout_timestamp: packet.timeout_timestamp.as_nanoseconds(), + timeout_timestamp: packet.timeout_timestamp.nanoseconds(), } } } diff --git a/modules/src/mock/client_state.rs b/modules/src/mock/client_state.rs index a218255942..d0779a0b77 100644 --- a/modules/src/mock/client_state.rs +++ b/modules/src/mock/client_state.rs @@ -85,7 +85,7 @@ impl From for RawMockClientState { RawMockClientState { header: Some(ibc_proto::ibc::mock::Header { height: Some(value.header.height().into()), - timestamp: value.header.timestamp.as_nanoseconds(), + timestamp: value.header.timestamp.nanoseconds(), }), } } @@ -158,7 +158,7 @@ impl From for RawMockConsensusState { RawMockConsensusState { header: Some(ibc_proto::ibc::mock::Header { height: Some(value.header.height().into()), - timestamp: value.header.timestamp.as_nanoseconds(), + timestamp: value.header.timestamp.nanoseconds(), }), } } diff --git a/modules/src/mock/header.rs b/modules/src/mock/header.rs index 1678a62407..381975ae48 100644 --- a/modules/src/mock/header.rs +++ b/modules/src/mock/header.rs @@ -37,7 +37,7 @@ impl From for RawMockHeader { fn from(value: MockHeader) -> Self { RawMockHeader { height: Some(value.height.into()), - timestamp: value.timestamp.as_nanoseconds(), + timestamp: value.timestamp.nanoseconds(), } } } diff --git a/modules/src/mock/host.rs b/modules/src/mock/host.rs index ec65ad7171..71cdd6d39b 100644 --- a/modules/src/mock/host.rs +++ b/modules/src/mock/host.rs @@ -3,6 +3,7 @@ use tendermint::time::Time; use tendermint_testgen::light_block::TmLightBlock; use tendermint_testgen::{Generator, LightBlock as TestgenLightBlock}; +use time::OffsetDateTime; use crate::clients::ics07_tendermint::consensus_state::ConsensusState as TMConsensusState; use crate::clients::ics07_tendermint::header::Header as TMHeader; @@ -63,10 +64,8 @@ impl HostBlock { // 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(chrono::Utc::now()) - .duration_since(Time::unix_epoch()) - .unwrap() - .as_secs(); + let time: Time = OffsetDateTime::now_utc().try_into().unwrap(); + let time = time.duration_since(Time::unix_epoch()).unwrap().as_secs(); TestgenLightBlock::new_default_with_time_and_chain_id(chain_id.to_string(), time, height) .generate() diff --git a/modules/src/timestamp.rs b/modules/src/timestamp.rs index 500ea37377..1dbb29e719 100644 --- a/modules/src/timestamp.rs +++ b/modules/src/timestamp.rs @@ -1,28 +1,39 @@ use crate::prelude::*; use core::fmt::Display; +use core::hash::{Hash, Hasher}; use core::num::ParseIntError; use core::ops::{Add, Sub}; use core::str::FromStr; use core::time::Duration; -use chrono::{offset::Utc, DateTime, TimeZone}; use flex_error::{define_error, TraceError}; use serde_derive::{Deserialize, Serialize}; use tendermint::Time; +use time::OffsetDateTime; pub const ZERO_DURATION: Duration = Duration::from_secs(0); -/// A newtype wrapper over `Option>` to keep track of +/// A newtype wrapper over `Option