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
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
// Copyright 2020 ChainSafe Systems
// SPDX-License-Identifier: Apache-2.0

//! Account Timing

use ark_ff::{One, Zero};
use mina_serialization_types_macros::AutoFrom;
use proof_systems::{mina_hasher::Fp, ChunkedROInput, ToChunkedROInput};
use smart_default::SmartDefault;

use crate::{
    from_graphql_json::FromGraphQLJson,
    numbers::{Amount, BlockTime},
};

/// Payload for the timing variant Timed
#[derive(Clone, Debug, Eq, PartialEq, AutoFrom)]
#[auto_from(mina_serialization_types::account::TimedData)]
#[auto_from(mina_serialization_types::account::TimedDataV0)]
pub struct TimedData {
    /// Initial balance for the account
    pub initial_minimum_balance: Amount,
    /// Time when all balance is avaiable
    pub cliff_time: BlockTime,
    /// Amount extra available when fully fested
    pub cliff_amount: Amount,
    /// Period in whcih allocation is released in chunks
    pub vesting_period: BlockTime,
    /// Amount released in each vesting period
    pub vesting_increment: Amount,
}

impl Default for TimedData {
    fn default() -> Self {
        Self {
            initial_minimum_balance: 0.into(),
            cliff_time: 0.into(),
            cliff_amount: 0.into(),
            vesting_period: 1.into(),
            vesting_increment: 0.into(),
        }
    }
}

impl ToChunkedROInput for TimedData {
    fn to_chunked_roinput(&self) -> ChunkedROInput {
        ChunkedROInput::new()
            .append_chunked(&self.initial_minimum_balance)
            .append_u32(self.cliff_time.0 as u32)
            .append_chunked(&self.cliff_amount)
            .append_u32(self.vesting_period.0 as u32)
            .append_chunked(&self.vesting_increment)
    }
}

impl FromGraphQLJson for TimedData {
    fn from_graphql_json(json: &serde_json::Value) -> anyhow::Result<Self>
    where
        Self: Sized,
    {
        Ok(Self {
            initial_minimum_balance: Amount(
                json["initialMinimumBalance"]
                    .as_str()
                    .unwrap_or_default()
                    .parse()?,
            ),
            cliff_time: BlockTime(json["cliffTime"].as_str().unwrap_or_default().parse()?),
            cliff_amount: Amount(json["cliffAmount"].as_str().unwrap_or_default().parse()?),
            vesting_period: BlockTime(json["vestingPeriod"].as_str().unwrap_or_default().parse()?),
            vesting_increment: Amount(
                json["vestingIncrement"]
                    .as_str()
                    .unwrap_or_default()
                    .parse()?,
            ),
        })
    }
}

/// Timing information for an account with regard to when its balance is accessable
/// This is to allow vesting from an initial genesis allocation
#[derive(Debug, Clone, SmartDefault, AutoFrom)]
#[auto_from(mina_serialization_types::account::Timing)]
#[auto_from(mina_serialization_types::account::TimingV0)]
pub enum Timing {
    /// Account does not have any timing limitations
    #[default]
    Untimed,
    /// Account does have timing limitations as specified
    Timed(TimedData),
}

impl FromGraphQLJson for Timing {
    fn from_graphql_json(json: &serde_json::Value) -> anyhow::Result<Self>
    where
        Self: Sized,
    {
        Ok(match TimedData::from_graphql_json(json) {
            Ok(data) => Self::Timed(data),
            _ => Self::Untimed,
        })
    }
}

impl ToChunkedROInput for Timing {
    fn to_chunked_roinput(&self) -> ChunkedROInput {
        match &self {
            Self::Untimed => ChunkedROInput::new()
                .append_packed(Fp::zero(), 1)
                .append_chunked(&TimedData::default()),
            Self::Timed(timed) => ChunkedROInput::new()
                .append_packed(Fp::one(), 1)
                .append_chunked(timed),
        }
    }
}