Files
darkfi/example/smart-contract/tests/runtime.rs
2022-11-06 11:35:26 +01:00

101 lines
4.0 KiB
Rust

/* This file is part of DarkFi (https://dark.fi)
*
* Copyright (C) 2020-2022 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 darkfi::{
blockchain::Blockchain,
consensus::{TESTNET_GENESIS_HASH_BYTES, TESTNET_GENESIS_TIMESTAMP},
runtime::vm_runtime::Runtime,
Result,
};
use darkfi_sdk::{crypto::ContractId, pasta::pallas, tx::FuncCall};
use darkfi_serial::{serialize, Decodable, Encodable, WriteExt};
use std::io::Cursor;
use smart_contract::{FooCallData, Function};
#[test]
fn run_contract() -> Result<()> {
// Debug log configuration
let mut cfg = simplelog::ConfigBuilder::new();
cfg.add_filter_ignore("sled".to_string());
simplelog::TermLogger::init(
simplelog::LevelFilter::Debug,
cfg.build(),
simplelog::TerminalMode::Mixed,
simplelog::ColorChoice::Auto,
)?;
// =============================
// Initialize a dummy blockchain
// =============================
// TODO: This blockchain interface should perhaps be ValidatorState and Mutex/RwLock.
let db = sled::Config::new().temporary(true).open()?;
let blockchain = Blockchain::new(&db, *TESTNET_GENESIS_TIMESTAMP, *TESTNET_GENESIS_HASH_BYTES)?;
// ================================================================
// Load the wasm binary into memory and create an execution runtime
// ================================================================
let wasm_bytes = std::fs::read("contract.wasm")?;
let contract_id = ContractId::from(pallas::Base::from(1));
let mut runtime = Runtime::new(&wasm_bytes, blockchain.clone(), contract_id)?;
// Deploy function to initialize the smart contract state.
// Here we pass an empty payload, but it's possible to feed in arbitrary data.
runtime.deploy(&[])?;
// This is another call so we instantiate a new runtime.
let mut runtime = Runtime::new(&wasm_bytes, blockchain, contract_id)?;
// =============================================
// Build some kind of payload to show an example
// =============================================
let func_calls = vec![FuncCall {
contract_id: pallas::Base::from(110),
func_id: pallas::Base::from(4),
call_data: serialize(&FooCallData { a: 777, b: 666 }),
}];
let func_call_index: u32 = 0;
let mut payload = Vec::new();
// Selects which path executes in the contract.
payload.write_u8(Function::Foo as u8)?;
// Write the actual payload data
payload.write_u32(func_call_index)?;
func_calls.encode(&mut payload)?;
// ============================================================
// Serialize the payload into the runtime format and execute it
// ============================================================
let update = runtime.exec(&payload)?;
// =====================================================
// If exec was successful, try to apply the state change
// =====================================================
runtime.apply(&update)?;
// =====================================================
// Verify ZK proofs and signatures
// =====================================================
let metadata = runtime.metadata(&payload)?;
let mut decoder = Cursor::new(&metadata);
let zk_public_values: Vec<(String, Vec<pallas::Base>)> = Decodable::decode(&mut decoder)?;
let signature_public_keys: Vec<pallas::Point> = Decodable::decode(decoder)?;
Ok(())
}