use std::{
future::Future,
pin::Pin,
task::{Context, Poll},
vec,
};
use crate::PenumbraHost;
use anyhow::Context as _;
use cnidarium::Storage;
use futures::FutureExt;
use ibc_proto::ibc::core::{
channel::v1::{
PacketState, QueryChannelsRequest, QueryChannelsResponse,
QueryPacketAcknowledgementsRequest, QueryPacketAcknowledgementsResponse,
QueryPacketCommitmentsRequest, QueryPacketCommitmentsResponse, QueryUnreceivedAcksRequest,
QueryUnreceivedAcksResponse, QueryUnreceivedPacketsRequest, QueryUnreceivedPacketsResponse,
},
client::v1::{IdentifiedClientState, QueryClientStatesRequest, QueryClientStatesResponse},
};
use ibc_proto::ibc::core::{
channel::v1::{QueryConnectionChannelsRequest, QueryConnectionChannelsResponse},
connection::v1::{QueryConnectionsRequest, QueryConnectionsResponse},
};
use ibc_types::core::channel::IdentifiedChannelEnd;
use ibc_types::core::channel::{ChannelId, PortId};
use ibc_types::core::client::ClientId;
use ibc_types::core::connection::ConnectionId;
use ibc_types::core::connection::IdentifiedConnectionEnd;
use penumbra_sdk_ibc::component::ChannelStateReadExt as _;
use penumbra_sdk_ibc::component::ClientStateReadExt as _;
use penumbra_sdk_ibc::component::ConnectionStateReadExt as _;
use penumbra_sdk_ibc::component::HostInterface;
use prost::Message;
use std::str::FromStr;
use tendermint::v0_37::abci::{
request,
response::{self, Echo},
InfoRequest, InfoResponse,
};
use tower_abci::BoxError;
use tracing::Instrument;
use penumbra_sdk_tower_trace::v037::RequestExt;
const ABCI_INFO_VERSION: &str = env!("CARGO_PKG_VERSION");
#[derive(Clone, Debug)]
pub struct Info {
storage: Storage,
}
impl Info {
pub fn new(storage: Storage) -> Self {
Self { storage }
}
async fn info(&self, info: request::Info) -> anyhow::Result<response::Info> {
let state = self.storage.latest_snapshot();
let last_block_height = PenumbraHost::get_block_height(&state)
.await
.unwrap_or_default()
.try_into()?;
let app_version = crate::APP_VERSION;
tracing::info!(?info, state_version = ?state.version(), last_block_height = ?last_block_height, ?app_version,"reporting height in info query");
let last_block_app_hash = state.root_hash().await?.0.to_vec().try_into()?;
Ok(response::Info {
data: "penumbra".to_string(),
version: ABCI_INFO_VERSION.to_string(),
app_version,
last_block_height,
last_block_app_hash,
})
}
async fn get_snapshot_for_height(
&self,
height: u64,
) -> anyhow::Result<(cnidarium::Snapshot, u64)> {
match height {
0 => {
let height = self.storage.latest_snapshot().version().saturating_sub(1);
let snapshot = self
.storage
.snapshot(height)
.ok_or_else(|| anyhow::anyhow!("no snapshot of height {height}"))?;
Ok((snapshot, height))
}
height => {
let snapshot = self
.storage
.snapshot(height)
.ok_or_else(|| anyhow::anyhow!("no snapshot of height {height}"))?;
Ok((snapshot, height))
}
}
}
async fn query(&self, query: request::Query) -> anyhow::Result<response::Query> {
tracing::debug!("got query");
match query.path.as_str() {
"state/key" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let key = hex::decode(&query.data).unwrap_or_else(|_| query.data.to_vec());
let (value, proof) =
snapshot
.get_with_proof(key.clone())
.await
.with_context(|| {
format!("failed to get key {}", String::from_utf8_lossy(&key))
})?;
let mut ops = vec![];
for commitment_proof in proof.proofs {
match commitment_proof
.clone()
.proof
.expect("should have non empty commitment proofs")
{
ics23::commitment_proof::Proof::Exist(x_proof) => {
let proof_op = tendermint::merkle::proof::ProofOp {
field_type: "jmt:v".to_string(),
key: x_proof.key,
data: commitment_proof.encode_to_vec(),
};
ops.push(proof_op);
}
ics23::commitment_proof::Proof::Nonexist(nx_proof) => {
let proof_op = tendermint::merkle::proof::ProofOp {
field_type: "jmt:v".to_string(),
key: nx_proof.key,
data: commitment_proof.encode_to_vec(),
};
ops.push(proof_op);
}
ics23::commitment_proof::Proof::Batch(_) => {
anyhow::bail!("batch proofs not supported in abci query")
}
ics23::commitment_proof::Proof::Compressed(_) => {
anyhow::bail!("compressed proofs not supported in abci query")
}
}
}
let proof_ops = tendermint::merkle::proof::ProofOps { ops };
let value = value.unwrap_or_else(Vec::new);
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: value.into(),
proof: Some(proof_ops),
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.connection.v1.Query/Connections" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let _request = QueryConnectionsRequest::decode(query.data.clone())
.context("failed to decode QueryConnectionsRequest")?;
let connection_counter = snapshot.get_connection_counter().await?;
let mut connections = vec![];
for conn_idx in 0..connection_counter.0 {
let conn_id = ConnectionId(format!("connection-{}", conn_idx));
let connection = snapshot
.get_connection(&conn_id)
.await?
.context("couldn't find connection")?;
let id_conn = IdentifiedConnectionEnd {
connection_id: conn_id,
connection_end: connection,
};
connections.push(id_conn.into());
}
let res_value = QueryConnectionsResponse {
connections,
pagination: None,
height: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.channel.v1.Query/Channels" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let _request = QueryChannelsRequest::decode(query.data.clone())
.context("failed to decode QueryConnectionsRequest")?;
let channel_counter = snapshot.get_channel_counter().await?;
let mut channels = vec![];
for chan_idx in 0..channel_counter {
let chan_id = ChannelId(format!("channel-{}", chan_idx));
let channel = snapshot
.get_channel(&chan_id, &PortId::transfer())
.await?
.context("couldn't find channel")?;
let id_chan = IdentifiedChannelEnd {
channel_id: chan_id,
port_id: PortId::transfer(),
channel_end: channel,
upgrade_sequence: 0,
};
channels.push(id_chan.into());
}
let res_value = QueryChannelsResponse {
channels,
pagination: None,
height: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.channel.v1.Query/ConnectionChannels" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let request = QueryConnectionChannelsRequest::decode(query.data.clone())
.context("failed to decode QueryConnectionChannelsRequest")?;
let connection_id: ConnectionId = ConnectionId::from_str(&request.connection)
.context("couldn't decode connection id from request")?;
let channel_counter = snapshot.get_channel_counter().await?;
let mut channels = vec![];
for chan_idx in 0..channel_counter {
let chan_id = ChannelId(format!("channel-{}", chan_idx));
let channel = snapshot
.get_channel(&chan_id, &PortId::transfer())
.await?
.context("couldn't find channel")?;
if channel.connection_hops.contains(&connection_id) {
let id_chan = IdentifiedChannelEnd {
channel_id: chan_id,
port_id: PortId::transfer(),
channel_end: channel,
upgrade_sequence: 0,
};
channels.push(id_chan.into());
}
}
let res_value = QueryConnectionChannelsResponse {
channels,
pagination: None,
height: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.client.v1.Query/ClientStates" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let _request = QueryClientStatesRequest::decode(query.data.clone())
.context("failed to decode QueryClientStatesRequest")?;
let client_counter = snapshot.client_counter().await?.0;
let mut client_states = vec![];
for client_idx in 0..client_counter {
let client_id =
ClientId::from_str(format!("07-tendermint-{}", client_idx).as_str())?;
let client_state = snapshot.get_client_state(&client_id).await;
let id_client = IdentifiedClientState {
client_id: client_id.to_string(),
client_state: client_state.ok().map(|state| state.into()), };
client_states.push(id_client);
}
let res_value = QueryClientStatesResponse {
client_states,
pagination: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.channel.v1.Query/PacketCommitments" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let request = QueryPacketCommitmentsRequest::decode(query.data.clone())
.context("failed to decode QueryPacketCommitmentsRequest")?;
let chan_id: ChannelId =
ChannelId::from_str(&request.channel_id).context("invalid channel id")?;
let port_id: PortId =
PortId::from_str(&request.port_id).context("invalid port id")?;
let mut commitment_states = vec![];
let commitment_counter = snapshot.get_send_sequence(&chan_id, &port_id).await?;
for commitment_idx in 1..commitment_counter {
let commitment = snapshot
.get_packet_commitment_by_id(&chan_id, &port_id, commitment_idx)
.await?;
if commitment.is_none() {
continue;
}
let commitment = commitment.expect("commitment is Some");
let commitment_state = PacketState {
port_id: request.port_id.clone(),
channel_id: request.channel_id.clone(),
sequence: commitment_idx,
data: commitment.clone(),
};
commitment_states.push(commitment_state);
}
let res_value = QueryPacketCommitmentsResponse {
commitments: commitment_states,
pagination: None,
height: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.channel.v1.Query/PacketAcknowledgements" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let request = QueryPacketAcknowledgementsRequest::decode(query.data.clone())
.context("failed to decode QueryPacketAcknowledgementsRequest")?;
let chan_id: ChannelId =
ChannelId::from_str(&request.channel_id).context("invalid channel id")?;
let port_id: PortId =
PortId::from_str(&request.port_id).context("invalid port id")?;
let ack_counter = snapshot.get_ack_sequence(&chan_id, &port_id).await?;
let mut acks = vec![];
for ack_idx in 0..ack_counter {
let ack = snapshot
.get_packet_acknowledgement(&port_id, &chan_id, ack_idx)
.await?
.ok_or_else(|| anyhow::anyhow!("couldn't find ack"))?;
let ack_state = PacketState {
port_id: request.port_id.clone(),
channel_id: request.channel_id.clone(),
sequence: ack_idx,
data: ack.clone(),
};
acks.push(ack_state);
}
let res_value = QueryPacketAcknowledgementsResponse {
acknowledgements: acks,
pagination: None,
height: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.channel.v1.Query/UnreceivedPackets" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let request = QueryUnreceivedPacketsRequest::decode(query.data.clone())
.context("failed to decode QueryUnreceivedPacketsRequest")?;
let chan_id: ChannelId =
ChannelId::from_str(&request.channel_id).context("invalid channel id")?;
let port_id: PortId =
PortId::from_str(&request.port_id).context("invalid port id")?;
let mut unreceived_seqs = vec![];
for seq in request.packet_commitment_sequences {
if seq == 0 {
anyhow::bail!("packet sequence {} cannot be 0", seq);
}
if !snapshot
.seen_packet_by_channel(&chan_id, &port_id, seq)
.await?
{
unreceived_seqs.push(seq);
}
}
let res_value = QueryUnreceivedPacketsResponse {
sequences: unreceived_seqs,
height: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
"/ibc.core.channel.v1.Query/UnreceivedAcks" => {
let (snapshot, height) = self
.get_snapshot_for_height(u64::from(query.height))
.await?;
let request = QueryUnreceivedAcksRequest::decode(query.data.clone())
.context("failed to decode QueryUnreceivedAcksRequest")?;
let chan_id: ChannelId =
ChannelId::from_str(&request.channel_id).context("invalid channel id")?;
let port_id: PortId =
PortId::from_str(&request.port_id).context("invalid port id")?;
let mut unreceived_seqs = vec![];
for seq in request.packet_ack_sequences {
if seq == 0 {
anyhow::bail!("packet sequence {} cannot be 0", seq);
}
if snapshot
.get_packet_commitment_by_id(&chan_id, &port_id, seq)
.await?
.is_some()
{
unreceived_seqs.push(seq);
}
}
let res_value = QueryUnreceivedAcksResponse {
sequences: unreceived_seqs,
height: None,
}
.encode_to_vec();
Ok(response::Query {
code: 0.into(),
key: query.data,
log: "".to_string(),
value: res_value.into(),
proof: None,
height: height.try_into().context("failed to convert height")?,
codespace: "".to_string(),
info: "".to_string(),
index: 0,
})
}
_ => Err(anyhow::anyhow!(
"requested unrecognized path in ABCI query: {}",
query.path
)),
}
}
}
impl tower_service::Service<InfoRequest> for Info {
type Response = InfoResponse;
type Error = BoxError;
type Future = Pin<Box<dyn Future<Output = Result<InfoResponse, BoxError>> + Send + 'static>>;
fn poll_ready(&mut self, _cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
Poll::Ready(Ok(()))
}
fn call(&mut self, req: InfoRequest) -> Self::Future {
let span = req.create_span();
let self2 = self.clone();
async move {
match req {
InfoRequest::Info(info) => self2
.info(info)
.await
.map(InfoResponse::Info)
.map_err(Into::into),
InfoRequest::Query(query) => match self2.query(query).await {
Ok(rsp) => {
tracing::debug!(value = ?rsp.value);
Ok(InfoResponse::Query(rsp))
}
Err(e) => {
tracing::debug!(error = ?e);
Ok(InfoResponse::Query(response::Query {
code: 1.into(),
log: format!("{:#}", e),
..Default::default()
}))
}
},
InfoRequest::Echo(echo) => Ok(InfoResponse::Echo(Echo {
message: echo.message,
})),
}
}
.instrument(span)
.boxed()
}
}