pmonitor/
config.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
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
118
//! Logic for reading and writing config files for `pmonitor`, in the TOML format.
use anyhow::Result;
use regex::Regex;
use serde::{Deserialize, Serialize};
use url::Url;
use uuid::Uuid;

use penumbra_sdk_keys::FullViewingKey;
use penumbra_sdk_num::Amount;

#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct FvkEntry {
    pub fvk: FullViewingKey,
    pub wallet_id: Uuid,
}

#[derive(Clone, Debug, Serialize, Deserialize)]
/// Representation of a single Penumbra wallet to track.
pub struct AccountConfig {
    /// The initial [FullViewingKey] has specified during `pmonitor init`.
    ///
    /// Distinct because the tool understands account migrations.
    original: FvkEntry,
    /// The amount held by the account at the time of genesis.
    genesis_balance: Amount,
    /// List of account migrations, performed via `pcli migrate balance`, if any.
    migrations: Vec<FvkEntry>,
}

impl AccountConfig {
    pub fn new(original: FvkEntry, genesis_balance: Amount) -> Self {
        Self {
            original,
            genesis_balance,
            migrations: vec![],
        }
    }

    /// Get original/genesis FVK.
    pub fn original_fvk(&self) -> FullViewingKey {
        self.original.fvk.clone()
    }

    /// Get genesis balance.
    pub fn genesis_balance(&self) -> Amount {
        self.genesis_balance
    }

    /// Add migration to the account config.
    pub fn add_migration(&mut self, fvk_entry: FvkEntry) {
        self.migrations.push(fvk_entry);
    }

    /// Get the active wallet, which is the last migration or the original FVK if no migrations have occurred.
    pub fn active_wallet(&self) -> FvkEntry {
        if self.migrations.is_empty() {
            self.original.clone()
        } else {
            self.migrations
                .last()
                .expect("migrations must not be empty")
                .clone()
        }
    }

    pub fn active_fvk(&self) -> FullViewingKey {
        self.active_wallet().fvk
    }

    pub fn active_uuid(&self) -> Uuid {
        self.active_wallet().wallet_id
    }
}

#[derive(Clone, Debug, Serialize, Deserialize)]
/// The primary TOML file for configuring `pmonitor`, containing all its account info.
///
/// During `pmonitor audit` runs, the config will be automatically updated
/// if tracked FVKs were detected to migrate, via `pcli migrate balance`, to save time
/// on future syncs.
pub struct PmonitorConfig {
    /// The gRPC URL for a Penumbra node's `pd` endpoint, used for retrieving account activity.
    grpc_url: Url,
    /// The list of Penumbra wallets to track.
    accounts: Vec<AccountConfig>,
}

impl PmonitorConfig {
    pub fn new(grpc_url: Url, accounts: Vec<AccountConfig>) -> Self {
        Self { grpc_url, accounts }
    }

    pub fn grpc_url(&self) -> Url {
        self.grpc_url.clone()
    }

    pub fn accounts(&self) -> &Vec<AccountConfig> {
        &self.accounts
    }

    pub fn set_account(&mut self, index: usize, account: AccountConfig) {
        self.accounts[index] = account;
    }
}

/// Get the destination FVK from a migration memo.
pub fn parse_dest_fvk_from_memo(memo: &str) -> Result<FullViewingKey> {
    let re = Regex::new(r"Migrating balance from .+ to (.+)")?;
    if let Some(captures) = re.captures(memo) {
        if let Some(dest_fvk_str) = captures.get(1) {
            return dest_fvk_str
                .as_str()
                .parse::<FullViewingKey>()
                .map_err(|_| anyhow::anyhow!("Invalid destination FVK in memo"));
        }
    }
    Err(anyhow::anyhow!("Could not parse destination FVK from memo"))
}