From 17e6db0b76a7c0820dc7936b59e098caff00b9b1 Mon Sep 17 00:00:00 2001 From: Hubert Bugaj Date: Tue, 29 Oct 2024 22:55:14 +0100 Subject: [PATCH] feat: extend `forest-cli chain head` --- CHANGELOG.md | 5 +++++ scripts/tests/calibnet_other_check.sh | 4 ++++ src/cli/subcommands/chain_cmd.rs | 23 ++++++++++++++++++++--- 3 files changed, 29 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index cfc80be53f1d..e9fbd8d26823 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -27,6 +27,11 @@ ### Breaking +- [#4952](https://github.com/ChainSafe/forest/pull/4952) Extended the + `forest-cli chain head` command to allow for specifying number of last tipsets + to display. This change is breaking as the output now contains the epoch of + tipsets. + ### Added - [#4937](https://github.com/ChainSafe/forest/pull/4937) Added diff --git a/scripts/tests/calibnet_other_check.sh b/scripts/tests/calibnet_other_check.sh index 10c552f2bb27..c692bf3f7f49 100755 --- a/scripts/tests/calibnet_other_check.sh +++ b/scripts/tests/calibnet_other_check.sh @@ -29,6 +29,10 @@ echo "Test dev commands (which could brick the node/cause subsequent snapshots t echo "Test subcommand: chain set-head" $FOREST_CLI_PATH chain set-head --epoch -10 --force +echo "Test subcommand: chain head" +$FOREST_CLI_PATH chain head +$FOREST_CLI_PATH chain head --tipsets 10 + echo "Test subcommand: info show" $FOREST_CLI_PATH info show diff --git a/src/cli/subcommands/chain_cmd.rs b/src/cli/subcommands/chain_cmd.rs index fd8b7501802e..3091be1ab11e 100644 --- a/src/cli/subcommands/chain_cmd.rs +++ b/src/cli/subcommands/chain_cmd.rs @@ -5,7 +5,7 @@ use crate::blocks::{Tipset, TipsetKey}; use crate::lotus_json::HasLotusJson; use crate::message::ChainMessage; use crate::rpc::{self, prelude::*}; -use anyhow::bail; +use anyhow::{bail, ensure}; use cid::Cid; use clap::Subcommand; use nunny::Vec as NonEmpty; @@ -24,7 +24,12 @@ pub enum ChainCommands { Genesis, /// Prints out the canonical head of the chain - Head, + Head { + /// Print the first `n` tipsets from the head (inclusive). + /// Tipsets are categorized by epoch in descending order. + #[arg(short = 'n', long, default_value = "1")] + tipsets: i64, + }, /// Reads and prints out a message referenced by the specified CID from the /// chain block store @@ -63,7 +68,7 @@ impl ChainCommands { print_pretty_lotus_json(ChainGetBlock::call(&client, (cid,)).await?) } Self::Genesis => print_pretty_lotus_json(ChainGetGenesis::call(&client, ()).await?), - Self::Head => print_rpc_res_cids(ChainHead::call(&client, ()).await?), + Self::Head { tipsets } => print_chain_head(&client, tipsets).await, Self::Message { cid } => { let bytes = ChainReadObj::call(&client, (cid,)).await?; match fvm_ipld_encoding::from_slice::(&bytes)? { @@ -145,3 +150,15 @@ fn maybe_confirm(no_confirm: bool, prompt: impl Into) -> anyhow::Result< false => bail!("Operation cancelled by user"), } } + +/// Print the first `n` tipsets from the head (inclusive). +async fn print_chain_head(client: &rpc::Client, n: i64) -> anyhow::Result<()> { + ensure!(n > 0, "number of tipsets must be positive"); + let current_epoch = ChainHead::call(client, ()).await?.epoch(); + for epoch in (current_epoch.saturating_sub(n)..current_epoch).rev() { + let tipset = tipset_by_epoch_or_offset(client, epoch).await?; + println!("[{}]", epoch); + print_rpc_res_cids(tipset)?; + } + Ok(()) +}