-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathmod.rs
153 lines (142 loc) · 4.52 KB
/
mod.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
//! Building and submitting of the transaction that changes the current governance authority
//!
//! The transaction:
//! 1. Burns the previous governance UTXO from the version oracle validator address
//! 2. Mints exactly 1 multi-sig policy token as authentication
//! 3. Produces a new governance UTXO at the version oracle validator address with a version oracle
//! Plutus datum attached that contains the script ID (32) and policy hash.
use crate::csl::{Costs, TransactionExt};
use crate::governance::{GovernanceData, SimpleAtLeastN};
use crate::{
await_tx::AwaitTx,
cardano_keys::CardanoPaymentSigningKey,
csl::CostStore,
csl::{InputsBuilderExt, TransactionBuilderExt, TransactionContext},
init_governance::transaction::version_oracle_datum_output,
plutus_script::PlutusScript,
};
use anyhow::anyhow;
use cardano_serialization_lib::{
Language, PlutusData, Transaction, TransactionBuilder, TxInputsBuilder,
};
use ogmios_client::{
query_ledger_state::{QueryLedgerState, QueryUtxoByUtxoId},
query_network::QueryNetwork,
transactions::Transactions,
};
use serde_json::json;
use sidechain_domain::{
MainchainKeyHash, McSmartContractResult,
McSmartContractResult::{TxCBOR, TxHash},
McTxHash, UtxoId, UtxoIndex,
};
#[cfg(test)]
mod test;
#[cfg(test)]
mod test_values;
pub async fn run_update_governance<
T: QueryLedgerState + Transactions + QueryNetwork + QueryUtxoByUtxoId,
A: AwaitTx,
>(
new_governance_authority: &Vec<MainchainKeyHash>,
new_governance_threshold: u8,
payment_key: &CardanoPaymentSigningKey,
genesis_utxo_id: UtxoId,
client: &T,
await_tx: A,
) -> anyhow::Result<McSmartContractResult> {
let ctx = TransactionContext::for_payment_key(payment_key, client).await?;
let governance_data = GovernanceData::get(genesis_utxo_id, client).await?;
let tx = Costs::calculate_costs(
|costs| {
update_governance_tx(
raw_scripts::VERSION_ORACLE_VALIDATOR,
raw_scripts::VERSION_ORACLE_POLICY,
genesis_utxo_id,
new_governance_authority,
new_governance_threshold,
&governance_data,
costs,
&ctx,
)
},
client,
)
.await?;
if governance_data.policy.is_single_key_policy_for(&ctx.payment_key_hash()) {
let signed_tx = ctx.sign(&tx).to_bytes();
let res = client.submit_transaction(&signed_tx).await.map_err(|e| {
anyhow!(
"Submit governance update transaction request failed: {}, bytes: {}",
e,
hex::encode(signed_tx)
)
})?;
let tx_id = McTxHash(res.transaction.id);
log::info!("Update Governance transaction submitted: {}", hex::encode(tx_id.0));
await_tx
.await_tx_output(
client,
UtxoId { tx_hash: McTxHash(res.transaction.id), index: UtxoIndex(0) },
)
.await?;
Ok(TxHash(tx_id))
} else {
let tx_envelope = json!(
{ "type": "Unwitnessed Tx ConwayEra",
"description": "",
"cborHex": hex::encode(tx.to_bytes())
}
);
log::info!("Transaction envelope: {}", tx_envelope);
Ok(TxCBOR(tx.to_bytes()))
}
}
fn update_governance_tx(
version_oracle_validator: &[u8],
version_oracle_policy: &[u8],
genesis_utxo: UtxoId,
new_governance_authority: &Vec<MainchainKeyHash>,
new_governance_threshold: u8,
governance_data: &GovernanceData,
costs: Costs,
ctx: &TransactionContext,
) -> anyhow::Result<Transaction> {
let multi_sig_policy = SimpleAtLeastN {
threshold: new_governance_threshold.into(),
key_hashes: new_governance_authority.iter().map(|key| key.0).collect(),
}
.to_csl_native_script();
let version_oracle_validator =
PlutusScript::from_wrapped_cbor(version_oracle_validator, Language::new_plutus_v2())?
.apply_data(genesis_utxo)?;
let version_oracle_policy =
PlutusScript::from_wrapped_cbor(version_oracle_policy, Language::new_plutus_v2())?
.apply_data(genesis_utxo)?
.apply_uplc_data(version_oracle_validator.address_data(ctx.network)?)?;
let config = crate::csl::get_builder_config(ctx)?;
let mut tx_builder = TransactionBuilder::new(&config);
tx_builder.add_mint_one_script_token_using_reference_script(
&governance_data.policy.script(),
&governance_data.utxo_id_as_tx_input(),
&costs,
)?;
tx_builder.add_output(&version_oracle_datum_output(
version_oracle_validator.clone(),
version_oracle_policy.clone(),
multi_sig_policy,
ctx.network,
ctx,
)?)?;
tx_builder.set_inputs(&{
let mut inputs = TxInputsBuilder::new();
inputs.add_script_utxo_input(
&governance_data.utxo,
&version_oracle_validator,
&PlutusData::new_integer(&(raw_scripts::ScriptId::GovernancePolicy as u32).into()),
&costs.get_one_spend(),
)?;
inputs
});
Ok(tx_builder.balance_update_and_build(ctx)?.remove_native_script_witnesses())
}