penumbra_sdk_community_pool/action/
community_pool_output.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
use anyhow::{Context, Error};
use serde::{Deserialize, Serialize};
use std::convert::{TryFrom, TryInto};

use penumbra_sdk_asset::{Balance, Value};
use penumbra_sdk_keys::Address;
use penumbra_sdk_proto::{penumbra::core::component::governance::v1 as pb, DomainType};
use penumbra_sdk_txhash::{EffectHash, EffectingData};

#[derive(Clone, Debug, Serialize, Deserialize)]
#[serde(try_from = "pb::CommunityPoolOutput", into = "pb::CommunityPoolOutput")]
pub struct CommunityPoolOutput {
    pub value: Value,
    pub address: Address,
}

impl CommunityPoolOutput {
    pub fn balance(&self) -> Balance {
        // Outputs from the Community Pool require value
        -Balance::from(self.value)
    }
}

impl EffectingData for CommunityPoolOutput {
    fn effect_hash(&self) -> EffectHash {
        EffectHash::from_proto_effecting_data(&self.to_proto())
    }
}

impl DomainType for CommunityPoolOutput {
    type Proto = pb::CommunityPoolOutput;
}

impl From<CommunityPoolOutput> for pb::CommunityPoolOutput {
    fn from(msg: CommunityPoolOutput) -> Self {
        pb::CommunityPoolOutput {
            value: Some(msg.value.into()),
            address: Some(msg.address.into()),
        }
    }
}

impl TryFrom<pb::CommunityPoolOutput> for CommunityPoolOutput {
    type Error = Error;

    fn try_from(proto: pb::CommunityPoolOutput) -> anyhow::Result<Self, Self::Error> {
        let value = proto
            .value
            .ok_or_else(|| anyhow::anyhow!("missing value"))?
            .try_into()
            .context("malformed value")?;
        let address = proto
            .address
            .ok_or_else(|| anyhow::anyhow!("missing address"))?
            .try_into()
            .context("malformed address")?;

        Ok(CommunityPoolOutput { value, address })
    }
}