use anyhow::Context;
use bytes::Bytes;
use ibc_types::core::client::ClientId;
use serde::{Deserialize, Serialize};
use std::str::FromStr;
use crate::change::ParameterChange;
use penumbra_sdk_proto::{penumbra::core::component::governance::v1 as pb, DomainType};
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
#[serde(try_from = "pb::Proposal", into = "pb::Proposal")]
pub struct Proposal {
pub id: u64,
pub title: String,
pub description: String,
pub payload: ProposalPayload,
}
pub const TRANSACTION_PLAN_TYPE_URL: &str = "/penumbra.core.transaction.v1.TransactionPlan";
impl From<Proposal> for pb::Proposal {
fn from(inner: Proposal) -> pb::Proposal {
let mut proposal = pb::Proposal {
id: inner.id,
title: inner.title,
description: inner.description,
..Default::default() };
use pb::proposal::Payload;
let payload = match inner.payload {
ProposalPayload::Signaling { commit } => {
Some(Payload::Signaling(pb::proposal::Signaling {
commit: if let Some(c) = commit {
c
} else {
String::default()
},
}))
}
ProposalPayload::Emergency { halt_chain } => {
Some(Payload::Emergency(pb::proposal::Emergency { halt_chain }))
}
ProposalPayload::ParameterChange(change) => {
Some(Payload::ParameterChange(change.into()))
}
ProposalPayload::CommunityPoolSpend { transaction_plan } => Some(
Payload::CommunityPoolSpend(pb::proposal::CommunityPoolSpend {
transaction_plan: Some(pbjson_types::Any {
type_url: TRANSACTION_PLAN_TYPE_URL.to_owned(),
value: transaction_plan.into(),
}),
}),
),
ProposalPayload::UpgradePlan { height } => {
Some(Payload::UpgradePlan(pb::proposal::UpgradePlan { height }))
}
ProposalPayload::FreezeIbcClient { client_id } => {
Some(Payload::FreezeIbcClient(pb::proposal::FreezeIbcClient {
client_id: client_id.into(),
}))
}
ProposalPayload::UnfreezeIbcClient { client_id } => Some(Payload::UnfreezeIbcClient(
pb::proposal::UnfreezeIbcClient {
client_id: client_id.into(),
},
)),
};
proposal.payload = payload;
proposal
}
}
impl TryFrom<pb::Proposal> for Proposal {
type Error = anyhow::Error;
fn try_from(inner: pb::Proposal) -> Result<Proposal, Self::Error> {
if inner.title.len() > 80 {
anyhow::bail!("proposal title field must be less than 80 characters");
}
if inner.description.len() > 10_000 {
anyhow::bail!("proposal description must be less than 10,000 characters");
}
use pb::proposal::Payload;
Ok(Proposal {
id: inner.id,
title: inner.title,
description: inner.description,
payload: match inner
.payload
.ok_or_else(|| anyhow::anyhow!("missing proposal payload"))?
{
Payload::Signaling(signaling) => ProposalPayload::Signaling {
commit: if signaling.commit.is_empty() {
None
} else {
if signaling.commit.len() > 255 {
anyhow::bail!("proposal commit hash must be less than 255 bytes");
}
Some(signaling.commit)
},
},
Payload::Emergency(emergency) => ProposalPayload::Emergency {
halt_chain: emergency.halt_chain,
},
Payload::ParameterChange(change) => {
ProposalPayload::ParameterChange(change.try_into()?)
}
Payload::CommunityPoolSpend(community_pool_spend) => {
ProposalPayload::CommunityPoolSpend {
transaction_plan: {
let transaction_plan = community_pool_spend
.transaction_plan
.ok_or_else(|| anyhow::anyhow!("missing transaction plan"))?;
if transaction_plan.type_url != TRANSACTION_PLAN_TYPE_URL {
anyhow::bail!(
"unknown transaction plan type url: {}",
transaction_plan.type_url
);
}
transaction_plan.value.to_vec()
},
}
}
Payload::UpgradePlan(upgrade_plan) => ProposalPayload::UpgradePlan {
height: upgrade_plan.height,
},
Payload::FreezeIbcClient(freeze_ibc_client) => {
if freeze_ibc_client.client_id.len() > 128 {
anyhow::bail!("client ID must be less than 128 bytes");
}
ClientId::from_str(&freeze_ibc_client.client_id)
.map_err(|e| anyhow::anyhow!("invalid client id: {e}"))?;
ProposalPayload::FreezeIbcClient {
client_id: freeze_ibc_client.client_id,
}
}
Payload::UnfreezeIbcClient(unfreeze_ibc_client) => {
if unfreeze_ibc_client.client_id.len() > 128 {
anyhow::bail!("client ID must be less than 128 bytes");
}
ClientId::from_str(&unfreeze_ibc_client.client_id)
.map_err(|e| anyhow::anyhow!("invalid client id: {e}"))?;
ProposalPayload::UnfreezeIbcClient {
client_id: unfreeze_ibc_client.client_id,
}
}
},
})
}
}
impl DomainType for Proposal {
type Proto = pb::Proposal;
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ProposalToml {
pub id: u64,
pub title: String,
pub description: String,
#[serde(flatten)]
pub payload: ProposalPayloadToml,
}
impl From<Proposal> for ProposalToml {
fn from(proposal: Proposal) -> ProposalToml {
ProposalToml {
id: proposal.id,
title: proposal.title,
description: proposal.description,
payload: proposal.payload.into(),
}
}
}
impl TryFrom<ProposalToml> for Proposal {
type Error = anyhow::Error;
fn try_from(proposal: ProposalToml) -> Result<Proposal, Self::Error> {
Ok(Proposal {
id: proposal.id,
title: proposal.title,
description: proposal.description,
payload: proposal.payload.try_into()?,
})
}
}
#[derive(Debug, Clone, Eq, PartialEq, Serialize, Deserialize)]
#[serde(try_from = "pb::ProposalKind", into = "pb::ProposalKind")]
pub enum ProposalKind {
Signaling,
Emergency,
ParameterChange,
CommunityPoolSpend,
UpgradePlan,
FreezeIbcClient,
UnfreezeIbcClient,
}
impl From<ProposalKind> for pb::ProposalKind {
fn from(kind: ProposalKind) -> pb::ProposalKind {
match kind {
ProposalKind::Signaling => pb::ProposalKind::Signaling,
ProposalKind::Emergency => pb::ProposalKind::Emergency,
ProposalKind::ParameterChange => pb::ProposalKind::ParameterChange,
ProposalKind::CommunityPoolSpend => pb::ProposalKind::CommunityPoolSpend,
ProposalKind::UpgradePlan => pb::ProposalKind::UpgradePlan,
ProposalKind::FreezeIbcClient => pb::ProposalKind::FreezeIbcClient,
ProposalKind::UnfreezeIbcClient => pb::ProposalKind::UnfreezeIbcClient,
}
}
}
impl TryFrom<pb::ProposalKind> for ProposalKind {
type Error = anyhow::Error;
fn try_from(kind: pb::ProposalKind) -> anyhow::Result<ProposalKind> {
let kind = match kind {
pb::ProposalKind::Unspecified => anyhow::bail!("unspecified proposal kind"),
pb::ProposalKind::Signaling => ProposalKind::Signaling,
pb::ProposalKind::Emergency => ProposalKind::Emergency,
pb::ProposalKind::ParameterChange => ProposalKind::ParameterChange,
pb::ProposalKind::CommunityPoolSpend => ProposalKind::CommunityPoolSpend,
pb::ProposalKind::UpgradePlan => ProposalKind::UpgradePlan,
pb::ProposalKind::FreezeIbcClient => ProposalKind::FreezeIbcClient,
pb::ProposalKind::UnfreezeIbcClient => ProposalKind::UnfreezeIbcClient,
};
Ok(kind)
}
}
impl FromStr for ProposalKind {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"signaling" => Ok(ProposalKind::Signaling),
"emergency" => Ok(ProposalKind::Emergency),
"parameter_change" => Ok(ProposalKind::ParameterChange),
"community_pool_spend" => Ok(ProposalKind::CommunityPoolSpend),
"upgrade_plan" => Ok(ProposalKind::UpgradePlan),
_ => Err(anyhow::anyhow!("invalid proposal kind: {}", s)),
}
}
}
impl Proposal {
pub fn kind(&self) -> ProposalKind {
match self.payload {
ProposalPayload::Signaling { .. } => ProposalKind::Signaling,
ProposalPayload::Emergency { .. } => ProposalKind::Emergency,
ProposalPayload::ParameterChange { .. } => ProposalKind::ParameterChange,
ProposalPayload::CommunityPoolSpend { .. } => ProposalKind::CommunityPoolSpend,
ProposalPayload::UpgradePlan { .. } => ProposalKind::UpgradePlan,
ProposalPayload::FreezeIbcClient { .. } => ProposalKind::FreezeIbcClient,
ProposalPayload::UnfreezeIbcClient { .. } => ProposalKind::UnfreezeIbcClient,
}
}
}
#[derive(Debug, Clone, Eq, PartialEq)]
pub enum ProposalPayload {
Signaling {
commit: Option<String>,
},
Emergency {
halt_chain: bool,
},
ParameterChange(ParameterChange),
CommunityPoolSpend {
transaction_plan: Vec<u8>,
},
UpgradePlan { height: u64 },
FreezeIbcClient {
client_id: String,
},
UnfreezeIbcClient {
client_id: String,
},
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(tag = "kind", rename_all = "snake_case")]
pub enum ProposalPayloadToml {
Signaling { commit: Option<String> },
Emergency { halt_chain: bool },
ParameterChange(ParameterChange),
CommunityPoolSpend { transaction: String },
UpgradePlan { height: u64 },
FreezeIbcClient { client_id: String },
UnfreezeIbcClient { client_id: String },
}
impl TryFrom<ProposalPayloadToml> for ProposalPayload {
type Error = anyhow::Error;
fn try_from(toml: ProposalPayloadToml) -> Result<Self, Self::Error> {
Ok(match toml {
ProposalPayloadToml::Signaling { commit } => ProposalPayload::Signaling { commit },
ProposalPayloadToml::Emergency { halt_chain } => {
ProposalPayload::Emergency { halt_chain }
}
ProposalPayloadToml::ParameterChange(change) => {
ProposalPayload::ParameterChange(change)
}
ProposalPayloadToml::CommunityPoolSpend { transaction } => {
ProposalPayload::CommunityPoolSpend {
transaction_plan: Bytes::from(
base64::Engine::decode(
&base64::engine::general_purpose::STANDARD,
transaction,
)
.context("couldn't decode transaction plan from base64")?,
)
.to_vec(),
}
}
ProposalPayloadToml::UpgradePlan { height } => ProposalPayload::UpgradePlan { height },
ProposalPayloadToml::FreezeIbcClient { client_id } => {
ProposalPayload::FreezeIbcClient { client_id }
}
ProposalPayloadToml::UnfreezeIbcClient { client_id } => {
ProposalPayload::UnfreezeIbcClient { client_id }
}
})
}
}
impl From<ProposalPayload> for ProposalPayloadToml {
fn from(payload: ProposalPayload) -> Self {
match payload {
ProposalPayload::Signaling { commit } => ProposalPayloadToml::Signaling { commit },
ProposalPayload::Emergency { halt_chain } => {
ProposalPayloadToml::Emergency { halt_chain }
}
ProposalPayload::ParameterChange(change) => {
ProposalPayloadToml::ParameterChange(change)
}
ProposalPayload::CommunityPoolSpend { transaction_plan } => {
ProposalPayloadToml::CommunityPoolSpend {
transaction: base64::Engine::encode(
&base64::engine::general_purpose::STANDARD,
transaction_plan,
),
}
}
ProposalPayload::UpgradePlan { height } => ProposalPayloadToml::UpgradePlan { height },
ProposalPayload::FreezeIbcClient { client_id } => {
ProposalPayloadToml::FreezeIbcClient { client_id }
}
ProposalPayload::UnfreezeIbcClient { client_id } => {
ProposalPayloadToml::UnfreezeIbcClient { client_id }
}
}
}
}
impl ProposalPayload {
pub fn is_signaling(&self) -> bool {
matches!(self, ProposalPayload::Signaling { .. })
}
pub fn is_emergency(&self) -> bool {
matches!(self, ProposalPayload::Emergency { .. })
}
pub fn is_ibc_freeze(&self) -> bool {
matches!(self, ProposalPayload::FreezeIbcClient { .. })
|| matches!(self, ProposalPayload::UnfreezeIbcClient { .. })
}
pub fn is_parameter_change(&self) -> bool {
matches!(self, ProposalPayload::ParameterChange { .. })
}
pub fn is_community_pool_spend(&self) -> bool {
matches!(self, ProposalPayload::CommunityPoolSpend { .. })
}
}