mirror of
https://github.com/darkrenaissance/darkfi.git
synced 2026-04-28 03:00:18 -04:00
3549 lines
138 KiB
Rust
3549 lines
138 KiB
Rust
/* This file is part of DarkFi (https://dark.fi)
|
|
*
|
|
* Copyright (C) 2020-2026 Dyne.org foundation
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
use std::{collections::HashMap, fmt, str::FromStr};
|
|
|
|
use lazy_static::lazy_static;
|
|
use num_bigint::BigUint;
|
|
use rand::rngs::OsRng;
|
|
use rusqlite::types::Value;
|
|
|
|
use darkfi::{
|
|
tx::{ContractCallLeaf, Transaction, TransactionBuilder},
|
|
util::{
|
|
encoding::base64,
|
|
parse::{decode_base10, encode_base10},
|
|
},
|
|
zk::{empty_witnesses, halo2::Field, ProvingKey, ZkCircuit},
|
|
zkas::ZkBinary,
|
|
Error, Result,
|
|
};
|
|
use darkfi_dao_contract::{
|
|
blockwindow,
|
|
client::{
|
|
make_mint_call, DaoAuthMoneyTransferCall, DaoExecCall, DaoProposeCall,
|
|
DaoProposeStakeInput, DaoVoteCall, DaoVoteInput,
|
|
},
|
|
model::{
|
|
Dao, DaoAuthCall, DaoBulla, DaoExecParams, DaoMintParams, DaoProposal, DaoProposalBulla,
|
|
DaoProposeParams, DaoVoteParams,
|
|
},
|
|
DaoFunction, DAO_CONTRACT_ZKAS_AUTH_MONEY_TRANSFER_ENC_COIN_NS,
|
|
DAO_CONTRACT_ZKAS_AUTH_MONEY_TRANSFER_NS, DAO_CONTRACT_ZKAS_EARLY_EXEC_NS,
|
|
DAO_CONTRACT_ZKAS_EXEC_NS, DAO_CONTRACT_ZKAS_MINT_NS, DAO_CONTRACT_ZKAS_PROPOSE_INPUT_NS,
|
|
DAO_CONTRACT_ZKAS_PROPOSE_MAIN_NS, DAO_CONTRACT_ZKAS_VOTE_INPUT_NS,
|
|
DAO_CONTRACT_ZKAS_VOTE_MAIN_NS,
|
|
};
|
|
use darkfi_money_contract::{
|
|
client::transfer_v1::{select_coins, TransferCallBuilder, TransferCallInput},
|
|
model::{CoinAttributes, Nullifier, TokenId},
|
|
MoneyFunction, MONEY_CONTRACT_ZKAS_BURN_NS_V1, MONEY_CONTRACT_ZKAS_FEE_NS_V1,
|
|
MONEY_CONTRACT_ZKAS_MINT_NS_V1,
|
|
};
|
|
use darkfi_sdk::{
|
|
bridgetree,
|
|
crypto::{
|
|
keypair::{Address, PublicKey, SecretKey, StandardAddress},
|
|
pasta_prelude::PrimeField,
|
|
poseidon_hash,
|
|
smt::{MemoryStorageFp, PoseidonFp, SmtMemoryFp, EMPTY_NODES_FP},
|
|
util::{fp_mod_fv, fp_to_u64},
|
|
BaseBlind, Blind, FuncId, FuncRef, MerkleNode, MerkleTree, ScalarBlind, DAO_CONTRACT_ID,
|
|
MONEY_CONTRACT_ID,
|
|
},
|
|
dark_tree::DarkTree,
|
|
pasta::pallas,
|
|
tx::TransactionHash,
|
|
ContractCall,
|
|
};
|
|
use darkfi_serial::{
|
|
async_trait, deserialize_async, serialize, serialize_async, AsyncEncodable, SerialDecodable,
|
|
SerialEncodable,
|
|
};
|
|
|
|
use crate::{
|
|
cache::{CacheOverlay, CacheSmt, CacheSmtStorage, SLED_MONEY_SMT_TREE},
|
|
convert_named_params,
|
|
error::{WalletDbError, WalletDbResult},
|
|
money::BALANCE_BASE10_DECIMALS,
|
|
rpc::ScanCache,
|
|
Drk,
|
|
};
|
|
|
|
// DAO Merkle trees Sled keys
|
|
pub const SLED_MERKLE_TREES_DAO_DAOS: &[u8] = b"_dao_daos";
|
|
pub const SLED_MERKLE_TREES_DAO_PROPOSALS: &[u8] = b"_dao_proposals";
|
|
|
|
// Wallet SQL table constant names. These have to represent the `dao.sql`
|
|
// SQL schema. Table names are prefixed with the contract ID to avoid collisions.
|
|
lazy_static! {
|
|
pub static ref DAO_DAOS_TABLE: String = format!("{}_dao_daos", DAO_CONTRACT_ID.to_string());
|
|
pub static ref DAO_COINS_TABLE: String = format!("{}_dao_coins", DAO_CONTRACT_ID.to_string());
|
|
pub static ref DAO_PROPOSALS_TABLE: String =
|
|
format!("{}_dao_proposals", DAO_CONTRACT_ID.to_string());
|
|
pub static ref DAO_VOTES_TABLE: String = format!("{}_dao_votes", DAO_CONTRACT_ID.to_string());
|
|
}
|
|
|
|
// DAO_DAOS_TABLE
|
|
pub const DAO_DAOS_COL_BULLA: &str = "bulla";
|
|
pub const DAO_DAOS_COL_NAME: &str = "name";
|
|
pub const DAO_DAOS_COL_PARAMS: &str = "params";
|
|
pub const DAO_DAOS_COL_LEAF_POSITION: &str = "leaf_position";
|
|
pub const DAO_DAOS_COL_MINT_HEIGHT: &str = "mint_height";
|
|
pub const DAO_DAOS_COL_TX_HASH: &str = "tx_hash";
|
|
pub const DAO_DAOS_COL_CALL_INDEX: &str = "call_index";
|
|
|
|
// DAO_PROPOSALS_TABLE
|
|
pub const DAO_PROPOSALS_COL_BULLA: &str = "bulla";
|
|
pub const DAO_PROPOSALS_COL_DAO_BULLA: &str = "dao_bulla";
|
|
pub const DAO_PROPOSALS_COL_PROPOSAL: &str = "proposal";
|
|
pub const DAO_PROPOSALS_COL_DATA: &str = "data";
|
|
pub const DAO_PROPOSALS_COL_LEAF_POSITION: &str = "leaf_position";
|
|
pub const DAO_PROPOSALS_COL_MONEY_SNAPSHOT_TREE: &str = "money_snapshot_tree";
|
|
pub const DAO_PROPOSALS_COL_NULLIFIERS_SMT_SNAPSHOT: &str = "nullifiers_smt_snapshot";
|
|
pub const DAO_PROPOSALS_COL_MINT_HEIGHT: &str = "mint_height";
|
|
pub const DAO_PROPOSALS_COL_TX_HASH: &str = "tx_hash";
|
|
pub const DAO_PROPOSALS_COL_CALL_INDEX: &str = "call_index";
|
|
pub const DAO_PROPOSALS_COL_EXEC_HEIGHT: &str = "exec_height";
|
|
pub const DAO_PROPOSALS_COL_EXEC_TX_HASH: &str = "exec_tx_hash";
|
|
|
|
// DAO_VOTES_TABLE
|
|
pub const DAO_VOTES_COL_PROPOSAL_BULLA: &str = "proposal_bulla";
|
|
pub const DAO_VOTES_COL_VOTE_OPTION: &str = "vote_option";
|
|
pub const DAO_VOTES_COL_YES_VOTE_BLIND: &str = "yes_vote_blind";
|
|
pub const DAO_VOTES_COL_ALL_VOTE_VALUE: &str = "all_vote_value";
|
|
pub const DAO_VOTES_COL_ALL_VOTE_BLIND: &str = "all_vote_blind";
|
|
pub const DAO_VOTES_COL_BLOCK_HEIGHT: &str = "block_height";
|
|
pub const DAO_VOTES_COL_TX_HASH: &str = "tx_hash";
|
|
pub const DAO_VOTES_COL_CALL_INDEX: &str = "call_index";
|
|
pub const DAO_VOTES_COL_NULLIFIERS: &str = "nullifiers";
|
|
|
|
#[derive(Debug, Clone, SerialEncodable, SerialDecodable)]
|
|
/// Parameters representing a DAO to be initialized
|
|
pub struct DaoParams {
|
|
/// The on chain representation of the DAO
|
|
pub dao: Dao,
|
|
/// DAO notes decryption secret key
|
|
pub notes_secret_key: Option<SecretKey>,
|
|
/// DAO proposals creator secret key
|
|
pub proposer_secret_key: Option<SecretKey>,
|
|
/// DAO proposals viewer secret key
|
|
pub proposals_secret_key: Option<SecretKey>,
|
|
/// DAO votes viewer secret key
|
|
pub votes_secret_key: Option<SecretKey>,
|
|
/// DAO proposals executor secret key
|
|
pub exec_secret_key: Option<SecretKey>,
|
|
/// DAO strongly supported proposals executor secret key
|
|
pub early_exec_secret_key: Option<SecretKey>,
|
|
}
|
|
|
|
impl DaoParams {
|
|
/// Generate new `DaoParams`. If a specific secret key is provided,
|
|
/// the corresponding public key will be derived from it and ignore the provided one.
|
|
#[allow(clippy::too_many_arguments)]
|
|
pub fn new(
|
|
proposer_limit: u64,
|
|
quorum: u64,
|
|
early_exec_quorum: u64,
|
|
approval_ratio_base: u64,
|
|
approval_ratio_quot: u64,
|
|
gov_token_id: TokenId,
|
|
notes_secret_key: Option<SecretKey>,
|
|
notes_public_key: PublicKey,
|
|
proposer_secret_key: Option<SecretKey>,
|
|
proposer_public_key: PublicKey,
|
|
proposals_secret_key: Option<SecretKey>,
|
|
proposals_public_key: PublicKey,
|
|
votes_secret_key: Option<SecretKey>,
|
|
votes_public_key: PublicKey,
|
|
exec_secret_key: Option<SecretKey>,
|
|
exec_public_key: PublicKey,
|
|
early_exec_secret_key: Option<SecretKey>,
|
|
early_exec_public_key: PublicKey,
|
|
bulla_blind: BaseBlind,
|
|
) -> Self {
|
|
// Derive corresponding keys from their secret or use the provided ones.
|
|
let notes_public_key = match notes_secret_key {
|
|
Some(secret_key) => PublicKey::from_secret(secret_key),
|
|
None => notes_public_key,
|
|
};
|
|
let proposer_public_key = match proposer_secret_key {
|
|
Some(secret_key) => PublicKey::from_secret(secret_key),
|
|
None => proposer_public_key,
|
|
};
|
|
let proposals_public_key = match proposals_secret_key {
|
|
Some(secret_key) => PublicKey::from_secret(secret_key),
|
|
None => proposals_public_key,
|
|
};
|
|
let votes_public_key = match votes_secret_key {
|
|
Some(secret_key) => PublicKey::from_secret(secret_key),
|
|
None => votes_public_key,
|
|
};
|
|
let exec_public_key = match exec_secret_key {
|
|
Some(secret_key) => PublicKey::from_secret(secret_key),
|
|
None => exec_public_key,
|
|
};
|
|
let early_exec_public_key = match early_exec_secret_key {
|
|
Some(secret_key) => PublicKey::from_secret(secret_key),
|
|
None => early_exec_public_key,
|
|
};
|
|
|
|
let dao = Dao {
|
|
proposer_limit,
|
|
quorum,
|
|
early_exec_quorum,
|
|
approval_ratio_base,
|
|
approval_ratio_quot,
|
|
gov_token_id,
|
|
notes_public_key,
|
|
proposer_public_key,
|
|
proposals_public_key,
|
|
votes_public_key,
|
|
exec_public_key,
|
|
early_exec_public_key,
|
|
bulla_blind,
|
|
};
|
|
Self {
|
|
dao,
|
|
notes_secret_key,
|
|
proposer_secret_key,
|
|
proposals_secret_key,
|
|
votes_secret_key,
|
|
exec_secret_key,
|
|
early_exec_secret_key,
|
|
}
|
|
}
|
|
|
|
/// Parse provided toml string into `DaoParams`.
|
|
/// If a specific secret key is provided, the corresponding public key
|
|
/// will be derived from it and ignore the provided one.
|
|
pub fn from_toml_str(toml: &str) -> Result<Self> {
|
|
// Parse TOML file contents
|
|
let Ok(contents) = toml::from_str::<toml::Value>(toml) else {
|
|
return Err(Error::ParseFailed("Failed parsing TOML config"))
|
|
};
|
|
let Some(table) = contents.as_table() else {
|
|
return Err(Error::ParseFailed("TOML not a map"))
|
|
};
|
|
|
|
// Grab configuration parameters
|
|
let Some(proposer_limit) = table.get("proposer_limit") else {
|
|
return Err(Error::ParseFailed("TOML does not contain proposer limit"))
|
|
};
|
|
let Some(proposer_limit) = proposer_limit.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid proposer limit: Not a string"))
|
|
};
|
|
if f64::from_str(proposer_limit).is_err() {
|
|
return Err(Error::ParseFailed("Invalid proposer limit: Cannot be parsed to float"))
|
|
}
|
|
let proposer_limit = decode_base10(proposer_limit, BALANCE_BASE10_DECIMALS, true)?;
|
|
|
|
let Some(quorum) = table.get("quorum") else {
|
|
return Err(Error::ParseFailed("TOML does not contain quorum"))
|
|
};
|
|
let Some(quorum) = quorum.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid quorum: Not a string"))
|
|
};
|
|
if f64::from_str(quorum).is_err() {
|
|
return Err(Error::ParseFailed("Invalid quorum: Cannot be parsed to float"))
|
|
}
|
|
let quorum = decode_base10(quorum, BALANCE_BASE10_DECIMALS, true)?;
|
|
|
|
let Some(early_exec_quorum) = table.get("early_exec_quorum") else {
|
|
return Err(Error::ParseFailed("TOML does not contain early exec quorum"))
|
|
};
|
|
let Some(early_exec_quorum) = early_exec_quorum.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid early exec quorum: Not a string"))
|
|
};
|
|
if f64::from_str(early_exec_quorum).is_err() {
|
|
return Err(Error::ParseFailed("Invalid early exec quorum: Cannot be parsed to float"))
|
|
}
|
|
let early_exec_quorum = decode_base10(early_exec_quorum, BALANCE_BASE10_DECIMALS, true)?;
|
|
|
|
let Some(approval_ratio) = table.get("approval_ratio") else {
|
|
return Err(Error::ParseFailed("TOML does not contain approval ratio"))
|
|
};
|
|
let Some(approval_ratio) = approval_ratio.as_float() else {
|
|
return Err(Error::ParseFailed("Invalid approval ratio: Not a float"))
|
|
};
|
|
if approval_ratio > 1.0 {
|
|
return Err(Error::ParseFailed("Approval ratio cannot be >1.0"))
|
|
}
|
|
let approval_ratio_base = 100_u64;
|
|
let approval_ratio_quot = (approval_ratio * approval_ratio_base as f64) as u64;
|
|
|
|
let Some(gov_token_id) = table.get("gov_token_id") else {
|
|
return Err(Error::ParseFailed("TOML does not contain gov token id"))
|
|
};
|
|
let Some(gov_token_id) = gov_token_id.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid gov token id: Not a string"))
|
|
};
|
|
let gov_token_id = TokenId::from_str(gov_token_id)?;
|
|
|
|
let Some(bulla_blind) = table.get("bulla_blind") else {
|
|
return Err(Error::ParseFailed("TOML does not contain bulla blind"))
|
|
};
|
|
let Some(bulla_blind) = bulla_blind.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid bulla blind: Not a string"))
|
|
};
|
|
let bulla_blind = BaseBlind::from_str(bulla_blind)?;
|
|
|
|
// Grab DAO actions keypairs
|
|
let notes_secret_key = match table.get("notes_secret_key") {
|
|
Some(notes_secret_key) => {
|
|
let Some(notes_secret_key) = notes_secret_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid notes secret key: Not a string"))
|
|
};
|
|
let Ok(notes_secret_key) = SecretKey::from_str(notes_secret_key) else {
|
|
return Err(Error::ParseFailed("Invalid notes secret key: Decoding failed"))
|
|
};
|
|
Some(notes_secret_key)
|
|
}
|
|
None => None,
|
|
};
|
|
let notes_public_key = match notes_secret_key {
|
|
Some(notes_secret_key) => PublicKey::from_secret(notes_secret_key),
|
|
None => {
|
|
let Some(notes_public_key) = table.get("notes_public_key") else {
|
|
return Err(Error::ParseFailed("TOML does not contain notes public key"))
|
|
};
|
|
let Some(notes_public_key) = notes_public_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid notes public key: Not a string"))
|
|
};
|
|
let Ok(notes_public_key) = PublicKey::from_str(notes_public_key) else {
|
|
return Err(Error::ParseFailed("Invalid notes public key: Decoding failed"))
|
|
};
|
|
notes_public_key
|
|
}
|
|
};
|
|
|
|
let proposer_secret_key = match table.get("proposer_secret_key") {
|
|
Some(proposer_secret_key) => {
|
|
let Some(proposer_secret_key) = proposer_secret_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid proposer secret key: Not a string"))
|
|
};
|
|
let Ok(proposer_secret_key) = SecretKey::from_str(proposer_secret_key) else {
|
|
return Err(Error::ParseFailed("Invalid proposer secret key: Decoding failed"))
|
|
};
|
|
Some(proposer_secret_key)
|
|
}
|
|
None => None,
|
|
};
|
|
let proposer_public_key = match proposer_secret_key {
|
|
Some(proposer_secret_key) => PublicKey::from_secret(proposer_secret_key),
|
|
None => {
|
|
let Some(proposer_public_key) = table.get("proposer_public_key") else {
|
|
return Err(Error::ParseFailed("TOML does not contain proposer public key"))
|
|
};
|
|
let Some(proposer_public_key) = proposer_public_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid proposer public key: Not a string"))
|
|
};
|
|
let Ok(proposer_public_key) = PublicKey::from_str(proposer_public_key) else {
|
|
return Err(Error::ParseFailed("Invalid proposer public key: Decoding failed"))
|
|
};
|
|
proposer_public_key
|
|
}
|
|
};
|
|
|
|
let proposals_secret_key = match table.get("proposals_secret_key") {
|
|
Some(proposals_secret_key) => {
|
|
let Some(proposals_secret_key) = proposals_secret_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid proposals secret key: Not a string"))
|
|
};
|
|
let Ok(proposals_secret_key) = SecretKey::from_str(proposals_secret_key) else {
|
|
return Err(Error::ParseFailed("Invalid proposals secret key: Decoding failed"))
|
|
};
|
|
Some(proposals_secret_key)
|
|
}
|
|
None => None,
|
|
};
|
|
let proposals_public_key = match proposals_secret_key {
|
|
Some(proposals_secret_key) => PublicKey::from_secret(proposals_secret_key),
|
|
None => {
|
|
let Some(proposals_public_key) = table.get("proposals_public_key") else {
|
|
return Err(Error::ParseFailed("TOML does not contain proposals public key"))
|
|
};
|
|
let Some(proposals_public_key) = proposals_public_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid proposals public key: Not a string"))
|
|
};
|
|
let Ok(proposals_public_key) = PublicKey::from_str(proposals_public_key) else {
|
|
return Err(Error::ParseFailed("Invalid proposals public key: Decoding failed"))
|
|
};
|
|
proposals_public_key
|
|
}
|
|
};
|
|
|
|
let votes_secret_key = match table.get("votes_secret_key") {
|
|
Some(votes_secret_key) => {
|
|
let Some(votes_secret_key) = votes_secret_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid votes secret key: Not a string"))
|
|
};
|
|
let Ok(votes_secret_key) = SecretKey::from_str(votes_secret_key) else {
|
|
return Err(Error::ParseFailed("Invalid votes secret key: Decoding failed"))
|
|
};
|
|
Some(votes_secret_key)
|
|
}
|
|
None => None,
|
|
};
|
|
let votes_public_key = match votes_secret_key {
|
|
Some(votes_secret_key) => PublicKey::from_secret(votes_secret_key),
|
|
None => {
|
|
let Some(votes_public_key) = table.get("votes_public_key") else {
|
|
return Err(Error::ParseFailed("TOML does not contain votes public key"))
|
|
};
|
|
let Some(votes_public_key) = votes_public_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid votes public key: Not a string"))
|
|
};
|
|
let Ok(votes_public_key) = PublicKey::from_str(votes_public_key) else {
|
|
return Err(Error::ParseFailed("Invalid votes public key: Decoding failed"))
|
|
};
|
|
votes_public_key
|
|
}
|
|
};
|
|
|
|
let exec_secret_key = match table.get("exec_secret_key") {
|
|
Some(exec_secret_key) => {
|
|
let Some(exec_secret_key) = exec_secret_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid exec secret key: Not a string"))
|
|
};
|
|
let Ok(exec_secret_key) = SecretKey::from_str(exec_secret_key) else {
|
|
return Err(Error::ParseFailed("Invalid exec secret key: Decoding failed"))
|
|
};
|
|
Some(exec_secret_key)
|
|
}
|
|
None => None,
|
|
};
|
|
let exec_public_key = match exec_secret_key {
|
|
Some(exec_secret_key) => PublicKey::from_secret(exec_secret_key),
|
|
None => {
|
|
let Some(exec_public_key) = table.get("exec_public_key") else {
|
|
return Err(Error::ParseFailed("TOML does not contain exec public key"))
|
|
};
|
|
let Some(exec_public_key) = exec_public_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid exec public key: Not a string"))
|
|
};
|
|
let Ok(exec_public_key) = PublicKey::from_str(exec_public_key) else {
|
|
return Err(Error::ParseFailed("Invalid exec public key: Decoding failed"))
|
|
};
|
|
exec_public_key
|
|
}
|
|
};
|
|
|
|
let early_exec_secret_key = match table.get("early_exec_secret_key") {
|
|
Some(early_exec_secret_key) => {
|
|
let Some(early_exec_secret_key) = early_exec_secret_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid early exec secret key: Not a string"))
|
|
};
|
|
let Ok(early_exec_secret_key) = SecretKey::from_str(early_exec_secret_key) else {
|
|
return Err(Error::ParseFailed("Invalid early exec secret key: Decoding failed"))
|
|
};
|
|
Some(early_exec_secret_key)
|
|
}
|
|
None => None,
|
|
};
|
|
let early_exec_public_key = match early_exec_secret_key {
|
|
Some(early_exec_secret_key) => PublicKey::from_secret(early_exec_secret_key),
|
|
None => {
|
|
let Some(early_exec_public_key) = table.get("early_exec_public_key") else {
|
|
return Err(Error::ParseFailed("TOML does not contain early exec public key"))
|
|
};
|
|
let Some(early_exec_public_key) = early_exec_public_key.as_str() else {
|
|
return Err(Error::ParseFailed("Invalid early exec public key: Not a string"))
|
|
};
|
|
let Ok(early_exec_public_key) = PublicKey::from_str(early_exec_public_key) else {
|
|
return Err(Error::ParseFailed("Invalid early exec public key: Decoding failed"))
|
|
};
|
|
early_exec_public_key
|
|
}
|
|
};
|
|
|
|
Ok(Self::new(
|
|
proposer_limit,
|
|
quorum,
|
|
early_exec_quorum,
|
|
approval_ratio_base,
|
|
approval_ratio_quot,
|
|
gov_token_id,
|
|
notes_secret_key,
|
|
notes_public_key,
|
|
proposer_secret_key,
|
|
proposer_public_key,
|
|
proposals_secret_key,
|
|
proposals_public_key,
|
|
votes_secret_key,
|
|
votes_public_key,
|
|
exec_secret_key,
|
|
exec_public_key,
|
|
early_exec_secret_key,
|
|
early_exec_public_key,
|
|
bulla_blind,
|
|
))
|
|
}
|
|
|
|
/// Generate a toml string containing the DAO configuration.
|
|
pub fn toml_str(&self) -> String {
|
|
// Header comments
|
|
let mut toml = String::from(
|
|
"## DAO configuration file\n\
|
|
##\n\
|
|
## Please make sure you go through all the settings so you can configure\n\
|
|
## your DAO properly.\n\
|
|
##\n\
|
|
## If you want to restrict access to certain actions, the corresponding\n\
|
|
## secret key can be omitted. All public keys, along with the DAO configuration\n\
|
|
## parameters must be shared.\n\
|
|
##\n\
|
|
## If you want to combine access to certain actions, you can use the same\n\
|
|
## secret and public key combination for them.\n\n",
|
|
);
|
|
|
|
// Configuration parameters
|
|
toml += &format!(
|
|
"## ====== DAO configuration parameters =====\n\n\
|
|
## The minimum amount of governance tokens needed to open a proposal for this DAO\n\
|
|
proposer_limit = \"{}\"\n\n\
|
|
## Minimal threshold of participating total tokens needed for a proposal to pass\n\
|
|
quorum = \"{}\"\n\n\
|
|
## Minimal threshold of participating total tokens needed for a proposal to\n\
|
|
## be considered strongly supported, enabling early execution.\n\
|
|
## Must be greater than or equal to normal quorum.\n\
|
|
early_exec_quorum = \"{}\"\n\n\
|
|
## The ratio of yes votes/total votes needed for a proposal to pass (2 decimals)\n\
|
|
approval_ratio = {}\n\n\
|
|
## DAO's governance token ID\n\
|
|
gov_token_id = \"{}\"\n\n\
|
|
## Bulla blind\n\
|
|
bulla_blind = \"{}\"\n\n",
|
|
encode_base10(self.dao.proposer_limit, BALANCE_BASE10_DECIMALS),
|
|
encode_base10(self.dao.quorum, BALANCE_BASE10_DECIMALS),
|
|
encode_base10(self.dao.early_exec_quorum, BALANCE_BASE10_DECIMALS),
|
|
self.dao.approval_ratio_quot as f64 / self.dao.approval_ratio_base as f64,
|
|
self.dao.gov_token_id,
|
|
self.dao.bulla_blind,
|
|
);
|
|
|
|
// DAO actions keypairs
|
|
toml += &format!(
|
|
"## ====== DAO actions keypairs =====\n\n\
|
|
## DAO notes decryption keypair\n\
|
|
notes_public_key = \"{}\"\n",
|
|
self.dao.notes_public_key,
|
|
);
|
|
match self.notes_secret_key {
|
|
Some(secret_key) => toml += &format!("notes_secret_key = \"{secret_key}\"\n\n"),
|
|
None => toml += "\n",
|
|
}
|
|
toml += &format!(
|
|
"## DAO proposals creator keypair\n\
|
|
proposer_public_key = \"{}\"\n",
|
|
self.dao.proposer_public_key,
|
|
);
|
|
match self.proposer_secret_key {
|
|
Some(secret_key) => toml += &format!("proposer_secret_key = \"{secret_key}\"\n\n"),
|
|
None => toml += "\n",
|
|
}
|
|
toml += &format!(
|
|
"## DAO proposals viewer keypair\n\
|
|
proposals_public_key = \"{}\"\n",
|
|
self.dao.proposals_public_key,
|
|
);
|
|
match self.proposals_secret_key {
|
|
Some(secret_key) => toml += &format!("proposals_secret_key = \"{secret_key}\"\n\n"),
|
|
None => toml += "\n",
|
|
}
|
|
toml += &format!(
|
|
"## DAO votes viewer keypair\n\
|
|
votes_public_key = \"{}\"\n",
|
|
self.dao.votes_public_key,
|
|
);
|
|
match self.votes_secret_key {
|
|
Some(secret_key) => toml += &format!("votes_secret_key = \"{secret_key}\"\n\n"),
|
|
None => toml += "\n",
|
|
}
|
|
toml += &format!(
|
|
"## DAO proposals executor keypair\n\
|
|
exec_public_key = \"{}\"\n",
|
|
self.dao.exec_public_key,
|
|
);
|
|
match self.exec_secret_key {
|
|
Some(secret_key) => toml += &format!("exec_secret_key = \"{secret_key}\"\n\n"),
|
|
None => toml += "\n",
|
|
}
|
|
toml += &format!(
|
|
"## DAO strongly supported proposals executor keypair\n\
|
|
early_exec_public_key = \"{}\"",
|
|
self.dao.early_exec_public_key,
|
|
);
|
|
if let Some(secret_key) = self.early_exec_secret_key {
|
|
toml += &format!("\nearly_exec_secret_key = \"{secret_key}\"")
|
|
}
|
|
|
|
toml
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for DaoParams {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
// Grab known secret keys
|
|
let notes_secret_key = match self.notes_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let proposer_secret_key = match self.proposer_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let proposals_secret_key = match self.proposals_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let votes_secret_key = match self.votes_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let exec_secret_key = match self.exec_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let early_exec_secret_key = match self.early_exec_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
|
|
let s = format!(
|
|
"{}\n{}\n{}: {} ({})\n{}: {} ({})\n{}: {} ({})\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}",
|
|
"DAO Parameters",
|
|
"==============",
|
|
"Proposer limit",
|
|
encode_base10(self.dao.proposer_limit, BALANCE_BASE10_DECIMALS),
|
|
self.dao.proposer_limit,
|
|
"Quorum",
|
|
encode_base10(self.dao.quorum, BALANCE_BASE10_DECIMALS),
|
|
self.dao.quorum,
|
|
"Early Exec Quorum",
|
|
encode_base10(self.dao.early_exec_quorum, BALANCE_BASE10_DECIMALS),
|
|
self.dao.early_exec_quorum,
|
|
"Approval ratio",
|
|
self.dao.approval_ratio_quot as f64 / self.dao.approval_ratio_base as f64,
|
|
"Governance Token ID",
|
|
self.dao.gov_token_id,
|
|
"Notes Public key",
|
|
self.dao.notes_public_key,
|
|
"Notes Secret key",
|
|
notes_secret_key,
|
|
"Proposer Public key",
|
|
self.dao.proposer_public_key,
|
|
"Proposer Secret key",
|
|
proposer_secret_key,
|
|
"Proposals Public key",
|
|
self.dao.proposals_public_key,
|
|
"Proposals Secret key",
|
|
proposals_secret_key,
|
|
"Votes Public key",
|
|
self.dao.votes_public_key,
|
|
"Votes Secret key",
|
|
votes_secret_key,
|
|
"Exec Public key",
|
|
self.dao.exec_public_key,
|
|
"Exec Secret key",
|
|
exec_secret_key,
|
|
"Early Exec Public key",
|
|
self.dao.early_exec_public_key,
|
|
"Early Exec Secret key",
|
|
early_exec_secret_key,
|
|
"Bulla blind",
|
|
self.dao.bulla_blind,
|
|
);
|
|
|
|
write!(f, "{s}")
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
/// Structure representing a `DAO_DAOS_TABLE` record.
|
|
pub struct DaoRecord {
|
|
/// Name identifier for the DAO
|
|
pub name: String,
|
|
/// DAO parameters
|
|
pub params: DaoParams,
|
|
/// Leaf position of the DAO in the Merkle tree of DAOs
|
|
pub leaf_position: Option<bridgetree::Position>,
|
|
/// Block height of the transaction this DAO was deployed
|
|
pub mint_height: Option<u32>,
|
|
/// The transaction hash where the DAO was deployed
|
|
pub tx_hash: Option<TransactionHash>,
|
|
/// The call index in the transaction where the DAO was deployed
|
|
pub call_index: Option<u8>,
|
|
}
|
|
|
|
impl DaoRecord {
|
|
pub fn new(
|
|
name: String,
|
|
params: DaoParams,
|
|
leaf_position: Option<bridgetree::Position>,
|
|
mint_height: Option<u32>,
|
|
tx_hash: Option<TransactionHash>,
|
|
call_index: Option<u8>,
|
|
) -> Self {
|
|
Self { name, params, leaf_position, mint_height, tx_hash, call_index }
|
|
}
|
|
|
|
pub fn bulla(&self) -> DaoBulla {
|
|
self.params.dao.to_bulla()
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for DaoRecord {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
// Grab known secret keys
|
|
let notes_secret_key = match self.params.notes_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let proposer_secret_key = match self.params.proposer_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let proposals_secret_key = match self.params.proposals_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let votes_secret_key = match self.params.votes_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let exec_secret_key = match self.params.exec_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let early_exec_secret_key = match self.params.early_exec_secret_key {
|
|
Some(secret_key) => format!("{secret_key}"),
|
|
None => "None".to_string(),
|
|
};
|
|
|
|
// Grab mint information
|
|
let leaf_position = match self.leaf_position {
|
|
Some(p) => format!("{p:?}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let mint_height = match self.mint_height {
|
|
Some(h) => format!("{h}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let tx_hash = match self.tx_hash {
|
|
Some(t) => format!("{t}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let call_index = match self.call_index {
|
|
Some(c) => format!("{c}"),
|
|
None => "None".to_string(),
|
|
};
|
|
|
|
let s = format!(
|
|
"{}\n{}\n{}: {}\n{}: {}\n{}: {} ({})\n{}: {} ({})\n{}: {} ({})\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}",
|
|
"DAO Parameters",
|
|
"==============",
|
|
"Name",
|
|
self.name,
|
|
"Bulla",
|
|
self.bulla(),
|
|
"Proposer limit",
|
|
encode_base10(self.params.dao.proposer_limit, BALANCE_BASE10_DECIMALS),
|
|
self.params.dao.proposer_limit,
|
|
"Quorum",
|
|
encode_base10(self.params.dao.quorum, BALANCE_BASE10_DECIMALS),
|
|
self.params.dao.quorum,
|
|
"Early Exec Quorum",
|
|
encode_base10(self.params.dao.early_exec_quorum, BALANCE_BASE10_DECIMALS),
|
|
self.params.dao.early_exec_quorum,
|
|
"Approval ratio",
|
|
self.params.dao.approval_ratio_quot as f64 / self.params.dao.approval_ratio_base as f64,
|
|
"Governance Token ID",
|
|
self.params.dao.gov_token_id,
|
|
"Notes Public key",
|
|
self.params.dao.notes_public_key,
|
|
"Notes Secret key",
|
|
notes_secret_key,
|
|
"Proposer Public key",
|
|
self.params.dao.proposer_public_key,
|
|
"Proposer Secret key",
|
|
proposer_secret_key,
|
|
"Proposals Public key",
|
|
self.params.dao.proposals_public_key,
|
|
"Proposals Secret key",
|
|
proposals_secret_key,
|
|
"Votes Public key",
|
|
self.params.dao.votes_public_key,
|
|
"Votes Secret key",
|
|
votes_secret_key,
|
|
"Exec Public key",
|
|
self.params.dao.exec_public_key,
|
|
"Exec Secret key",
|
|
exec_secret_key,
|
|
"Early Exec Public key",
|
|
self.params.dao.early_exec_public_key,
|
|
"Early Exec Secret key",
|
|
early_exec_secret_key,
|
|
"Bulla blind",
|
|
self.params.dao.bulla_blind,
|
|
"Leaf position",
|
|
leaf_position,
|
|
"Mint height",
|
|
mint_height,
|
|
"Transaction hash",
|
|
tx_hash,
|
|
"Call index",
|
|
call_index,
|
|
);
|
|
|
|
write!(f, "{s}")
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone, SerialEncodable, SerialDecodable)]
|
|
/// Structure representing a `DAO_PROPOSALS_TABLE` record.
|
|
pub struct ProposalRecord {
|
|
/// The on chain representation of the proposal
|
|
pub proposal: DaoProposal,
|
|
/// Plaintext proposal call data the members share between them
|
|
pub data: Option<Vec<u8>>,
|
|
/// Leaf position of the proposal in the Merkle tree of proposals
|
|
pub leaf_position: Option<bridgetree::Position>,
|
|
/// Money merkle tree snapshot for reproducing the snapshot Merkle root
|
|
pub money_snapshot_tree: Option<MerkleTree>,
|
|
/// Money nullifiers SMT snapshot for reproducing the snapshot Merkle root
|
|
pub nullifiers_smt_snapshot: Option<HashMap<BigUint, pallas::Base>>,
|
|
/// Block height of the transaction this proposal was deployed
|
|
pub mint_height: Option<u32>,
|
|
/// The transaction hash where the proposal was deployed
|
|
pub tx_hash: Option<TransactionHash>,
|
|
/// The call index in the transaction where the proposal was deployed
|
|
pub call_index: Option<u8>,
|
|
/// Block height of the transaction this proposal was executed
|
|
pub exec_height: Option<u32>,
|
|
/// The transaction hash where the proposal was executed
|
|
pub exec_tx_hash: Option<TransactionHash>,
|
|
}
|
|
|
|
impl ProposalRecord {
|
|
pub fn bulla(&self) -> DaoProposalBulla {
|
|
self.proposal.to_bulla()
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for ProposalRecord {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
let leaf_position = match self.leaf_position {
|
|
Some(p) => format!("{p:?}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let mint_height = match self.mint_height {
|
|
Some(h) => format!("{h}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let tx_hash = match self.tx_hash {
|
|
Some(t) => format!("{t}"),
|
|
None => "None".to_string(),
|
|
};
|
|
let call_index = match self.call_index {
|
|
Some(c) => format!("{c}"),
|
|
None => "None".to_string(),
|
|
};
|
|
|
|
let s = format!(
|
|
"{}\n{}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {}\n{}: {} ({})",
|
|
"Proposal parameters",
|
|
"===================",
|
|
"Bulla",
|
|
self.bulla(),
|
|
"DAO Bulla",
|
|
self.proposal.dao_bulla,
|
|
"Proposal leaf position",
|
|
leaf_position,
|
|
"Proposal mint height",
|
|
mint_height,
|
|
"Proposal transaction hash",
|
|
tx_hash,
|
|
"Proposal call index",
|
|
call_index,
|
|
"Creation block window",
|
|
self.proposal.creation_blockwindow,
|
|
"Duration",
|
|
self.proposal.duration_blockwindows,
|
|
"Block windows"
|
|
);
|
|
|
|
write!(f, "{s}")
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
/// Structure representing a `DAO_VOTES_TABLE` record.
|
|
pub struct VoteRecord {
|
|
/// Numeric identifier for the vote
|
|
pub id: u64,
|
|
/// Bulla identifier of the proposal this vote is for
|
|
pub proposal: DaoProposalBulla,
|
|
/// The vote
|
|
pub vote_option: bool,
|
|
/// Blinding factor for the yes vote
|
|
pub yes_vote_blind: ScalarBlind,
|
|
/// Value of all votes
|
|
pub all_vote_value: u64,
|
|
/// Blinding facfor of all votes
|
|
pub all_vote_blind: ScalarBlind,
|
|
/// Block height of the transaction this vote was casted
|
|
pub block_height: u32,
|
|
/// Transaction hash where this vote was casted
|
|
pub tx_hash: TransactionHash,
|
|
/// Call index in the transaction where this vote was casted
|
|
pub call_index: u8,
|
|
/// Vote input nullifiers
|
|
pub nullifiers: Vec<Nullifier>,
|
|
}
|
|
|
|
impl Drk {
|
|
/// Initialize wallet with tables for the DAO contract.
|
|
pub async fn initialize_dao(&self) -> WalletDbResult<()> {
|
|
// Initialize DAO wallet schema
|
|
let wallet_schema = include_str!("../dao.sql");
|
|
self.wallet.exec_batch_sql(wallet_schema)?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Fetch DAO Merkle trees from the wallet.
|
|
/// If a tree doesn't exists a new Merkle Tree is returned.
|
|
pub async fn get_dao_trees(&self) -> Result<(MerkleTree, MerkleTree)> {
|
|
let daos_tree = match self.cache.merkle_trees.get(SLED_MERKLE_TREES_DAO_DAOS)? {
|
|
Some(tree_bytes) => deserialize_async(&tree_bytes).await?,
|
|
None => MerkleTree::new(u32::MAX as usize),
|
|
};
|
|
let proposals_tree = match self.cache.merkle_trees.get(SLED_MERKLE_TREES_DAO_PROPOSALS)? {
|
|
Some(tree_bytes) => deserialize_async(&tree_bytes).await?,
|
|
None => MerkleTree::new(u32::MAX as usize),
|
|
};
|
|
Ok((daos_tree, proposals_tree))
|
|
}
|
|
|
|
/// Auxiliary function to parse a `DAO_DAOS_TABLE` record.
|
|
async fn parse_dao_record(&self, row: &[Value]) -> Result<DaoRecord> {
|
|
let Value::Text(ref name) = row[1] else {
|
|
return Err(Error::ParseFailed("[parse_dao_record] Name parsing failed"))
|
|
};
|
|
let name = name.clone();
|
|
|
|
let Value::Blob(ref params_bytes) = row[2] else {
|
|
return Err(Error::ParseFailed("[parse_dao_record] Params bytes parsing failed"))
|
|
};
|
|
let params = deserialize_async(params_bytes).await?;
|
|
|
|
let leaf_position = match row[3] {
|
|
Value::Blob(ref leaf_position_bytes) => {
|
|
Some(deserialize_async(leaf_position_bytes).await?)
|
|
}
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_record] Leaf position bytes parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
let mint_height = match row[4] {
|
|
Value::Integer(mint_height) => {
|
|
let Ok(mint_height) = u32::try_from(mint_height) else {
|
|
return Err(Error::ParseFailed("[parse_dao_record] Mint height parsing failed"))
|
|
};
|
|
Some(mint_height)
|
|
}
|
|
Value::Null => None,
|
|
_ => return Err(Error::ParseFailed("[parse_dao_record] Mint height parsing failed")),
|
|
};
|
|
|
|
let tx_hash = match row[5] {
|
|
Value::Blob(ref tx_hash_bytes) => Some(deserialize_async(tx_hash_bytes).await?),
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_record] Transaction hash bytes parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
let call_index = match row[6] {
|
|
Value::Integer(call_index) => {
|
|
let Ok(call_index) = u8::try_from(call_index) else {
|
|
return Err(Error::ParseFailed("[parse_dao_record] Call index parsing failed"))
|
|
};
|
|
Some(call_index)
|
|
}
|
|
Value::Null => None,
|
|
_ => return Err(Error::ParseFailed("[parse_dao_record] Call index parsing failed")),
|
|
};
|
|
|
|
let dao = DaoRecord::new(name, params, leaf_position, mint_height, tx_hash, call_index);
|
|
|
|
Ok(dao)
|
|
}
|
|
|
|
/// Fetch all known DAOs from the wallet.
|
|
pub async fn get_daos(&self) -> Result<Vec<DaoRecord>> {
|
|
let rows = match self.wallet.query_multiple(&DAO_DAOS_TABLE, &[], &[]) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!("[get_daos] DAOs retrieval failed: {e}")))
|
|
}
|
|
};
|
|
|
|
let mut daos = Vec::with_capacity(rows.len());
|
|
for row in rows {
|
|
daos.push(self.parse_dao_record(&row).await?);
|
|
}
|
|
|
|
Ok(daos)
|
|
}
|
|
|
|
/// Auxiliary function to parse a proposal record row.
|
|
async fn parse_dao_proposal(&self, row: &[Value]) -> Result<ProposalRecord> {
|
|
let Value::Blob(ref proposal_bytes) = row[2] else {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Proposal bytes bytes parsing failed",
|
|
))
|
|
};
|
|
let proposal = deserialize_async(proposal_bytes).await?;
|
|
|
|
let data = match row[3] {
|
|
Value::Blob(ref data_bytes) => Some(data_bytes.clone()),
|
|
Value::Null => None,
|
|
_ => return Err(Error::ParseFailed("[parse_dao_proposal] Data bytes parsing failed")),
|
|
};
|
|
|
|
let leaf_position = match row[4] {
|
|
Value::Blob(ref leaf_position_bytes) => {
|
|
Some(deserialize_async(leaf_position_bytes).await?)
|
|
}
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Leaf position bytes parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
let money_snapshot_tree = match row[5] {
|
|
Value::Blob(ref money_snapshot_tree_bytes) => {
|
|
Some(deserialize_async(money_snapshot_tree_bytes).await?)
|
|
}
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Money snapshot tree bytes parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
let nullifiers_smt_snapshot = match row[6] {
|
|
Value::Blob(ref nullifiers_smt_snapshot_bytes) => {
|
|
Some(deserialize_async(nullifiers_smt_snapshot_bytes).await?)
|
|
}
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Nullifiers SMT snapshot bytes parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
let mint_height = match row[7] {
|
|
Value::Integer(mint_height) => {
|
|
let Ok(mint_height) = u32::try_from(mint_height) else {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Mint height parsing failed",
|
|
))
|
|
};
|
|
Some(mint_height)
|
|
}
|
|
Value::Null => None,
|
|
_ => return Err(Error::ParseFailed("[parse_dao_proposal] Mint height parsing failed")),
|
|
};
|
|
|
|
let tx_hash = match row[8] {
|
|
Value::Blob(ref tx_hash_bytes) => Some(deserialize_async(tx_hash_bytes).await?),
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Transaction hash bytes parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
let call_index = match row[9] {
|
|
Value::Integer(call_index) => {
|
|
let Ok(call_index) = u8::try_from(call_index) else {
|
|
return Err(Error::ParseFailed("[parse_dao_proposal] Call index parsing failed"))
|
|
};
|
|
Some(call_index)
|
|
}
|
|
Value::Null => None,
|
|
_ => return Err(Error::ParseFailed("[parse_dao_proposal] Call index parsing failed")),
|
|
};
|
|
|
|
let exec_height = match row[10] {
|
|
Value::Integer(exec_height) => {
|
|
let Ok(exec_height) = u32::try_from(exec_height) else {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Execution height parsing failed",
|
|
))
|
|
};
|
|
Some(exec_height)
|
|
}
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Execution height parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
let exec_tx_hash = match row[11] {
|
|
Value::Blob(ref exec_tx_hash_bytes) => {
|
|
Some(deserialize_async(exec_tx_hash_bytes).await?)
|
|
}
|
|
Value::Null => None,
|
|
_ => {
|
|
return Err(Error::ParseFailed(
|
|
"[parse_dao_proposal] Execution transaction hash bytes parsing failed",
|
|
))
|
|
}
|
|
};
|
|
|
|
Ok(ProposalRecord {
|
|
proposal,
|
|
data,
|
|
leaf_position,
|
|
money_snapshot_tree,
|
|
nullifiers_smt_snapshot,
|
|
mint_height,
|
|
tx_hash,
|
|
call_index,
|
|
exec_height,
|
|
exec_tx_hash,
|
|
})
|
|
}
|
|
|
|
/// Fetch all known DAO proposals from the wallet given a DAO name.
|
|
pub async fn get_dao_proposals(&self, name: &str) -> Result<Vec<ProposalRecord>> {
|
|
let Ok(dao) = self.get_dao_by_name(name).await else {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[get_dao_proposals] DAO with name {name} not found in wallet"
|
|
)))
|
|
};
|
|
|
|
let rows = match self.wallet.query_multiple(
|
|
&DAO_PROPOSALS_TABLE,
|
|
&[],
|
|
convert_named_params! {(DAO_PROPOSALS_COL_DAO_BULLA, serialize_async(&dao.bulla()).await)},
|
|
) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[get_dao_proposals] Proposals retrieval failed: {e}"
|
|
)))
|
|
}
|
|
};
|
|
|
|
let mut proposals = Vec::with_capacity(rows.len());
|
|
for row in rows {
|
|
let proposal = self.parse_dao_proposal(&row).await?;
|
|
proposals.push(proposal);
|
|
}
|
|
|
|
Ok(proposals)
|
|
}
|
|
|
|
/// Auxiliary function to apply `DaoFunction::Mint` call data to
|
|
/// the wallet and update the provided scan cache.
|
|
/// Returns a flag indicating if the provided call refers to our
|
|
/// own wallet.
|
|
async fn apply_dao_mint_data(
|
|
&self,
|
|
scan_cache: &mut ScanCache,
|
|
new_bulla: &DaoBulla,
|
|
tx_hash: &TransactionHash,
|
|
call_index: &u8,
|
|
mint_height: &u32,
|
|
) -> Result<bool> {
|
|
// Append the new dao bulla to the Merkle tree.
|
|
// Every dao bulla has to be added.
|
|
scan_cache.dao_daos_tree.append(MerkleNode::from(new_bulla.inner()));
|
|
|
|
// Check if we have the DAO
|
|
if !scan_cache.own_daos.contains_key(new_bulla) {
|
|
return Ok(false)
|
|
}
|
|
|
|
// Confirm it
|
|
scan_cache.log(format!(
|
|
"[apply_dao_mint_data] Found minted DAO {new_bulla}, noting down for wallet update"
|
|
));
|
|
if let Err(e) = self
|
|
.confirm_dao(
|
|
new_bulla,
|
|
&scan_cache.dao_daos_tree.mark().unwrap(),
|
|
tx_hash,
|
|
call_index,
|
|
mint_height,
|
|
)
|
|
.await
|
|
{
|
|
return Err(Error::DatabaseError(format!(
|
|
"[apply_dao_mint_data] Confirm DAO failed: {e}"
|
|
)))
|
|
}
|
|
|
|
Ok(true)
|
|
}
|
|
|
|
/// Auxiliary function to apply `DaoFunction::Propose` call data to
|
|
/// the wallet and update the provided scan cache.
|
|
/// Returns a flag indicating if the provided call refers to our
|
|
/// own wallet.
|
|
async fn apply_dao_propose_data(
|
|
&self,
|
|
scan_cache: &mut ScanCache,
|
|
params: &DaoProposeParams,
|
|
tx_hash: &TransactionHash,
|
|
call_index: &u8,
|
|
mint_height: &u32,
|
|
) -> Result<bool> {
|
|
// Append the new proposal bulla to the Merkle tree.
|
|
// Every proposal bulla has to be added.
|
|
scan_cache.dao_proposals_tree.append(MerkleNode::from(params.proposal_bulla.inner()));
|
|
|
|
// If we're able to decrypt this note, that's the way to link it
|
|
// to a specific DAO.
|
|
for (dao, (proposals_secret_key, _)) in &scan_cache.own_daos {
|
|
// Check if we have the proposals key
|
|
let Some(proposals_secret_key) = proposals_secret_key else { continue };
|
|
|
|
// Try to decrypt the proposal note
|
|
let Ok(note) = params.note.decrypt::<DaoProposal>(proposals_secret_key) else {
|
|
continue
|
|
};
|
|
|
|
// We managed to decrypt it. Let's place this in a proper ProposalRecord object
|
|
scan_cache.messages_buffer.push(format!(
|
|
"[apply_dao_propose_data] Managed to decrypt proposal note for DAO: {dao}"
|
|
));
|
|
|
|
// Check if we already got the record
|
|
let our_proposal = if scan_cache.own_proposals.contains_key(¶ms.proposal_bulla) {
|
|
// Grab the record from the db
|
|
let mut our_proposal =
|
|
self.get_dao_proposal_by_bulla(¶ms.proposal_bulla).await?;
|
|
our_proposal.leaf_position = scan_cache.dao_proposals_tree.mark();
|
|
our_proposal.money_snapshot_tree = Some(scan_cache.money_tree.clone());
|
|
our_proposal.nullifiers_smt_snapshot = Some(scan_cache.money_smt.store.snapshot()?);
|
|
our_proposal.mint_height = Some(*mint_height);
|
|
our_proposal.tx_hash = Some(*tx_hash);
|
|
our_proposal.call_index = Some(*call_index);
|
|
our_proposal
|
|
} else {
|
|
let our_proposal = ProposalRecord {
|
|
proposal: note,
|
|
data: None,
|
|
leaf_position: scan_cache.dao_proposals_tree.mark(),
|
|
money_snapshot_tree: Some(scan_cache.money_tree.clone()),
|
|
nullifiers_smt_snapshot: Some(scan_cache.money_smt.store.snapshot()?),
|
|
mint_height: Some(*mint_height),
|
|
tx_hash: Some(*tx_hash),
|
|
call_index: Some(*call_index),
|
|
exec_height: None,
|
|
exec_tx_hash: None,
|
|
};
|
|
scan_cache.own_proposals.insert(params.proposal_bulla, *dao);
|
|
our_proposal
|
|
};
|
|
|
|
// Update/store our record
|
|
if let Err(e) = self.put_dao_proposal(&our_proposal).await {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[apply_dao_propose_data] Put DAO proposals failed: {e}"
|
|
)))
|
|
}
|
|
|
|
return Ok(true)
|
|
}
|
|
|
|
Ok(false)
|
|
}
|
|
|
|
/// Auxiliary function to apply `DaoFunction::Vote` call data to
|
|
/// the wallet.
|
|
/// Returns a flag indicating if the provided call refers to our
|
|
/// own wallet.
|
|
async fn apply_dao_vote_data(
|
|
&self,
|
|
scan_cache: &ScanCache,
|
|
params: &DaoVoteParams,
|
|
tx_hash: &TransactionHash,
|
|
call_index: &u8,
|
|
block_height: &u32,
|
|
) -> Result<bool> {
|
|
// Check if we got the corresponding proposal
|
|
let Some(dao_bulla) = scan_cache.own_proposals.get(¶ms.proposal_bulla) else {
|
|
return Ok(false)
|
|
};
|
|
|
|
// Grab the proposal DAO votes key
|
|
let Some((_, votes_secret_key)) = scan_cache.own_daos.get(dao_bulla) else {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[apply_dao_vote_data] Couldn't find proposal {} DAO {}",
|
|
params.proposal_bulla, dao_bulla,
|
|
)))
|
|
};
|
|
|
|
// Check if we actually have the votes key
|
|
let Some(votes_secret_key) = votes_secret_key else { return Ok(false) };
|
|
|
|
// Decrypt the vote note
|
|
let note = match params.note.decrypt_unsafe(votes_secret_key) {
|
|
Ok(n) => n,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[apply_dao_vote_data] Couldn't decrypt proposal {} vote with DAO {} keys: {e}",
|
|
params.proposal_bulla, dao_bulla,
|
|
)))
|
|
}
|
|
};
|
|
|
|
// Create the DAO vote record
|
|
let vote_option = fp_to_u64(note[0]).unwrap();
|
|
if vote_option > 1 {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[apply_dao_vote_data] Malformed vote for proposal {}: {vote_option}",
|
|
params.proposal_bulla,
|
|
)))
|
|
}
|
|
let vote_option = vote_option != 0;
|
|
let yes_vote_blind = Blind(fp_mod_fv(note[1]));
|
|
let all_vote_value = fp_to_u64(note[2]).unwrap();
|
|
let all_vote_blind = Blind(fp_mod_fv(note[3]));
|
|
|
|
let v = VoteRecord {
|
|
id: 0, // This will be set by SQLite AUTOINCREMENT
|
|
proposal: params.proposal_bulla,
|
|
vote_option,
|
|
yes_vote_blind,
|
|
all_vote_value,
|
|
all_vote_blind,
|
|
block_height: *block_height,
|
|
tx_hash: *tx_hash,
|
|
call_index: *call_index,
|
|
nullifiers: params.inputs.iter().map(|i| i.vote_nullifier).collect(),
|
|
};
|
|
|
|
if let Err(e) = self.put_dao_vote(&v).await {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[apply_dao_vote_data] Put DAO votes failed: {e}"
|
|
)))
|
|
}
|
|
|
|
Ok(true)
|
|
}
|
|
|
|
/// Auxiliary function to apply `DaoFunction::Exec` call data to
|
|
/// the wallet and update the provided scan cache.
|
|
/// Returns a flag indicating if the provided call refers to our
|
|
/// own wallet.
|
|
async fn apply_dao_exec_data(
|
|
&self,
|
|
scan_cache: &ScanCache,
|
|
params: &DaoExecParams,
|
|
tx_hash: &TransactionHash,
|
|
exec_height: &u32,
|
|
) -> Result<bool> {
|
|
// Check if we got the corresponding proposal
|
|
if !scan_cache.own_proposals.contains_key(¶ms.proposal_bulla) {
|
|
return Ok(false)
|
|
}
|
|
|
|
// Grab proposal record key
|
|
let key = serialize_async(¶ms.proposal_bulla).await;
|
|
|
|
// Create an SQL `UPDATE` query to update proposal exec transaction hash
|
|
let query = format!(
|
|
"UPDATE {} SET {} = ?1, {} = ?2 WHERE {} = ?3;",
|
|
*DAO_PROPOSALS_TABLE,
|
|
DAO_PROPOSALS_COL_EXEC_HEIGHT,
|
|
DAO_PROPOSALS_COL_EXEC_TX_HASH,
|
|
DAO_PROPOSALS_COL_BULLA,
|
|
);
|
|
|
|
// Execute the query
|
|
if let Err(e) = self
|
|
.wallet
|
|
.exec_sql(&query, rusqlite::params![Some(*exec_height), Some(serialize(tx_hash)), key])
|
|
{
|
|
return Err(Error::DatabaseError(format!(
|
|
"[apply_dao_exec_data] Update DAO proposal failed: {e}"
|
|
)))
|
|
}
|
|
|
|
Ok(true)
|
|
}
|
|
|
|
/// Append data related to DAO contract transactions into the
|
|
/// wallet database and update the provided scan cache.
|
|
/// Returns a flag indicating if provided data refer to our own
|
|
/// wallet.
|
|
pub async fn apply_tx_dao_data(
|
|
&self,
|
|
scan_cache: &mut ScanCache,
|
|
data: &[u8],
|
|
tx_hash: &TransactionHash,
|
|
call_idx: &u8,
|
|
block_height: &u32,
|
|
) -> Result<bool> {
|
|
// Run through the transaction call data and see what we got:
|
|
match DaoFunction::try_from(data[0])? {
|
|
DaoFunction::Mint => {
|
|
scan_cache.log(String::from("[apply_tx_dao_data] Found Dao::Mint call"));
|
|
let params: DaoMintParams = deserialize_async(&data[1..]).await?;
|
|
self.apply_dao_mint_data(
|
|
scan_cache,
|
|
¶ms.dao_bulla,
|
|
tx_hash,
|
|
call_idx,
|
|
block_height,
|
|
)
|
|
.await
|
|
}
|
|
DaoFunction::Propose => {
|
|
scan_cache.log(String::from("[apply_tx_dao_data] Found Dao::Propose call"));
|
|
let params: DaoProposeParams = deserialize_async(&data[1..]).await?;
|
|
self.apply_dao_propose_data(scan_cache, ¶ms, tx_hash, call_idx, block_height)
|
|
.await
|
|
}
|
|
DaoFunction::Vote => {
|
|
scan_cache.log(String::from("[apply_tx_dao_data] Found Dao::Vote call"));
|
|
let params: DaoVoteParams = deserialize_async(&data[1..]).await?;
|
|
self.apply_dao_vote_data(scan_cache, ¶ms, tx_hash, call_idx, block_height).await
|
|
}
|
|
DaoFunction::Exec => {
|
|
scan_cache.log(String::from("[apply_tx_dao_data] Found Dao::Exec call"));
|
|
let params: DaoExecParams = deserialize_async(&data[1..]).await?;
|
|
self.apply_dao_exec_data(scan_cache, ¶ms, tx_hash, block_height).await
|
|
}
|
|
DaoFunction::AuthMoneyTransfer => {
|
|
scan_cache
|
|
.log(String::from("[apply_tx_dao_data] Found Dao::AuthMoneyTransfer call"));
|
|
// Does nothing, just verifies the other calls are correct
|
|
Ok(false)
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Confirm already imported DAO metadata into the wallet.
|
|
/// Here we just write the leaf position, mint height, tx hash,
|
|
/// and call index.
|
|
pub async fn confirm_dao(
|
|
&self,
|
|
dao: &DaoBulla,
|
|
leaf_position: &bridgetree::Position,
|
|
tx_hash: &TransactionHash,
|
|
call_index: &u8,
|
|
mint_height: &u32,
|
|
) -> WalletDbResult<()> {
|
|
// Grab dao record key
|
|
let key = serialize_async(dao).await;
|
|
|
|
// Create an SQL `UPDATE` query
|
|
let query = format!(
|
|
"UPDATE {} SET {} = ?1, {} = ?2, {} = ?3, {} = ?4 WHERE {} = ?5;",
|
|
*DAO_DAOS_TABLE,
|
|
DAO_DAOS_COL_LEAF_POSITION,
|
|
DAO_DAOS_COL_MINT_HEIGHT,
|
|
DAO_DAOS_COL_TX_HASH,
|
|
DAO_DAOS_COL_CALL_INDEX,
|
|
DAO_DAOS_COL_BULLA
|
|
);
|
|
|
|
// Create its params
|
|
let params = rusqlite::params![
|
|
serialize(leaf_position),
|
|
Some(*mint_height),
|
|
serialize(tx_hash),
|
|
call_index,
|
|
key,
|
|
];
|
|
|
|
// Execute the query
|
|
self.wallet.exec_sql(&query, params)
|
|
}
|
|
|
|
/// Import given DAO proposal into the wallet.
|
|
pub async fn put_dao_proposal(&self, proposal: &ProposalRecord) -> Result<()> {
|
|
// Check that we already have the proposal DAO
|
|
if let Err(e) = self.get_dao_by_bulla(&proposal.proposal.dao_bulla).await {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[put_dao_proposal] Couldn't find proposal {} DAO {}: {e}",
|
|
proposal.bulla(),
|
|
proposal.proposal.dao_bulla
|
|
)))
|
|
}
|
|
|
|
// Grab proposal record key
|
|
let key = serialize_async(&proposal.bulla()).await;
|
|
|
|
// Create an SQL `INSERT OR REPLACE` query
|
|
let query = format!(
|
|
"INSERT OR REPLACE INTO {} ({}, {}, {}, {}, {}, {}, {}, {}, {}, {}, {}, {}) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8, ?9, ?10, ?11, ?12);",
|
|
*DAO_PROPOSALS_TABLE,
|
|
DAO_PROPOSALS_COL_BULLA,
|
|
DAO_PROPOSALS_COL_DAO_BULLA,
|
|
DAO_PROPOSALS_COL_PROPOSAL,
|
|
DAO_PROPOSALS_COL_DATA,
|
|
DAO_PROPOSALS_COL_LEAF_POSITION,
|
|
DAO_PROPOSALS_COL_MONEY_SNAPSHOT_TREE,
|
|
DAO_PROPOSALS_COL_NULLIFIERS_SMT_SNAPSHOT,
|
|
DAO_PROPOSALS_COL_MINT_HEIGHT,
|
|
DAO_PROPOSALS_COL_TX_HASH,
|
|
DAO_PROPOSALS_COL_CALL_INDEX,
|
|
DAO_PROPOSALS_COL_EXEC_HEIGHT,
|
|
DAO_PROPOSALS_COL_EXEC_TX_HASH,
|
|
);
|
|
|
|
// Create its params
|
|
let data = match &proposal.data {
|
|
Some(data) => Some(data),
|
|
None => None,
|
|
};
|
|
|
|
let leaf_position = match &proposal.leaf_position {
|
|
Some(leaf_position) => Some(serialize_async(leaf_position).await),
|
|
None => None,
|
|
};
|
|
|
|
let money_snapshot_tree = match &proposal.money_snapshot_tree {
|
|
Some(money_snapshot_tree) => Some(serialize_async(money_snapshot_tree).await),
|
|
None => None,
|
|
};
|
|
|
|
let nullifiers_smt_snapshot = match &proposal.nullifiers_smt_snapshot {
|
|
Some(nullifiers_smt_snapshot) => Some(serialize_async(nullifiers_smt_snapshot).await),
|
|
None => None,
|
|
};
|
|
|
|
let tx_hash = match &proposal.tx_hash {
|
|
Some(tx_hash) => Some(serialize_async(tx_hash).await),
|
|
None => None,
|
|
};
|
|
|
|
let exec_tx_hash = match &proposal.exec_tx_hash {
|
|
Some(exec_tx_hash) => Some(serialize_async(exec_tx_hash).await),
|
|
None => None,
|
|
};
|
|
|
|
let params = rusqlite::params![
|
|
key,
|
|
serialize(&proposal.proposal.dao_bulla),
|
|
serialize(&proposal.proposal),
|
|
data,
|
|
leaf_position,
|
|
money_snapshot_tree,
|
|
nullifiers_smt_snapshot,
|
|
proposal.mint_height,
|
|
tx_hash,
|
|
proposal.call_index,
|
|
proposal.exec_height,
|
|
exec_tx_hash,
|
|
];
|
|
|
|
// Execute the query
|
|
if let Err(e) = self.wallet.exec_sql(&query, params) {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[put_dao_proposal] Proposal insert failed: {e}"
|
|
)))
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Import given DAO vote into the wallet.
|
|
pub async fn put_dao_vote(&self, vote: &VoteRecord) -> WalletDbResult<()> {
|
|
// Create an SQL `INSERT OR REPLACE` query
|
|
let query = format!(
|
|
"INSERT INTO {} ({}, {}, {}, {}, {}, {}, {}, {}, {}) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8, ?9);",
|
|
*DAO_VOTES_TABLE,
|
|
DAO_VOTES_COL_PROPOSAL_BULLA,
|
|
DAO_VOTES_COL_VOTE_OPTION,
|
|
DAO_VOTES_COL_YES_VOTE_BLIND,
|
|
DAO_VOTES_COL_ALL_VOTE_VALUE,
|
|
DAO_VOTES_COL_ALL_VOTE_BLIND,
|
|
DAO_VOTES_COL_BLOCK_HEIGHT,
|
|
DAO_VOTES_COL_TX_HASH,
|
|
DAO_VOTES_COL_CALL_INDEX,
|
|
DAO_VOTES_COL_NULLIFIERS,
|
|
);
|
|
|
|
// Create its params
|
|
let params = rusqlite::params![
|
|
serialize(&vote.proposal),
|
|
vote.vote_option as u64,
|
|
serialize(&vote.yes_vote_blind),
|
|
serialize(&vote.all_vote_value),
|
|
serialize(&vote.all_vote_blind),
|
|
vote.block_height,
|
|
serialize(&vote.tx_hash),
|
|
vote.call_index,
|
|
serialize(&vote.nullifiers),
|
|
];
|
|
|
|
// Execute the query
|
|
self.wallet.exec_sql(&query, params)?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reset the DAO Merkle trees in the cache.
|
|
pub fn reset_dao_trees(&self, output: &mut Vec<String>) -> WalletDbResult<()> {
|
|
output.push(String::from("Resetting DAO Merkle trees"));
|
|
if let Err(e) = self.cache.merkle_trees.remove(SLED_MERKLE_TREES_DAO_DAOS) {
|
|
output.push(format!("[reset_dao_trees] Resetting DAO DAOs Merkle tree failed: {e}"));
|
|
return Err(WalletDbError::GenericError)
|
|
}
|
|
if let Err(e) = self.cache.merkle_trees.remove(SLED_MERKLE_TREES_DAO_PROPOSALS) {
|
|
output
|
|
.push(format!("[reset_dao_trees] Resetting DAO Proposals Merkle tree failed: {e}"));
|
|
return Err(WalletDbError::GenericError)
|
|
}
|
|
output.push(String::from("Successfully reset DAO Merkle trees"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reset confirmed DAOs in the wallet.
|
|
pub fn reset_daos(&self, output: &mut Vec<String>) -> WalletDbResult<()> {
|
|
output.push(String::from("Resetting DAO confirmations"));
|
|
let query = format!(
|
|
"UPDATE {} SET {} = NULL, {} = NULL, {} = NULL, {} = NULL;",
|
|
*DAO_DAOS_TABLE,
|
|
DAO_DAOS_COL_LEAF_POSITION,
|
|
DAO_DAOS_COL_MINT_HEIGHT,
|
|
DAO_DAOS_COL_TX_HASH,
|
|
DAO_DAOS_COL_CALL_INDEX,
|
|
);
|
|
self.wallet.exec_sql(&query, &[])?;
|
|
output.push(String::from("Successfully unconfirmed DAOs"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reset confirmed DAOs in the wallet that were minted after
|
|
/// provided height.
|
|
pub fn unconfirm_daos_after(
|
|
&self,
|
|
height: &u32,
|
|
output: &mut Vec<String>,
|
|
) -> WalletDbResult<()> {
|
|
output.push(format!("Resetting DAO confirmations after: {height}"));
|
|
let query = format!(
|
|
"UPDATE {} SET {} = NULL, {} = NULL, {} = NULL, {} = NULL WHERE {} > ?1;",
|
|
*DAO_DAOS_TABLE,
|
|
DAO_DAOS_COL_LEAF_POSITION,
|
|
DAO_DAOS_COL_MINT_HEIGHT,
|
|
DAO_DAOS_COL_TX_HASH,
|
|
DAO_DAOS_COL_CALL_INDEX,
|
|
DAO_DAOS_COL_MINT_HEIGHT,
|
|
);
|
|
self.wallet.exec_sql(&query, rusqlite::params![Some(*height)])?;
|
|
output.push(String::from("Successfully unconfirmed DAOs"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reset all DAO proposals in the wallet.
|
|
pub fn reset_dao_proposals(&self, output: &mut Vec<String>) -> WalletDbResult<()> {
|
|
output.push(String::from("Resetting DAO proposals confirmations"));
|
|
let query = format!(
|
|
"UPDATE {} SET {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL;",
|
|
*DAO_PROPOSALS_TABLE,
|
|
DAO_PROPOSALS_COL_LEAF_POSITION,
|
|
DAO_PROPOSALS_COL_MONEY_SNAPSHOT_TREE,
|
|
DAO_PROPOSALS_COL_NULLIFIERS_SMT_SNAPSHOT,
|
|
DAO_PROPOSALS_COL_MINT_HEIGHT,
|
|
DAO_PROPOSALS_COL_TX_HASH,
|
|
DAO_PROPOSALS_COL_CALL_INDEX,
|
|
DAO_PROPOSALS_COL_EXEC_HEIGHT,
|
|
DAO_PROPOSALS_COL_EXEC_TX_HASH,
|
|
);
|
|
self.wallet.exec_sql(&query, &[])?;
|
|
output.push(String::from("Successfully unconfirmed DAO proposals"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reset DAO proposals in the wallet that were minted after
|
|
/// provided height.
|
|
pub fn unconfirm_dao_proposals_after(
|
|
&self,
|
|
height: &u32,
|
|
output: &mut Vec<String>,
|
|
) -> WalletDbResult<()> {
|
|
output.push(format!("Resetting DAO proposals confirmations after: {height}"));
|
|
let query = format!(
|
|
"UPDATE {} SET {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL, {} = NULL WHERE {} > ?1;",
|
|
*DAO_PROPOSALS_TABLE,
|
|
DAO_PROPOSALS_COL_LEAF_POSITION,
|
|
DAO_PROPOSALS_COL_MONEY_SNAPSHOT_TREE,
|
|
DAO_PROPOSALS_COL_NULLIFIERS_SMT_SNAPSHOT,
|
|
DAO_PROPOSALS_COL_MINT_HEIGHT,
|
|
DAO_PROPOSALS_COL_TX_HASH,
|
|
DAO_PROPOSALS_COL_CALL_INDEX,
|
|
DAO_PROPOSALS_COL_EXEC_HEIGHT,
|
|
DAO_PROPOSALS_COL_EXEC_TX_HASH,
|
|
DAO_PROPOSALS_COL_MINT_HEIGHT,
|
|
);
|
|
self.wallet.exec_sql(&query, rusqlite::params![Some(*height)])?;
|
|
output.push(String::from("Successfully unconfirmed DAO proposals"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reset execution information in the wallet for DAO proposals
|
|
/// that were executed after provided height.
|
|
pub fn unexec_dao_proposals_after(
|
|
&self,
|
|
height: &u32,
|
|
output: &mut Vec<String>,
|
|
) -> WalletDbResult<()> {
|
|
output.push(format!("Resetting DAO proposals execution information after: {height}"));
|
|
let query = format!(
|
|
"UPDATE {} SET {} = NULL, {} = NULL WHERE {} > ?1;",
|
|
*DAO_PROPOSALS_TABLE,
|
|
DAO_PROPOSALS_COL_EXEC_HEIGHT,
|
|
DAO_PROPOSALS_COL_EXEC_TX_HASH,
|
|
DAO_PROPOSALS_COL_EXEC_HEIGHT,
|
|
);
|
|
self.wallet.exec_sql(&query, rusqlite::params![Some(*height)])?;
|
|
output.push(String::from("Successfully reset DAO proposals execution information"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reset all DAO votes in the wallet.
|
|
pub fn reset_dao_votes(&self, output: &mut Vec<String>) -> WalletDbResult<()> {
|
|
output.push(String::from("Resetting DAO votes"));
|
|
let query = format!("DELETE FROM {};", *DAO_VOTES_TABLE);
|
|
self.wallet.exec_sql(&query, &[])?;
|
|
output.push(String::from("Successfully reset DAO votes"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Remove the DAO votes in the wallet that were created after
|
|
/// provided height.
|
|
pub fn remove_dao_votes_after(
|
|
&self,
|
|
height: &u32,
|
|
output: &mut Vec<String>,
|
|
) -> WalletDbResult<()> {
|
|
output.push(format!("Removing DAO votes after: {height}"));
|
|
let query =
|
|
format!("DELETE FROM {} WHERE {} > ?1;", *DAO_VOTES_TABLE, DAO_VOTES_COL_BLOCK_HEIGHT);
|
|
self.wallet.exec_sql(&query, rusqlite::params![height])?;
|
|
output.push(String::from("Successfully removed DAO votes"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Import given DAO params into the wallet with a given name.
|
|
pub async fn import_dao(
|
|
&self,
|
|
name: &str,
|
|
params: &DaoParams,
|
|
output: &mut Vec<String>,
|
|
) -> Result<()> {
|
|
// Grab the params DAO
|
|
let bulla = params.dao.to_bulla();
|
|
|
|
// Check if we already have imported the DAO so we retain its
|
|
// mint information.
|
|
if let Ok(dao) = self.get_dao_by_bulla(&bulla).await {
|
|
output.push(format!("Updating \"{}\" DAO keys and name into the wallet", dao.name));
|
|
let query = format!(
|
|
"UPDATE {} SET {} = ?1, {} = ?2 WHERE {} = ?3;",
|
|
*DAO_DAOS_TABLE, DAO_DAOS_COL_NAME, DAO_DAOS_COL_PARAMS, DAO_DAOS_COL_BULLA
|
|
);
|
|
if let Err(e) = self.wallet.exec_sql(
|
|
&query,
|
|
rusqlite::params![
|
|
name,
|
|
serialize_async(params).await,
|
|
serialize_async(&bulla).await
|
|
],
|
|
) {
|
|
return Err(Error::DatabaseError(format!("[import_dao] DAO update failed: {e}")))
|
|
};
|
|
return Ok(())
|
|
}
|
|
|
|
// Import the new DAO
|
|
output.push(format!("Importing \"{name}\" DAO into the wallet"));
|
|
let query = format!(
|
|
"INSERT INTO {} ({}, {}, {}) VALUES (?1, ?2, ?3);",
|
|
*DAO_DAOS_TABLE, DAO_DAOS_COL_BULLA, DAO_DAOS_COL_NAME, DAO_DAOS_COL_PARAMS
|
|
);
|
|
if let Err(e) = self.wallet.exec_sql(
|
|
&query,
|
|
rusqlite::params![
|
|
serialize_async(¶ms.dao.to_bulla()).await,
|
|
name,
|
|
serialize_async(params).await
|
|
],
|
|
) {
|
|
return Err(Error::DatabaseError(format!("[import_dao] DAO insert failed: {e}")))
|
|
};
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Remove a DAO and all its records given its name.
|
|
pub async fn remove_dao(&self, name: &str, output: &mut Vec<String>) -> Result<()> {
|
|
output.push(format!("Removing \"{name}\" DAO from the wallet"));
|
|
let query = format!("DELETE FROM {} WHERE {} = ?1;", *DAO_DAOS_TABLE, DAO_DAOS_COL_NAME);
|
|
if let Err(e) = self.wallet.exec_sql(&query, rusqlite::params![name]) {
|
|
return Err(Error::DatabaseError(format!("[remove_dao] DAO removal failed: {e}")))
|
|
};
|
|
output.push(String::from("Successfully removed DAO"));
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Fetch a DAO given its bulla.
|
|
pub async fn get_dao_by_bulla(&self, bulla: &DaoBulla) -> Result<DaoRecord> {
|
|
let row = match self.wallet.query_single(
|
|
&DAO_DAOS_TABLE,
|
|
&[],
|
|
convert_named_params! {(DAO_DAOS_COL_BULLA, serialize_async(bulla).await)},
|
|
) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[get_dao_by_bulla] DAO retrieval failed: {e}"
|
|
)))
|
|
}
|
|
};
|
|
|
|
self.parse_dao_record(&row).await
|
|
}
|
|
|
|
/// Fetch a DAO given its name.
|
|
pub async fn get_dao_by_name(&self, name: &str) -> Result<DaoRecord> {
|
|
let row = match self.wallet.query_single(
|
|
&DAO_DAOS_TABLE,
|
|
&[],
|
|
convert_named_params! {(DAO_DAOS_COL_NAME, name)},
|
|
) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[get_dao_by_name] DAO retrieval failed: {e}"
|
|
)))
|
|
}
|
|
};
|
|
|
|
self.parse_dao_record(&row).await
|
|
}
|
|
|
|
/// List DAO(s) imported in the wallet. If a name is given, just print the
|
|
/// metadata for that specific one, if found.
|
|
pub async fn dao_list(&self, name: &Option<String>, output: &mut Vec<String>) -> Result<()> {
|
|
if let Some(name) = name {
|
|
let dao = self.get_dao_by_name(name).await?;
|
|
output.push(format!("{dao}"));
|
|
let address: Address =
|
|
StandardAddress::from_public(self.network, dao.params.dao.notes_public_key).into();
|
|
output.push(format!("Wallet Address: {address}"));
|
|
return Ok(());
|
|
}
|
|
|
|
let daos = self.get_daos().await?;
|
|
for (i, dao) in daos.iter().enumerate() {
|
|
output.push(format!("{i}. {}", dao.name));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Fetch known unspent balances from the wallet for the given DAO name.
|
|
pub async fn dao_balance(&self, name: &str) -> Result<HashMap<String, u64>> {
|
|
let dao = self.get_dao_by_name(name).await?;
|
|
|
|
let dao_spend_hook =
|
|
FuncRef { contract_id: *DAO_CONTRACT_ID, func_code: DaoFunction::Exec as u8 }
|
|
.to_func_id();
|
|
|
|
let mut coins = self.get_coins(false).await?;
|
|
coins.retain(|x| x.0.note.spend_hook == dao_spend_hook);
|
|
coins.retain(|x| x.0.note.user_data == dao.bulla().inner());
|
|
|
|
// Fill this map with balances
|
|
let mut balmap: HashMap<String, u64> = HashMap::new();
|
|
|
|
for coin in coins {
|
|
let mut value = coin.0.note.value;
|
|
|
|
if let Some(prev) = balmap.get(&coin.0.note.token_id.to_string()) {
|
|
value += prev;
|
|
}
|
|
|
|
balmap.insert(coin.0.note.token_id.to_string(), value);
|
|
}
|
|
|
|
Ok(balmap)
|
|
}
|
|
|
|
/// Fetch known unspent balances from the wallet for the given DAO name.
|
|
pub async fn dao_mining_config(&self, name: &str, output: &mut Vec<String>) -> Result<()> {
|
|
let dao = self.get_dao_by_name(name).await?;
|
|
let address: Address =
|
|
StandardAddress::from_public(self.network, dao.params.dao.notes_public_key).into();
|
|
let recipient = address.to_string();
|
|
let spend_hook = format!(
|
|
"{}",
|
|
FuncRef { contract_id: *DAO_CONTRACT_ID, func_code: DaoFunction::Exec as u8 }
|
|
.to_func_id()
|
|
);
|
|
let user_data = bs58::encode(dao.bulla().inner().to_repr()).into_string();
|
|
output.push(String::from("DarkFi DAO mining configuration address:"));
|
|
output.push(
|
|
base64::encode(&serialize(&(recipient, Some(spend_hook), Some(user_data)))).to_string(),
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Fetch all known DAO proposalss from the wallet.
|
|
pub async fn get_proposals(&self) -> Result<Vec<ProposalRecord>> {
|
|
let rows = match self.wallet.query_multiple(&DAO_PROPOSALS_TABLE, &[], &[]) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[get_proposals] DAO proposalss retrieval failed: {e}"
|
|
)))
|
|
}
|
|
};
|
|
|
|
let mut daos = Vec::with_capacity(rows.len());
|
|
for row in rows {
|
|
daos.push(self.parse_dao_proposal(&row).await?);
|
|
}
|
|
|
|
Ok(daos)
|
|
}
|
|
|
|
/// Fetch a DAO proposal by its bulla.
|
|
pub async fn get_dao_proposal_by_bulla(
|
|
&self,
|
|
bulla: &DaoProposalBulla,
|
|
) -> Result<ProposalRecord> {
|
|
// Grab the proposal record
|
|
let row = match self.wallet.query_single(
|
|
&DAO_PROPOSALS_TABLE,
|
|
&[],
|
|
convert_named_params! {(DAO_PROPOSALS_COL_BULLA, serialize_async(bulla).await)},
|
|
) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[get_dao_proposal_by_bulla] DAO proposal retrieval failed: {e}"
|
|
)))
|
|
}
|
|
};
|
|
|
|
// Parse rest of the record
|
|
self.parse_dao_proposal(&row).await
|
|
}
|
|
|
|
// Fetch all known DAO proposal votes from the wallet given a proposal ID.
|
|
pub async fn get_dao_proposal_votes(
|
|
&self,
|
|
proposal: &DaoProposalBulla,
|
|
) -> Result<Vec<VoteRecord>> {
|
|
let rows = match self.wallet.query_multiple(
|
|
&DAO_VOTES_TABLE,
|
|
&[],
|
|
convert_named_params! {(DAO_VOTES_COL_PROPOSAL_BULLA, serialize_async(proposal).await)},
|
|
) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[get_dao_proposal_votes] Votes retrieval failed: {e}"
|
|
)))
|
|
}
|
|
};
|
|
|
|
let mut votes = Vec::with_capacity(rows.len());
|
|
for row in rows {
|
|
let Value::Integer(id) = row[0] else {
|
|
return Err(Error::ParseFailed("[get_dao_proposal_votes] ID parsing failed"))
|
|
};
|
|
let Ok(id) = u64::try_from(id) else {
|
|
return Err(Error::ParseFailed("[get_dao_proposal_votes] ID parsing failed"))
|
|
};
|
|
|
|
let Value::Blob(ref proposal_bytes) = row[1] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Proposal bytes bytes parsing failed",
|
|
))
|
|
};
|
|
let proposal = deserialize_async(proposal_bytes).await?;
|
|
|
|
let Value::Integer(vote_option) = row[2] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Vote option parsing failed",
|
|
))
|
|
};
|
|
let Ok(vote_option) = u32::try_from(vote_option) else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Vote option parsing failed",
|
|
))
|
|
};
|
|
let vote_option = vote_option != 0;
|
|
|
|
let Value::Blob(ref yes_vote_blind_bytes) = row[3] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Yes vote blind bytes parsing failed",
|
|
))
|
|
};
|
|
let yes_vote_blind = deserialize_async(yes_vote_blind_bytes).await?;
|
|
|
|
let Value::Blob(ref all_vote_value_bytes) = row[4] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] All vote value bytes parsing failed",
|
|
))
|
|
};
|
|
let all_vote_value = deserialize_async(all_vote_value_bytes).await?;
|
|
|
|
let Value::Blob(ref all_vote_blind_bytes) = row[5] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] All vote blind bytes parsing failed",
|
|
))
|
|
};
|
|
let all_vote_blind = deserialize_async(all_vote_blind_bytes).await?;
|
|
|
|
let Value::Integer(block_height) = row[6] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Block height parsing failed",
|
|
))
|
|
};
|
|
let Ok(block_height) = u32::try_from(block_height) else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Block height parsing failed",
|
|
))
|
|
};
|
|
|
|
let Value::Blob(ref tx_hash_bytes) = row[7] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Transaction hash bytes parsing failed",
|
|
))
|
|
};
|
|
let tx_hash = deserialize_async(tx_hash_bytes).await?;
|
|
|
|
let Value::Integer(call_index) = row[8] else {
|
|
return Err(Error::ParseFailed("[get_dao_proposal_votes] Call index parsing failed"))
|
|
};
|
|
let Ok(call_index) = u8::try_from(call_index) else {
|
|
return Err(Error::ParseFailed("[get_dao_proposal_votes] Call index parsing failed"))
|
|
};
|
|
|
|
let Value::Blob(ref nullifiers_bytes) = row[9] else {
|
|
return Err(Error::ParseFailed(
|
|
"[get_dao_proposal_votes] Nullifiers bytes parsing failed",
|
|
))
|
|
};
|
|
let nullifiers = deserialize_async(nullifiers_bytes).await?;
|
|
|
|
let vote = VoteRecord {
|
|
id,
|
|
proposal,
|
|
vote_option,
|
|
yes_vote_blind,
|
|
all_vote_value,
|
|
all_vote_blind,
|
|
block_height,
|
|
tx_hash,
|
|
call_index,
|
|
nullifiers,
|
|
};
|
|
|
|
votes.push(vote);
|
|
}
|
|
|
|
Ok(votes)
|
|
}
|
|
|
|
/// Mint a DAO on-chain.
|
|
pub async fn dao_mint(&self, name: &str) -> Result<Transaction> {
|
|
// Retrieve the dao record
|
|
let dao = self.get_dao_by_name(name).await?;
|
|
|
|
// Check its not already minted
|
|
if dao.tx_hash.is_some() {
|
|
return Err(Error::Custom(
|
|
"[dao_mint] This DAO seems to have already been minted on-chain".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check that we have all the keys
|
|
if dao.params.notes_secret_key.is_none() ||
|
|
dao.params.proposer_secret_key.is_none() ||
|
|
dao.params.proposals_secret_key.is_none() ||
|
|
dao.params.votes_secret_key.is_none() ||
|
|
dao.params.exec_secret_key.is_none() ||
|
|
dao.params.early_exec_secret_key.is_none()
|
|
{
|
|
return Err(Error::Custom(
|
|
"[dao_mint] We need all the secrets key to mint the DAO on-chain".to_string(),
|
|
))
|
|
}
|
|
|
|
// Now we need to do a lookup for the zkas proof bincodes, and create
|
|
// the circuit objects and proving keys so we can build the transaction.
|
|
// We also do this through the RPC. First we grab the fee call from money.
|
|
let zkas_bins = self.lookup_zkas(&MONEY_CONTRACT_ID).await?;
|
|
|
|
let Some(fee_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_FEE_NS_V1)
|
|
else {
|
|
return Err(Error::Custom("Fee circuit not found".to_string()))
|
|
};
|
|
|
|
let fee_zkbin = ZkBinary::decode(&fee_zkbin.1, false)?;
|
|
|
|
let fee_circuit = ZkCircuit::new(empty_witnesses(&fee_zkbin)?, &fee_zkbin);
|
|
|
|
// Creating Fee circuit proving key
|
|
let fee_pk = ProvingKey::build(fee_zkbin.k, &fee_circuit);
|
|
|
|
// Now we grab the DAO mint
|
|
let zkas_bins = self.lookup_zkas(&DAO_CONTRACT_ID).await?;
|
|
|
|
let Some(dao_mint_zkbin) = zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_MINT_NS)
|
|
else {
|
|
return Err(Error::DatabaseError("[dao_mint] DAO Mint circuit not found".to_string()))
|
|
};
|
|
|
|
let dao_mint_zkbin = ZkBinary::decode(&dao_mint_zkbin.1, false)?;
|
|
|
|
let dao_mint_circuit = ZkCircuit::new(empty_witnesses(&dao_mint_zkbin)?, &dao_mint_zkbin);
|
|
|
|
// Creating DAO Mint circuit proving key
|
|
let dao_mint_pk = ProvingKey::build(dao_mint_zkbin.k, &dao_mint_circuit);
|
|
|
|
// Create the DAO mint call
|
|
let notes_secret_key = dao.params.notes_secret_key.unwrap();
|
|
let (params, proofs) = make_mint_call(
|
|
&dao.params.dao,
|
|
¬es_secret_key,
|
|
&dao.params.proposer_secret_key.unwrap(),
|
|
&dao.params.proposals_secret_key.unwrap(),
|
|
&dao.params.votes_secret_key.unwrap(),
|
|
&dao.params.exec_secret_key.unwrap(),
|
|
&dao.params.early_exec_secret_key.unwrap(),
|
|
&dao_mint_zkbin,
|
|
&dao_mint_pk,
|
|
)?;
|
|
let mut data = vec![DaoFunction::Mint as u8];
|
|
params.encode_async(&mut data).await?;
|
|
let call = ContractCall { contract_id: *DAO_CONTRACT_ID, data };
|
|
|
|
// Create the TransactionBuilder containing above call
|
|
let mut tx_builder = TransactionBuilder::new(ContractCallLeaf { call, proofs }, vec![])?;
|
|
|
|
// We first have to execute the fee-less tx to gather its used gas, and then we feed
|
|
// it into the fee-creating function.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&[notes_secret_key])?;
|
|
tx.signatures.push(sigs);
|
|
|
|
let tree = self.get_money_tree().await?;
|
|
let (fee_call, fee_proofs, fee_secrets) =
|
|
self.append_fee_call(&tx, &tree, &fee_pk, &fee_zkbin, None).await?;
|
|
|
|
// Append the fee call to the transaction
|
|
tx_builder.append(ContractCallLeaf { call: fee_call, proofs: fee_proofs }, vec![])?;
|
|
|
|
// Now build the actual transaction and sign it with all necessary keys.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&[notes_secret_key])?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&fee_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
Ok(tx)
|
|
}
|
|
|
|
/// Create a DAO transfer proposal.
|
|
#[allow(clippy::too_many_arguments)]
|
|
pub async fn dao_propose_transfer(
|
|
&self,
|
|
name: &str,
|
|
duration_blockwindows: u64,
|
|
amount: &str,
|
|
token_id: TokenId,
|
|
recipient: PublicKey,
|
|
spend_hook: Option<FuncId>,
|
|
user_data: Option<pallas::Base>,
|
|
) -> Result<ProposalRecord> {
|
|
// Fetch DAO and check its deployed
|
|
let dao = self.get_dao_by_name(name).await?;
|
|
if dao.leaf_position.is_none() || dao.tx_hash.is_none() || dao.call_index.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_propose_transfer] DAO seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check that we have the proposer key
|
|
if dao.params.proposer_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_propose_transfer] We need the proposer secret key to create proposals for this DAO".to_string(),
|
|
))
|
|
}
|
|
|
|
// Fetch DAO unspent OwnCoins to see what its balance is
|
|
let dao_spend_hook =
|
|
FuncRef { contract_id: *DAO_CONTRACT_ID, func_code: DaoFunction::Exec as u8 }
|
|
.to_func_id();
|
|
let dao_bulla = dao.bulla();
|
|
let dao_owncoins =
|
|
self.get_contract_token_coins(&token_id, &dao_spend_hook, &dao_bulla.inner()).await?;
|
|
if dao_owncoins.is_empty() {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_propose_transfer] Did not find any {token_id} unspent coins owned by this DAO"
|
|
)))
|
|
}
|
|
|
|
// Check DAO balance is sufficient
|
|
let amount = decode_base10(amount, BALANCE_BASE10_DECIMALS, false)?;
|
|
if dao_owncoins.iter().map(|x| x.note.value).sum::<u64>() < amount {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_propose_transfer] Not enough DAO balance for token ID: {token_id}",
|
|
)))
|
|
}
|
|
|
|
// Generate proposal coin attributes
|
|
let proposal_coinattrs = CoinAttributes {
|
|
public_key: recipient,
|
|
value: amount,
|
|
token_id,
|
|
spend_hook: spend_hook.unwrap_or(FuncId::none()),
|
|
user_data: user_data.unwrap_or(pallas::Base::ZERO),
|
|
blind: Blind::random(&mut OsRng),
|
|
};
|
|
|
|
// Convert coin_params to actual coins
|
|
let proposal_coins = vec![proposal_coinattrs.to_coin()];
|
|
let mut proposal_data = vec![];
|
|
proposal_coins.encode_async(&mut proposal_data).await?;
|
|
|
|
// Create Auth calls
|
|
let auth_calls = vec![
|
|
DaoAuthCall {
|
|
contract_id: *DAO_CONTRACT_ID,
|
|
function_code: DaoFunction::AuthMoneyTransfer as u8,
|
|
auth_data: proposal_data,
|
|
},
|
|
DaoAuthCall {
|
|
contract_id: *MONEY_CONTRACT_ID,
|
|
function_code: MoneyFunction::TransferV1 as u8,
|
|
auth_data: vec![],
|
|
},
|
|
];
|
|
|
|
// Retrieve next block height and current block time target,
|
|
// to compute their window.
|
|
let next_block_height = self.get_next_block_height().await?;
|
|
let block_target = self.get_block_target().await?;
|
|
let creation_blockwindow = blockwindow(next_block_height, block_target);
|
|
|
|
// Create the actual proposal
|
|
let proposal = DaoProposal {
|
|
auth_calls,
|
|
creation_blockwindow,
|
|
duration_blockwindows,
|
|
user_data: user_data.unwrap_or(pallas::Base::ZERO),
|
|
dao_bulla,
|
|
blind: Blind::random(&mut OsRng),
|
|
};
|
|
|
|
let proposal_record = ProposalRecord {
|
|
proposal,
|
|
data: Some(serialize_async(&proposal_coinattrs).await),
|
|
leaf_position: None,
|
|
money_snapshot_tree: None,
|
|
nullifiers_smt_snapshot: None,
|
|
mint_height: None,
|
|
tx_hash: None,
|
|
call_index: None,
|
|
exec_height: None,
|
|
exec_tx_hash: None,
|
|
};
|
|
|
|
if let Err(e) = self.put_dao_proposal(&proposal_record).await {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[dao_propose_transfer] Put DAO proposal failed: {e}"
|
|
)))
|
|
}
|
|
|
|
Ok(proposal_record)
|
|
}
|
|
|
|
/// Create a DAO generic proposal.
|
|
pub async fn dao_propose_generic(
|
|
&self,
|
|
name: &str,
|
|
duration_blockwindows: u64,
|
|
user_data: Option<pallas::Base>,
|
|
) -> Result<ProposalRecord> {
|
|
// Fetch DAO and check its deployed
|
|
let dao = self.get_dao_by_name(name).await?;
|
|
if dao.leaf_position.is_none() || dao.tx_hash.is_none() || dao.call_index.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_propose_generic] DAO seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check that we have the proposer key
|
|
if dao.params.proposer_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_propose_generic] We need the proposer secret key to create proposals for this DAO".to_string(),
|
|
))
|
|
}
|
|
|
|
// Retrieve next block height and current block time target,
|
|
// to compute their window.
|
|
let next_block_height = self.get_next_block_height().await?;
|
|
let block_target = self.get_block_target().await?;
|
|
let creation_blockwindow = blockwindow(next_block_height, block_target);
|
|
|
|
// Create the actual proposal
|
|
let proposal = DaoProposal {
|
|
auth_calls: vec![],
|
|
creation_blockwindow,
|
|
duration_blockwindows,
|
|
user_data: user_data.unwrap_or(pallas::Base::ZERO),
|
|
dao_bulla: dao.bulla(),
|
|
blind: Blind::random(&mut OsRng),
|
|
};
|
|
|
|
let proposal_record = ProposalRecord {
|
|
proposal,
|
|
data: None,
|
|
leaf_position: None,
|
|
money_snapshot_tree: None,
|
|
nullifiers_smt_snapshot: None,
|
|
mint_height: None,
|
|
tx_hash: None,
|
|
call_index: None,
|
|
exec_height: None,
|
|
exec_tx_hash: None,
|
|
};
|
|
|
|
if let Err(e) = self.put_dao_proposal(&proposal_record).await {
|
|
return Err(Error::DatabaseError(format!(
|
|
"[dao_propose_generic] Put DAO proposal failed: {e}"
|
|
)))
|
|
}
|
|
|
|
Ok(proposal_record)
|
|
}
|
|
|
|
/// Create a DAO transfer proposal transaction.
|
|
pub async fn dao_transfer_proposal_tx(&self, proposal: &ProposalRecord) -> Result<Transaction> {
|
|
// Check we know the plaintext data
|
|
if proposal.data.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_transfer_proposal_tx] Proposal plainext data is empty".to_string(),
|
|
))
|
|
}
|
|
let proposal_coinattrs: CoinAttributes =
|
|
deserialize_async(proposal.data.as_ref().unwrap()).await?;
|
|
|
|
// Fetch DAO and check its deployed
|
|
let Ok(dao) = self.get_dao_by_bulla(&proposal.proposal.dao_bulla).await else {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_transfer_proposal_tx] DAO {} was not found",
|
|
proposal.proposal.dao_bulla
|
|
)))
|
|
};
|
|
if dao.leaf_position.is_none() || dao.tx_hash.is_none() || dao.call_index.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_transfer_proposal_tx] DAO seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check that we have the proposer key
|
|
if dao.params.proposer_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_transfer_proposal_tx] We need the proposer secret key to create proposals for this DAO".to_string(),
|
|
))
|
|
}
|
|
|
|
// Fetch DAO unspent OwnCoins to see what its balance is for the coin
|
|
let dao_spend_hook =
|
|
FuncRef { contract_id: *DAO_CONTRACT_ID, func_code: DaoFunction::Exec as u8 }
|
|
.to_func_id();
|
|
let dao_owncoins = self
|
|
.get_contract_token_coins(
|
|
&proposal_coinattrs.token_id,
|
|
&dao_spend_hook,
|
|
&proposal.proposal.dao_bulla.inner(),
|
|
)
|
|
.await?;
|
|
if dao_owncoins.is_empty() {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_transfer_proposal_tx] Did not find any {} unspent coins owned by this DAO",
|
|
proposal_coinattrs.token_id,
|
|
)))
|
|
}
|
|
|
|
// Check DAO balance is sufficient
|
|
if dao_owncoins.iter().map(|x| x.note.value).sum::<u64>() < proposal_coinattrs.value {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_transfer_proposal_tx] Not enough DAO balance for token ID: {}",
|
|
proposal_coinattrs.token_id,
|
|
)))
|
|
}
|
|
|
|
// Fetch our own governance OwnCoins to see what our balance is
|
|
let gov_owncoins = self.get_token_coins(&dao.params.dao.gov_token_id).await?;
|
|
if gov_owncoins.is_empty() {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_transfer_proposal_tx] Did not find any governance {} coins in wallet",
|
|
dao.params.dao.gov_token_id
|
|
)))
|
|
}
|
|
|
|
// Find which governance coins we can use
|
|
let mut total_value = 0;
|
|
let mut gov_owncoins_to_use = vec![];
|
|
for gov_owncoin in gov_owncoins {
|
|
if total_value >= dao.params.dao.proposer_limit {
|
|
break
|
|
}
|
|
|
|
total_value += gov_owncoin.note.value;
|
|
gov_owncoins_to_use.push(gov_owncoin);
|
|
}
|
|
|
|
// Check our governance coins balance is sufficient
|
|
if total_value < dao.params.dao.proposer_limit {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_transfer_proposal_tx] Not enough gov token {} balance to propose",
|
|
dao.params.dao.gov_token_id
|
|
)))
|
|
}
|
|
|
|
// Now we need to do a lookup for the zkas proof bincodes, and create
|
|
// the circuit objects and proving keys so we can build the transaction.
|
|
// We also do this through the RPC. First we grab the fee call from money.
|
|
let zkas_bins = self.lookup_zkas(&MONEY_CONTRACT_ID).await?;
|
|
|
|
let Some(fee_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_FEE_NS_V1)
|
|
else {
|
|
return Err(Error::Custom(
|
|
"[dao_transfer_proposal_tx] Fee circuit not found".to_string(),
|
|
))
|
|
};
|
|
|
|
let fee_zkbin = ZkBinary::decode(&fee_zkbin.1, false)?;
|
|
|
|
let fee_circuit = ZkCircuit::new(empty_witnesses(&fee_zkbin)?, &fee_zkbin);
|
|
|
|
// Creating Fee circuit proving key
|
|
let fee_pk = ProvingKey::build(fee_zkbin.k, &fee_circuit);
|
|
|
|
// Now we grab the DAO bins
|
|
let zkas_bins = self.lookup_zkas(&DAO_CONTRACT_ID).await?;
|
|
|
|
let Some(propose_burn_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_PROPOSE_INPUT_NS)
|
|
else {
|
|
return Err(Error::Custom(
|
|
"[dao_transfer_proposal_tx] Propose Burn circuit not found".to_string(),
|
|
))
|
|
};
|
|
|
|
let Some(propose_main_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_PROPOSE_MAIN_NS)
|
|
else {
|
|
return Err(Error::Custom(
|
|
"[dao_transfer_proposal_tx] Propose Main circuit not found".to_string(),
|
|
))
|
|
};
|
|
|
|
let propose_burn_zkbin = ZkBinary::decode(&propose_burn_zkbin.1, false)?;
|
|
let propose_main_zkbin = ZkBinary::decode(&propose_main_zkbin.1, false)?;
|
|
|
|
let propose_burn_circuit =
|
|
ZkCircuit::new(empty_witnesses(&propose_burn_zkbin)?, &propose_burn_zkbin);
|
|
let propose_main_circuit =
|
|
ZkCircuit::new(empty_witnesses(&propose_main_zkbin)?, &propose_main_zkbin);
|
|
|
|
// Creating DAO ProposeBurn and ProposeMain circuits proving keys
|
|
let propose_burn_pk = ProvingKey::build(propose_burn_zkbin.k, &propose_burn_circuit);
|
|
let propose_main_pk = ProvingKey::build(propose_main_zkbin.k, &propose_main_circuit);
|
|
|
|
// Fetch our money Merkle tree
|
|
let money_merkle_tree = self.get_money_tree().await?;
|
|
|
|
// Now we can create the proposal transaction parameters.
|
|
// We first generate the `DaoProposeStakeInput` inputs,
|
|
// using our governance OwnCoins.
|
|
let mut inputs = Vec::with_capacity(gov_owncoins_to_use.len());
|
|
for gov_owncoin in gov_owncoins_to_use {
|
|
let input = DaoProposeStakeInput {
|
|
secret: gov_owncoin.secret,
|
|
note: gov_owncoin.note.clone(),
|
|
leaf_position: gov_owncoin.leaf_position,
|
|
merkle_path: money_merkle_tree.witness(gov_owncoin.leaf_position, 0).unwrap(),
|
|
};
|
|
inputs.push(input);
|
|
}
|
|
|
|
// Now create the parameters for the proposal tx.
|
|
// Fetch the daos Merkle tree to compute the DAO Merkle path and root.
|
|
let (daos_tree, _) = self.get_dao_trees().await?;
|
|
let (dao_merkle_path, dao_merkle_root) = {
|
|
let root = daos_tree.root(0).unwrap();
|
|
let leaf_pos = dao.leaf_position.unwrap();
|
|
let dao_merkle_path = daos_tree.witness(leaf_pos, 0).unwrap();
|
|
(dao_merkle_path, root)
|
|
};
|
|
|
|
// Generate the Money nullifiers Sparse Merkle Tree
|
|
let store = CacheSmtStorage::new(CacheOverlay::new(&self.cache)?, SLED_MONEY_SMT_TREE);
|
|
let money_null_smt = CacheSmt::new(store, PoseidonFp::new(), &EMPTY_NODES_FP);
|
|
|
|
// Create the proposal call
|
|
let call = DaoProposeCall {
|
|
money_null_smt: &money_null_smt,
|
|
inputs,
|
|
proposal: proposal.proposal.clone(),
|
|
dao: dao.params.dao,
|
|
dao_leaf_position: dao.leaf_position.unwrap(),
|
|
dao_merkle_path,
|
|
dao_merkle_root,
|
|
};
|
|
|
|
let (params, proofs, signature_secrets) = call.make(
|
|
&dao.params.proposer_secret_key.unwrap(),
|
|
&propose_burn_zkbin,
|
|
&propose_burn_pk,
|
|
&propose_main_zkbin,
|
|
&propose_main_pk,
|
|
)?;
|
|
|
|
// Encode the call
|
|
let mut data = vec![DaoFunction::Propose as u8];
|
|
params.encode_async(&mut data).await?;
|
|
let call = ContractCall { contract_id: *DAO_CONTRACT_ID, data };
|
|
|
|
// Create the TransactionBuilder containing above call
|
|
let mut tx_builder = TransactionBuilder::new(ContractCallLeaf { call, proofs }, vec![])?;
|
|
|
|
// We first have to execute the fee-less tx to gather its used gas, and then we feed
|
|
// it into the fee-creating function.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&signature_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
let tree = self.get_money_tree().await?;
|
|
let (fee_call, fee_proofs, fee_secrets) =
|
|
self.append_fee_call(&tx, &tree, &fee_pk, &fee_zkbin, None).await?;
|
|
|
|
// Append the fee call to the transaction
|
|
tx_builder.append(ContractCallLeaf { call: fee_call, proofs: fee_proofs }, vec![])?;
|
|
|
|
// Now build the actual transaction and sign it with all necessary keys.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&signature_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&fee_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
Ok(tx)
|
|
}
|
|
|
|
/// Create a DAO generic proposal transaction.
|
|
pub async fn dao_generic_proposal_tx(&self, proposal: &ProposalRecord) -> Result<Transaction> {
|
|
// Fetch DAO and check its deployed
|
|
let Ok(dao) = self.get_dao_by_bulla(&proposal.proposal.dao_bulla).await else {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_generic_proposal_tx] DAO {} was not found",
|
|
proposal.proposal.dao_bulla
|
|
)))
|
|
};
|
|
if dao.leaf_position.is_none() || dao.tx_hash.is_none() || dao.call_index.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_generic_proposal_tx] DAO seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check that we have the proposer key
|
|
if dao.params.proposer_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_generic_proposal_tx] We need the proposer secret key to create proposals for this DAO".to_string(),
|
|
))
|
|
}
|
|
|
|
// Fetch our own governance OwnCoins to see what our balance is
|
|
let gov_owncoins = self.get_token_coins(&dao.params.dao.gov_token_id).await?;
|
|
if gov_owncoins.is_empty() {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_generic_proposal_tx] Did not find any governance {} coins in wallet",
|
|
dao.params.dao.gov_token_id
|
|
)))
|
|
}
|
|
|
|
// Find which governance coins we can use
|
|
let mut total_value = 0;
|
|
let mut gov_owncoins_to_use = vec![];
|
|
for gov_owncoin in gov_owncoins {
|
|
if total_value >= dao.params.dao.proposer_limit {
|
|
break
|
|
}
|
|
|
|
total_value += gov_owncoin.note.value;
|
|
gov_owncoins_to_use.push(gov_owncoin);
|
|
}
|
|
|
|
// Check our governance coins balance is sufficient
|
|
if total_value < dao.params.dao.proposer_limit {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_generic_proposal_tx] Not enough gov token {} balance to propose",
|
|
dao.params.dao.gov_token_id
|
|
)))
|
|
}
|
|
|
|
// Now we need to do a lookup for the zkas proof bincodes, and create
|
|
// the circuit objects and proving keys so we can build the transaction.
|
|
// We also do this through the RPC. First we grab the fee call from money.
|
|
let zkas_bins = self.lookup_zkas(&MONEY_CONTRACT_ID).await?;
|
|
|
|
let Some(fee_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_FEE_NS_V1)
|
|
else {
|
|
return Err(Error::Custom("[dao_generic_proposal_tx] Fee circuit not found".to_string()))
|
|
};
|
|
|
|
let fee_zkbin = ZkBinary::decode(&fee_zkbin.1, false)?;
|
|
|
|
let fee_circuit = ZkCircuit::new(empty_witnesses(&fee_zkbin)?, &fee_zkbin);
|
|
|
|
// Creating Fee circuit proving key
|
|
let fee_pk = ProvingKey::build(fee_zkbin.k, &fee_circuit);
|
|
|
|
// Now we grab the DAO bins
|
|
let zkas_bins = self.lookup_zkas(&DAO_CONTRACT_ID).await?;
|
|
|
|
let Some(propose_burn_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_PROPOSE_INPUT_NS)
|
|
else {
|
|
return Err(Error::Custom(
|
|
"[dao_generic_proposal_tx] Propose Burn circuit not found".to_string(),
|
|
))
|
|
};
|
|
|
|
let Some(propose_main_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_PROPOSE_MAIN_NS)
|
|
else {
|
|
return Err(Error::Custom(
|
|
"[dao_generic_proposal_tx] Propose Main circuit not found".to_string(),
|
|
))
|
|
};
|
|
|
|
let propose_burn_zkbin = ZkBinary::decode(&propose_burn_zkbin.1, false)?;
|
|
let propose_main_zkbin = ZkBinary::decode(&propose_main_zkbin.1, false)?;
|
|
|
|
let propose_burn_circuit =
|
|
ZkCircuit::new(empty_witnesses(&propose_burn_zkbin)?, &propose_burn_zkbin);
|
|
let propose_main_circuit =
|
|
ZkCircuit::new(empty_witnesses(&propose_main_zkbin)?, &propose_main_zkbin);
|
|
|
|
// Creating DAO ProposeBurn and ProposeMain circuits proving keys
|
|
let propose_burn_pk = ProvingKey::build(propose_burn_zkbin.k, &propose_burn_circuit);
|
|
let propose_main_pk = ProvingKey::build(propose_main_zkbin.k, &propose_main_circuit);
|
|
|
|
// Fetch our money Merkle tree
|
|
let money_merkle_tree = self.get_money_tree().await?;
|
|
|
|
// Now we can create the proposal transaction parameters.
|
|
// We first generate the `DaoProposeStakeInput` inputs,
|
|
// using our governance OwnCoins.
|
|
let mut inputs = Vec::with_capacity(gov_owncoins_to_use.len());
|
|
for gov_owncoin in gov_owncoins_to_use {
|
|
let input = DaoProposeStakeInput {
|
|
secret: gov_owncoin.secret,
|
|
note: gov_owncoin.note.clone(),
|
|
leaf_position: gov_owncoin.leaf_position,
|
|
merkle_path: money_merkle_tree.witness(gov_owncoin.leaf_position, 0).unwrap(),
|
|
};
|
|
inputs.push(input);
|
|
}
|
|
|
|
// Now create the parameters for the proposal tx.
|
|
// Fetch the daos Merkle tree to compute the DAO Merkle path and root.
|
|
let (daos_tree, _) = self.get_dao_trees().await?;
|
|
let (dao_merkle_path, dao_merkle_root) = {
|
|
let root = daos_tree.root(0).unwrap();
|
|
let leaf_pos = dao.leaf_position.unwrap();
|
|
let dao_merkle_path = daos_tree.witness(leaf_pos, 0).unwrap();
|
|
(dao_merkle_path, root)
|
|
};
|
|
|
|
// Generate the Money nullifiers Sparse Merkle Tree
|
|
let store = CacheSmtStorage::new(CacheOverlay::new(&self.cache)?, SLED_MONEY_SMT_TREE);
|
|
let money_null_smt = CacheSmt::new(store, PoseidonFp::new(), &EMPTY_NODES_FP);
|
|
|
|
// Create the proposal call
|
|
let call = DaoProposeCall {
|
|
money_null_smt: &money_null_smt,
|
|
inputs,
|
|
proposal: proposal.proposal.clone(),
|
|
dao: dao.params.dao,
|
|
dao_leaf_position: dao.leaf_position.unwrap(),
|
|
dao_merkle_path,
|
|
dao_merkle_root,
|
|
};
|
|
|
|
let (params, proofs, signature_secrets) = call.make(
|
|
&dao.params.proposer_secret_key.unwrap(),
|
|
&propose_burn_zkbin,
|
|
&propose_burn_pk,
|
|
&propose_main_zkbin,
|
|
&propose_main_pk,
|
|
)?;
|
|
|
|
// Encode the call
|
|
let mut data = vec![DaoFunction::Propose as u8];
|
|
params.encode_async(&mut data).await?;
|
|
let call = ContractCall { contract_id: *DAO_CONTRACT_ID, data };
|
|
|
|
// Create the TransactionBuilder containing above call
|
|
let mut tx_builder = TransactionBuilder::new(ContractCallLeaf { call, proofs }, vec![])?;
|
|
|
|
// We first have to execute the fee-less tx to gather its used gas, and then we feed
|
|
// it into the fee-creating function.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&signature_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
let tree = self.get_money_tree().await?;
|
|
let (fee_call, fee_proofs, fee_secrets) =
|
|
self.append_fee_call(&tx, &tree, &fee_pk, &fee_zkbin, None).await?;
|
|
|
|
// Append the fee call to the transaction
|
|
tx_builder.append(ContractCallLeaf { call: fee_call, proofs: fee_proofs }, vec![])?;
|
|
|
|
// Now build the actual transaction and sign it with all necessary keys.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&signature_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&fee_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
Ok(tx)
|
|
}
|
|
|
|
/// Vote on a DAO proposal
|
|
pub async fn dao_vote(
|
|
&self,
|
|
proposal_bulla: &DaoProposalBulla,
|
|
vote_option: bool,
|
|
weight: Option<u64>,
|
|
) -> Result<Transaction> {
|
|
// Feth the proposal and check its deployed
|
|
let Ok(proposal) = self.get_dao_proposal_by_bulla(proposal_bulla).await else {
|
|
return Err(Error::Custom(format!("[dao_vote] Proposal {proposal_bulla} was not found")))
|
|
};
|
|
if proposal.leaf_position.is_none() ||
|
|
proposal.money_snapshot_tree.is_none() ||
|
|
proposal.nullifiers_smt_snapshot.is_none() ||
|
|
proposal.tx_hash.is_none() ||
|
|
proposal.call_index.is_none()
|
|
{
|
|
return Err(Error::Custom(
|
|
"[dao_vote] Proposal seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check proposal is not executed
|
|
if let Some(exec_tx_hash) = proposal.exec_tx_hash {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_vote] Proposal was executed on transaction: {exec_tx_hash}"
|
|
)))
|
|
}
|
|
|
|
// Fetch DAO and check its deployed
|
|
let Ok(dao) = self.get_dao_by_bulla(&proposal.proposal.dao_bulla).await else {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_vote] DAO {} was not found",
|
|
proposal.proposal.dao_bulla
|
|
)))
|
|
};
|
|
if dao.leaf_position.is_none() || dao.tx_hash.is_none() || dao.call_index.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_vote] DAO seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Fetch all the proposal votes to check for duplicate nullifiers
|
|
let votes = self.get_dao_proposal_votes(proposal_bulla).await?;
|
|
let mut votes_nullifiers = vec![];
|
|
for vote in votes {
|
|
for nullifier in vote.nullifiers {
|
|
if !votes_nullifiers.contains(&nullifier) {
|
|
votes_nullifiers.push(nullifier);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fetch our own governance OwnCoins to see what our balance is
|
|
let gov_owncoins = self.get_token_coins(&dao.params.dao.gov_token_id).await?;
|
|
if gov_owncoins.is_empty() {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_vote] Did not find any governance {} coins in wallet",
|
|
dao.params.dao.gov_token_id
|
|
)))
|
|
}
|
|
|
|
// Find which governance coins we can use
|
|
let gov_owncoins_to_use = match weight {
|
|
Some(_weight) => {
|
|
// TODO: Build a proper coin selection algorithm so that we can use a
|
|
// coins combination that matches the requested weight
|
|
return Err(Error::Custom(
|
|
"[dao_vote] Fractional vote weight not supported yet".to_string(),
|
|
))
|
|
}
|
|
// If no weight was specified, use them all
|
|
None => gov_owncoins,
|
|
};
|
|
|
|
// Now we need to do a lookup for the zkas proof bincodes, and create
|
|
// the circuit objects and proving keys so we can build the transaction.
|
|
// We also do this through the RPC. First we grab the fee call from money.
|
|
let zkas_bins = self.lookup_zkas(&MONEY_CONTRACT_ID).await?;
|
|
|
|
let Some(fee_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_FEE_NS_V1)
|
|
else {
|
|
return Err(Error::Custom("[dao_vote] Fee circuit not found".to_string()))
|
|
};
|
|
|
|
let fee_zkbin = ZkBinary::decode(&fee_zkbin.1, false)?;
|
|
|
|
let fee_circuit = ZkCircuit::new(empty_witnesses(&fee_zkbin)?, &fee_zkbin);
|
|
|
|
// Creating Fee circuit proving key
|
|
let fee_pk = ProvingKey::build(fee_zkbin.k, &fee_circuit);
|
|
|
|
// Now we grab the DAO bins
|
|
let zkas_bins = self.lookup_zkas(&DAO_CONTRACT_ID).await?;
|
|
|
|
let Some(dao_vote_burn_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_VOTE_INPUT_NS)
|
|
else {
|
|
return Err(Error::Custom("[dao_vote] DAO Vote Burn circuit not found".to_string()))
|
|
};
|
|
|
|
let Some(dao_vote_main_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_VOTE_MAIN_NS)
|
|
else {
|
|
return Err(Error::Custom("[dao_vote] DAO Vote Main circuit not found".to_string()))
|
|
};
|
|
|
|
let dao_vote_burn_zkbin = ZkBinary::decode(&dao_vote_burn_zkbin.1, false)?;
|
|
let dao_vote_main_zkbin = ZkBinary::decode(&dao_vote_main_zkbin.1, false)?;
|
|
|
|
let dao_vote_burn_circuit =
|
|
ZkCircuit::new(empty_witnesses(&dao_vote_burn_zkbin)?, &dao_vote_burn_zkbin);
|
|
let dao_vote_main_circuit =
|
|
ZkCircuit::new(empty_witnesses(&dao_vote_main_zkbin)?, &dao_vote_main_zkbin);
|
|
|
|
// Creating DAO VoteBurn and VoteMain circuits proving keys
|
|
let dao_vote_burn_pk = ProvingKey::build(dao_vote_burn_zkbin.k, &dao_vote_burn_circuit);
|
|
let dao_vote_main_pk = ProvingKey::build(dao_vote_main_zkbin.k, &dao_vote_main_circuit);
|
|
|
|
// Now create the parameters for the vote tx
|
|
let mut inputs = Vec::with_capacity(gov_owncoins_to_use.len());
|
|
for gov_owncoin in gov_owncoins_to_use {
|
|
// Skip governance coins that are not part of the snapshot
|
|
let Ok(merkle_path) = proposal
|
|
.money_snapshot_tree
|
|
.as_ref()
|
|
.unwrap()
|
|
.witness(gov_owncoin.leaf_position, 0)
|
|
else {
|
|
continue
|
|
};
|
|
let nullifier = poseidon_hash([gov_owncoin.secret.inner(), gov_owncoin.coin.inner()]);
|
|
let vote_nullifier =
|
|
poseidon_hash([nullifier, gov_owncoin.secret.inner(), proposal_bulla.inner()]);
|
|
if votes_nullifiers.contains(&vote_nullifier.into()) {
|
|
return Err(Error::Custom("[dao_vote] Duplicate input nullifier found".to_string()))
|
|
};
|
|
|
|
let input = DaoVoteInput {
|
|
secret: gov_owncoin.secret,
|
|
note: gov_owncoin.note.clone(),
|
|
leaf_position: gov_owncoin.leaf_position,
|
|
merkle_path,
|
|
};
|
|
inputs.push(input);
|
|
}
|
|
if inputs.is_empty() {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_vote] Did not find any governance {} coins in wallet before proposal snapshot",
|
|
dao.params.dao.gov_token_id
|
|
)))
|
|
}
|
|
|
|
// Retrieve next block height and current block time target,
|
|
// to compute their window.
|
|
let next_block_height = self.get_next_block_height().await?;
|
|
let block_target = self.get_block_target().await?;
|
|
let current_blockwindow = blockwindow(next_block_height, block_target);
|
|
|
|
// Check if proposal has expired
|
|
if current_blockwindow >=
|
|
proposal.proposal.creation_blockwindow + proposal.proposal.duration_blockwindows
|
|
{
|
|
return Err(Error::Custom("[dao_vote] Proposal has expired".to_string()))
|
|
}
|
|
|
|
// Generate the Money nullifiers Sparse Merkle Tree
|
|
let store = MemoryStorageFp { tree: proposal.nullifiers_smt_snapshot.unwrap() };
|
|
let money_null_smt = SmtMemoryFp::new(store, PoseidonFp::new(), &EMPTY_NODES_FP);
|
|
|
|
// Create the vote call
|
|
let call = DaoVoteCall {
|
|
money_null_smt: &money_null_smt,
|
|
inputs,
|
|
vote_option,
|
|
proposal: proposal.proposal.clone(),
|
|
dao: dao.params.dao.clone(),
|
|
current_blockwindow,
|
|
};
|
|
|
|
let (params, proofs, signature_secrets) = call.make(
|
|
&dao_vote_burn_zkbin,
|
|
&dao_vote_burn_pk,
|
|
&dao_vote_main_zkbin,
|
|
&dao_vote_main_pk,
|
|
)?;
|
|
|
|
// Encode the call
|
|
let mut data = vec![DaoFunction::Vote as u8];
|
|
params.encode_async(&mut data).await?;
|
|
let call = ContractCall { contract_id: *DAO_CONTRACT_ID, data };
|
|
|
|
// Create the TransactionBuilder containing above call
|
|
let mut tx_builder = TransactionBuilder::new(ContractCallLeaf { call, proofs }, vec![])?;
|
|
|
|
// We first have to execute the fee-less tx to gather its used gas, and then we feed
|
|
// it into the fee-creating function.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&signature_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
let tree = self.get_money_tree().await?;
|
|
let (fee_call, fee_proofs, fee_secrets) =
|
|
self.append_fee_call(&tx, &tree, &fee_pk, &fee_zkbin, None).await?;
|
|
|
|
// Append the fee call to the transaction
|
|
tx_builder.append(ContractCallLeaf { call: fee_call, proofs: fee_proofs }, vec![])?;
|
|
|
|
// Now build the actual transaction and sign it with all necessary keys.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&signature_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&fee_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
Ok(tx)
|
|
}
|
|
|
|
/// Execute a DAO transfer proposal.
|
|
pub async fn dao_exec_transfer(
|
|
&self,
|
|
proposal: &ProposalRecord,
|
|
early: bool,
|
|
) -> Result<Transaction> {
|
|
if proposal.leaf_position.is_none() ||
|
|
proposal.money_snapshot_tree.is_none() ||
|
|
proposal.nullifiers_smt_snapshot.is_none() ||
|
|
proposal.tx_hash.is_none() ||
|
|
proposal.call_index.is_none()
|
|
{
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] Proposal seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check proposal is not executed
|
|
if let Some(exec_tx_hash) = proposal.exec_tx_hash {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_transfer] Proposal was executed on transaction: {exec_tx_hash}"
|
|
)))
|
|
}
|
|
|
|
// Check we know the plaintext data and they are valid
|
|
if proposal.data.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] Proposal plainext data is empty".to_string(),
|
|
))
|
|
}
|
|
let proposal_coinattrs: CoinAttributes =
|
|
deserialize_async(proposal.data.as_ref().unwrap()).await?;
|
|
|
|
// Fetch DAO and check its deployed
|
|
let Ok(dao) = self.get_dao_by_bulla(&proposal.proposal.dao_bulla).await else {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_transfer] DAO {} was not found",
|
|
proposal.proposal.dao_bulla
|
|
)))
|
|
};
|
|
if dao.leaf_position.is_none() || dao.tx_hash.is_none() || dao.call_index.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] DAO seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check that we have the exec key
|
|
if dao.params.exec_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] We need the exec secret key to execute proposals for this DAO"
|
|
.to_string(),
|
|
))
|
|
}
|
|
|
|
// If early flag is provided, check that we have the early exec key
|
|
if early && dao.params.early_exec_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] We need the early exec secret key to execute proposals early for this DAO"
|
|
.to_string(),
|
|
))
|
|
}
|
|
|
|
// Check proposal is approved
|
|
let votes = self.get_dao_proposal_votes(&proposal.bulla()).await?;
|
|
let mut yes_vote_value = 0;
|
|
let mut yes_vote_blind = Blind::ZERO;
|
|
let mut all_vote_value = 0;
|
|
let mut all_vote_blind = Blind::ZERO;
|
|
for vote in votes {
|
|
if vote.vote_option {
|
|
yes_vote_value += vote.all_vote_value;
|
|
};
|
|
yes_vote_blind += vote.yes_vote_blind;
|
|
all_vote_value += vote.all_vote_value;
|
|
all_vote_blind += vote.all_vote_blind;
|
|
}
|
|
let approval_ratio = (yes_vote_value as f64 * 100.0) / all_vote_value as f64;
|
|
let dao_approval_ratio =
|
|
(dao.params.dao.approval_ratio_quot / dao.params.dao.approval_ratio_base) as f64;
|
|
if approval_ratio < dao_approval_ratio {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] Proposal is not approved yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check the quorum has been met
|
|
if all_vote_value < dao.params.dao.quorum {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] DAO quorum is not reached yet".to_string(),
|
|
))
|
|
}
|
|
if early && all_vote_value < dao.params.dao.early_exec_quorum {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] DAO early execution quorum is not reached yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Fetch DAO unspent OwnCoins to see what its balance is for the coin
|
|
let dao_spend_hook =
|
|
FuncRef { contract_id: *DAO_CONTRACT_ID, func_code: DaoFunction::Exec as u8 }
|
|
.to_func_id();
|
|
let dao_owncoins = self
|
|
.get_contract_token_coins(
|
|
&proposal_coinattrs.token_id,
|
|
&dao_spend_hook,
|
|
&proposal.proposal.dao_bulla.inner(),
|
|
)
|
|
.await?;
|
|
if dao_owncoins.is_empty() {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_transfer] Did not find any {} unspent coins owned by this DAO",
|
|
proposal_coinattrs.token_id,
|
|
)))
|
|
}
|
|
|
|
// Check DAO balance is sufficient
|
|
if dao_owncoins.iter().map(|x| x.note.value).sum::<u64>() < proposal_coinattrs.value {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_transfer] Not enough DAO balance for token ID: {}",
|
|
proposal_coinattrs.token_id,
|
|
)))
|
|
}
|
|
|
|
// Find which DAO coins we can use
|
|
let (spent_coins, change_value) = select_coins(dao_owncoins, proposal_coinattrs.value)?;
|
|
|
|
// Now we need to do a lookup for the zkas proof bincodes, and create
|
|
// the circuit objects and proving keys so we can build the transaction.
|
|
// We also do this through the RPC. First we grab the calls from money.
|
|
let zkas_bins = self.lookup_zkas(&MONEY_CONTRACT_ID).await?;
|
|
|
|
let Some(mint_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_MINT_NS_V1)
|
|
else {
|
|
return Err(Error::Custom("[dao_exec_transfer] Mint circuit not found".to_string()))
|
|
};
|
|
|
|
let Some(burn_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_BURN_NS_V1)
|
|
else {
|
|
return Err(Error::Custom("[dao_exec_transfer] Burn circuit not found".to_string()))
|
|
};
|
|
|
|
let Some(fee_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_FEE_NS_V1)
|
|
else {
|
|
return Err(Error::Custom("[dao_exec_transfer] Fee circuit not found".to_string()))
|
|
};
|
|
|
|
let mint_zkbin = ZkBinary::decode(&mint_zkbin.1, false)?;
|
|
let burn_zkbin = ZkBinary::decode(&burn_zkbin.1, false)?;
|
|
let fee_zkbin = ZkBinary::decode(&fee_zkbin.1, false)?;
|
|
|
|
let mint_circuit = ZkCircuit::new(empty_witnesses(&mint_zkbin)?, &mint_zkbin);
|
|
let burn_circuit = ZkCircuit::new(empty_witnesses(&burn_zkbin)?, &burn_zkbin);
|
|
let fee_circuit = ZkCircuit::new(empty_witnesses(&fee_zkbin)?, &fee_zkbin);
|
|
|
|
// Creating Mint, Burn and Fee circuits proving keys
|
|
let mint_pk = ProvingKey::build(mint_zkbin.k, &mint_circuit);
|
|
let burn_pk = ProvingKey::build(burn_zkbin.k, &burn_circuit);
|
|
let fee_pk = ProvingKey::build(fee_zkbin.k, &fee_circuit);
|
|
|
|
// Now we grab the DAO bins
|
|
let zkas_bins = self.lookup_zkas(&DAO_CONTRACT_ID).await?;
|
|
|
|
let (namespace, early_exec_secret_key) = match early {
|
|
true => {
|
|
(DAO_CONTRACT_ZKAS_EARLY_EXEC_NS, Some(dao.params.early_exec_secret_key.unwrap()))
|
|
}
|
|
false => (DAO_CONTRACT_ZKAS_EXEC_NS, None),
|
|
};
|
|
|
|
let Some(dao_exec_zkbin) = zkas_bins.iter().find(|x| x.0 == namespace) else {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_transfer] DAO {namespace} circuit not found"
|
|
)))
|
|
};
|
|
|
|
let Some(dao_auth_transfer_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_AUTH_MONEY_TRANSFER_NS)
|
|
else {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] DAO AuthTransfer circuit not found".to_string(),
|
|
))
|
|
};
|
|
|
|
let Some(dao_auth_transfer_enc_coin_zkbin) =
|
|
zkas_bins.iter().find(|x| x.0 == DAO_CONTRACT_ZKAS_AUTH_MONEY_TRANSFER_ENC_COIN_NS)
|
|
else {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] DAO AuthTransferEncCoin circuit not found".to_string(),
|
|
))
|
|
};
|
|
|
|
let dao_exec_zkbin = ZkBinary::decode(&dao_exec_zkbin.1, false)?;
|
|
let dao_auth_transfer_zkbin = ZkBinary::decode(&dao_auth_transfer_zkbin.1, false)?;
|
|
let dao_auth_transfer_enc_coin_zkbin =
|
|
ZkBinary::decode(&dao_auth_transfer_enc_coin_zkbin.1, false)?;
|
|
|
|
let dao_exec_circuit = ZkCircuit::new(empty_witnesses(&dao_exec_zkbin)?, &dao_exec_zkbin);
|
|
let dao_auth_transfer_circuit =
|
|
ZkCircuit::new(empty_witnesses(&dao_auth_transfer_zkbin)?, &dao_auth_transfer_zkbin);
|
|
let dao_auth_transfer_enc_coin_circuit = ZkCircuit::new(
|
|
empty_witnesses(&dao_auth_transfer_enc_coin_zkbin)?,
|
|
&dao_auth_transfer_enc_coin_zkbin,
|
|
);
|
|
|
|
// Creating DAO Exec, AuthTransfer and AuthTransferEncCoin circuits proving keys
|
|
let dao_exec_pk = ProvingKey::build(dao_exec_zkbin.k, &dao_exec_circuit);
|
|
let dao_auth_transfer_pk =
|
|
ProvingKey::build(dao_auth_transfer_zkbin.k, &dao_auth_transfer_circuit);
|
|
let dao_auth_transfer_enc_coin_pk = ProvingKey::build(
|
|
dao_auth_transfer_enc_coin_zkbin.k,
|
|
&dao_auth_transfer_enc_coin_circuit,
|
|
);
|
|
|
|
// Fetch our money Merkle tree
|
|
let tree = self.get_money_tree().await?;
|
|
|
|
// Retrieve next block height and current block time target,
|
|
// to compute their window.
|
|
let next_block_height = self.get_next_block_height().await?;
|
|
let block_target = self.get_block_target().await?;
|
|
let current_blockwindow = blockwindow(next_block_height, block_target);
|
|
|
|
// Check if proposal duration has passed
|
|
if !early &&
|
|
current_blockwindow <
|
|
proposal.proposal.creation_blockwindow +
|
|
proposal.proposal.duration_blockwindows
|
|
{
|
|
return Err(Error::Custom(
|
|
"[dao_exec_transfer] Proposal period has not passed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Now we can create the transfer call parameters
|
|
let input_user_data_blind = Blind::random(&mut OsRng);
|
|
let mut inputs = vec![];
|
|
for coin in &spent_coins {
|
|
inputs.push(TransferCallInput {
|
|
coin: coin.clone(),
|
|
merkle_path: tree.witness(coin.leaf_position, 0).unwrap(),
|
|
user_data_blind: input_user_data_blind,
|
|
});
|
|
}
|
|
|
|
let mut outputs = vec![];
|
|
outputs.push(proposal_coinattrs.clone());
|
|
|
|
let dao_coin_attrs = CoinAttributes {
|
|
public_key: dao.params.dao.notes_public_key,
|
|
value: change_value,
|
|
token_id: proposal_coinattrs.token_id,
|
|
spend_hook: dao_spend_hook,
|
|
user_data: proposal.proposal.dao_bulla.inner(),
|
|
blind: Blind::random(&mut OsRng),
|
|
};
|
|
outputs.push(dao_coin_attrs.clone());
|
|
|
|
// Create the transfer call
|
|
let transfer_builder = TransferCallBuilder {
|
|
clear_inputs: vec![],
|
|
inputs,
|
|
outputs,
|
|
mint_zkbin: mint_zkbin.clone(),
|
|
mint_pk: mint_pk.clone(),
|
|
burn_zkbin: burn_zkbin.clone(),
|
|
burn_pk: burn_pk.clone(),
|
|
};
|
|
let (transfer_params, transfer_secrets) = transfer_builder.build()?;
|
|
|
|
// Encode the call
|
|
let mut data = vec![MoneyFunction::TransferV1 as u8];
|
|
transfer_params.encode_async(&mut data).await?;
|
|
let transfer_call = ContractCall { contract_id: *MONEY_CONTRACT_ID, data };
|
|
|
|
// Create the exec call
|
|
let exec_signature_secret = SecretKey::random(&mut OsRng);
|
|
let exec_builder = DaoExecCall {
|
|
proposal: proposal.proposal.clone(),
|
|
dao: dao.params.dao.clone(),
|
|
yes_vote_value,
|
|
all_vote_value,
|
|
yes_vote_blind,
|
|
all_vote_blind,
|
|
signature_secret: exec_signature_secret,
|
|
current_blockwindow,
|
|
};
|
|
let (exec_params, exec_proofs) = exec_builder.make(
|
|
&dao.params.exec_secret_key.unwrap(),
|
|
&early_exec_secret_key,
|
|
&dao_exec_zkbin,
|
|
&dao_exec_pk,
|
|
)?;
|
|
|
|
// Encode the call
|
|
let mut data = vec![DaoFunction::Exec as u8];
|
|
exec_params.encode_async(&mut data).await?;
|
|
let exec_call = ContractCall { contract_id: *DAO_CONTRACT_ID, data };
|
|
|
|
// Now we can create the auth call
|
|
// Auth module
|
|
let auth_transfer_builder = DaoAuthMoneyTransferCall {
|
|
proposal: proposal.proposal.clone(),
|
|
proposal_coinattrs: vec![proposal_coinattrs],
|
|
dao: dao.params.dao.clone(),
|
|
input_user_data_blind,
|
|
dao_coin_attrs,
|
|
};
|
|
let (auth_transfer_params, auth_transfer_proofs) = auth_transfer_builder.make(
|
|
&dao_auth_transfer_zkbin,
|
|
&dao_auth_transfer_pk,
|
|
&dao_auth_transfer_enc_coin_zkbin,
|
|
&dao_auth_transfer_enc_coin_pk,
|
|
)?;
|
|
|
|
// Encode the call
|
|
let mut data = vec![DaoFunction::AuthMoneyTransfer as u8];
|
|
auth_transfer_params.encode_async(&mut data).await?;
|
|
let auth_transfer_call = ContractCall { contract_id: *DAO_CONTRACT_ID, data };
|
|
|
|
// Create the TransactionBuilder containing above calls
|
|
let mut tx_builder = TransactionBuilder::new(
|
|
ContractCallLeaf { call: exec_call, proofs: exec_proofs },
|
|
vec![
|
|
DarkTree::new(
|
|
ContractCallLeaf { call: auth_transfer_call, proofs: auth_transfer_proofs },
|
|
vec![],
|
|
None,
|
|
None,
|
|
),
|
|
DarkTree::new(
|
|
ContractCallLeaf { call: transfer_call, proofs: transfer_secrets.proofs },
|
|
vec![],
|
|
None,
|
|
None,
|
|
),
|
|
],
|
|
)?;
|
|
|
|
// We first have to execute the fee-less tx to gather its used gas, and then we feed
|
|
// it into the fee-creating function.
|
|
let mut tx = tx_builder.build()?;
|
|
let auth_transfer_sigs = tx.create_sigs(&[])?;
|
|
let transfer_sigs = tx.create_sigs(&transfer_secrets.signature_secrets)?;
|
|
let exec_sigs = tx.create_sigs(&[exec_signature_secret])?;
|
|
tx.signatures = vec![auth_transfer_sigs, transfer_sigs, exec_sigs];
|
|
|
|
let (fee_call, fee_proofs, fee_secrets) =
|
|
self.append_fee_call(&tx, &tree, &fee_pk, &fee_zkbin, None).await?;
|
|
|
|
// Append the fee call to the transaction
|
|
tx_builder.append(ContractCallLeaf { call: fee_call, proofs: fee_proofs }, vec![])?;
|
|
|
|
// Now build the actual transaction and sign it with all necessary keys.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&[])?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&transfer_secrets.signature_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&[exec_signature_secret])?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&fee_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
Ok(tx)
|
|
}
|
|
|
|
/// Execute a DAO generic proposal.
|
|
pub async fn dao_exec_generic(
|
|
&self,
|
|
proposal: &ProposalRecord,
|
|
early: bool,
|
|
) -> Result<Transaction> {
|
|
if proposal.leaf_position.is_none() ||
|
|
proposal.money_snapshot_tree.is_none() ||
|
|
proposal.nullifiers_smt_snapshot.is_none() ||
|
|
proposal.tx_hash.is_none() ||
|
|
proposal.call_index.is_none()
|
|
{
|
|
return Err(Error::Custom(
|
|
"[dao_exec_generic] Proposal seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check proposal is not executed
|
|
if let Some(exec_tx_hash) = proposal.exec_tx_hash {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_generic] Proposal was executed on transaction: {exec_tx_hash}"
|
|
)))
|
|
}
|
|
|
|
// Fetch DAO and check its deployed
|
|
let Ok(dao) = self.get_dao_by_bulla(&proposal.proposal.dao_bulla).await else {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_generic] DAO {} was not found",
|
|
proposal.proposal.dao_bulla
|
|
)))
|
|
};
|
|
if dao.leaf_position.is_none() || dao.tx_hash.is_none() || dao.call_index.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_generic] DAO seems to not have been deployed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Check that we have the exec key
|
|
if dao.params.exec_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_generic] We need the exec secret key to execute proposals for this DAO"
|
|
.to_string(),
|
|
))
|
|
}
|
|
|
|
// If early flag is provided, check that we have the early exec key
|
|
if early && dao.params.early_exec_secret_key.is_none() {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_generic] We need the early exec secret key to execute proposals early for this DAO"
|
|
.to_string(),
|
|
))
|
|
}
|
|
|
|
// Check proposal is approved
|
|
let votes = self.get_dao_proposal_votes(&proposal.bulla()).await?;
|
|
let mut yes_vote_value = 0;
|
|
let mut yes_vote_blind = Blind::ZERO;
|
|
let mut all_vote_value = 0;
|
|
let mut all_vote_blind = Blind::ZERO;
|
|
for vote in votes {
|
|
if vote.vote_option {
|
|
yes_vote_value += vote.all_vote_value;
|
|
};
|
|
yes_vote_blind += vote.yes_vote_blind;
|
|
all_vote_value += vote.all_vote_value;
|
|
all_vote_blind += vote.all_vote_blind;
|
|
}
|
|
let approval_ratio = (yes_vote_value as f64 * 100.0) / all_vote_value as f64;
|
|
let dao_approval_ratio =
|
|
(dao.params.dao.approval_ratio_quot / dao.params.dao.approval_ratio_base) as f64;
|
|
if approval_ratio < dao_approval_ratio {
|
|
return Err(Error::Custom("[dao_exec_generic] Proposal is not approved yet".to_string()))
|
|
}
|
|
|
|
// Check the quorum has been met
|
|
if all_vote_value < dao.params.dao.quorum {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_generic] DAO quorum is not reached yet".to_string(),
|
|
))
|
|
}
|
|
if early && all_vote_value < dao.params.dao.early_exec_quorum {
|
|
return Err(Error::Custom(
|
|
"[dao_exec_generic] DAO early execution quorum is not reached yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Now we need to do a lookup for the zkas proof bincodes, and create
|
|
// the circuit objects and proving keys so we can build the transaction.
|
|
// We also do this through the RPC. First we grab the calls from money.
|
|
let zkas_bins = self.lookup_zkas(&MONEY_CONTRACT_ID).await?;
|
|
let Some(fee_zkbin) = zkas_bins.iter().find(|x| x.0 == MONEY_CONTRACT_ZKAS_FEE_NS_V1)
|
|
else {
|
|
return Err(Error::Custom("[dao_exec_generic] Fee circuit not found".to_string()))
|
|
};
|
|
let fee_zkbin = ZkBinary::decode(&fee_zkbin.1, false)?;
|
|
let fee_circuit = ZkCircuit::new(empty_witnesses(&fee_zkbin)?, &fee_zkbin);
|
|
let fee_pk = ProvingKey::build(fee_zkbin.k, &fee_circuit);
|
|
|
|
// Now we grab the DAO bins
|
|
let zkas_bins = self.lookup_zkas(&DAO_CONTRACT_ID).await?;
|
|
|
|
let (namespace, early_exec_secret_key) = match early {
|
|
true => {
|
|
(DAO_CONTRACT_ZKAS_EARLY_EXEC_NS, Some(dao.params.early_exec_secret_key.unwrap()))
|
|
}
|
|
false => (DAO_CONTRACT_ZKAS_EXEC_NS, None),
|
|
};
|
|
|
|
let Some(dao_exec_zkbin) = zkas_bins.iter().find(|x| x.0 == namespace) else {
|
|
return Err(Error::Custom(format!(
|
|
"[dao_exec_generic] DAO {namespace} circuit not found"
|
|
)))
|
|
};
|
|
let dao_exec_zkbin = ZkBinary::decode(&dao_exec_zkbin.1, false)?;
|
|
let dao_exec_circuit = ZkCircuit::new(empty_witnesses(&dao_exec_zkbin)?, &dao_exec_zkbin);
|
|
let dao_exec_pk = ProvingKey::build(dao_exec_zkbin.k, &dao_exec_circuit);
|
|
|
|
// Fetch our money Merkle tree
|
|
let tree = self.get_money_tree().await?;
|
|
|
|
// Retrieve next block height and current block time target,
|
|
// to compute their window.
|
|
let next_block_height = self.get_next_block_height().await?;
|
|
let block_target = self.get_block_target().await?;
|
|
let current_blockwindow = blockwindow(next_block_height, block_target);
|
|
|
|
// Check if proposal duration has passed
|
|
if !early &&
|
|
current_blockwindow <
|
|
proposal.proposal.creation_blockwindow +
|
|
proposal.proposal.duration_blockwindows
|
|
{
|
|
return Err(Error::Custom(
|
|
"[dao_exec_generic] Proposal period has not passed yet".to_string(),
|
|
))
|
|
}
|
|
|
|
// Create the exec call
|
|
let exec_signature_secret = SecretKey::random(&mut OsRng);
|
|
let exec_builder = DaoExecCall {
|
|
proposal: proposal.proposal.clone(),
|
|
dao: dao.params.dao.clone(),
|
|
yes_vote_value,
|
|
all_vote_value,
|
|
yes_vote_blind,
|
|
all_vote_blind,
|
|
signature_secret: exec_signature_secret,
|
|
current_blockwindow,
|
|
};
|
|
let (exec_params, exec_proofs) = exec_builder.make(
|
|
&dao.params.exec_secret_key.unwrap(),
|
|
&early_exec_secret_key,
|
|
&dao_exec_zkbin,
|
|
&dao_exec_pk,
|
|
)?;
|
|
|
|
// Encode the call
|
|
let mut data = vec![DaoFunction::Exec as u8];
|
|
exec_params.encode_async(&mut data).await?;
|
|
let exec_call = ContractCall { contract_id: *DAO_CONTRACT_ID, data };
|
|
|
|
// Create the TransactionBuilder containing above calls
|
|
let mut tx_builder = TransactionBuilder::new(
|
|
ContractCallLeaf { call: exec_call, proofs: exec_proofs },
|
|
vec![],
|
|
)?;
|
|
|
|
// We first have to execute the fee-less tx to gather its used gas, and then we feed
|
|
// it into the fee-creating function.
|
|
let mut tx = tx_builder.build()?;
|
|
let exec_sigs = tx.create_sigs(&[exec_signature_secret])?;
|
|
tx.signatures = vec![exec_sigs];
|
|
|
|
let (fee_call, fee_proofs, fee_secrets) =
|
|
self.append_fee_call(&tx, &tree, &fee_pk, &fee_zkbin, None).await?;
|
|
|
|
// Append the fee call to the transaction
|
|
tx_builder.append(ContractCallLeaf { call: fee_call, proofs: fee_proofs }, vec![])?;
|
|
|
|
// Now build the actual transaction and sign it with all necessary keys.
|
|
let mut tx = tx_builder.build()?;
|
|
let sigs = tx.create_sigs(&[exec_signature_secret])?;
|
|
tx.signatures.push(sigs);
|
|
let sigs = tx.create_sigs(&fee_secrets)?;
|
|
tx.signatures.push(sigs);
|
|
|
|
Ok(tx)
|
|
}
|
|
}
|