mirror of
https://github.com/mimblewimble/grin.git
synced 2025-05-10 19:21:16 +03:00
* pass slices around and not refs to vecs * use slice.swap() * use inputs() not body.inputs
311 lines
9.1 KiB
Rust
311 lines
9.1 KiB
Rust
// Copyright 2020 The Grin Developers
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
//! Build a block to mine: gathers transactions from the pool, assembles
|
|
//! them into a block and returns it.
|
|
|
|
use chrono::prelude::{DateTime, NaiveDateTime, Utc};
|
|
use rand::{thread_rng, Rng};
|
|
use serde_json::{json, Value};
|
|
use std::sync::Arc;
|
|
use std::thread;
|
|
use std::time::Duration;
|
|
|
|
use crate::api;
|
|
use crate::chain;
|
|
use crate::common::types::Error;
|
|
use crate::core::core::{Output, TxKernel};
|
|
use crate::core::libtx::secp_ser;
|
|
use crate::core::libtx::ProofBuilder;
|
|
use crate::core::{consensus, core, global};
|
|
use crate::keychain::{ExtKeychain, Identifier, Keychain};
|
|
use crate::{ServerTxPool, ServerVerifierCache};
|
|
|
|
/// Fees in block to use for coinbase amount calculation
|
|
/// (Duplicated from Grin wallet project)
|
|
#[derive(Serialize, Deserialize, Debug, Clone)]
|
|
pub struct BlockFees {
|
|
/// fees
|
|
#[serde(with = "secp_ser::string_or_u64")]
|
|
pub fees: u64,
|
|
/// height
|
|
#[serde(with = "secp_ser::string_or_u64")]
|
|
pub height: u64,
|
|
/// key id
|
|
pub key_id: Option<Identifier>,
|
|
}
|
|
|
|
impl BlockFees {
|
|
/// return key id
|
|
pub fn key_id(&self) -> Option<Identifier> {
|
|
self.key_id.clone()
|
|
}
|
|
}
|
|
|
|
/// Response to build a coinbase output.
|
|
#[derive(Serialize, Deserialize, Debug, Clone)]
|
|
pub struct CbData {
|
|
/// Output
|
|
pub output: Output,
|
|
/// Kernel
|
|
pub kernel: TxKernel,
|
|
/// Key Id
|
|
pub key_id: Option<Identifier>,
|
|
}
|
|
|
|
// Ensure a block suitable for mining is built and returned
|
|
// If a wallet listener URL is not provided the reward will be "burnt"
|
|
// Warning: This call does not return until/unless a new block can be built
|
|
pub fn get_block(
|
|
chain: &Arc<chain::Chain>,
|
|
tx_pool: &ServerTxPool,
|
|
verifier_cache: ServerVerifierCache,
|
|
key_id: Option<Identifier>,
|
|
wallet_listener_url: Option<String>,
|
|
) -> (core::Block, BlockFees) {
|
|
let wallet_retry_interval = 5;
|
|
// get the latest chain state and build a block on top of it
|
|
let mut result = build_block(
|
|
chain,
|
|
tx_pool,
|
|
verifier_cache.clone(),
|
|
key_id.clone(),
|
|
wallet_listener_url.clone(),
|
|
);
|
|
while let Err(e) = result {
|
|
let mut new_key_id = key_id.to_owned();
|
|
match e {
|
|
self::Error::Chain(c) => match c.kind() {
|
|
chain::ErrorKind::DuplicateCommitment(_) => {
|
|
debug!(
|
|
"Duplicate commit for potential coinbase detected. Trying next derivation."
|
|
);
|
|
// use the next available key to generate a different coinbase commitment
|
|
new_key_id = None;
|
|
}
|
|
_ => {
|
|
error!("Chain Error: {}", c);
|
|
}
|
|
},
|
|
self::Error::WalletComm(_) => {
|
|
error!(
|
|
"Error building new block: Can't connect to wallet listener at {:?}; will retry",
|
|
wallet_listener_url.as_ref().unwrap()
|
|
);
|
|
thread::sleep(Duration::from_secs(wallet_retry_interval));
|
|
}
|
|
ae => {
|
|
warn!("Error building new block: {:?}. Retrying.", ae);
|
|
}
|
|
}
|
|
|
|
// only wait if we are still using the same key: a different coinbase commitment is unlikely
|
|
// to have duplication
|
|
if new_key_id.is_some() {
|
|
thread::sleep(Duration::from_millis(100));
|
|
}
|
|
|
|
result = build_block(
|
|
chain,
|
|
tx_pool,
|
|
verifier_cache.clone(),
|
|
new_key_id,
|
|
wallet_listener_url.clone(),
|
|
);
|
|
}
|
|
return result.unwrap();
|
|
}
|
|
|
|
/// Builds a new block with the chain head as previous and eligible
|
|
/// transactions from the pool.
|
|
fn build_block(
|
|
chain: &Arc<chain::Chain>,
|
|
tx_pool: &ServerTxPool,
|
|
verifier_cache: ServerVerifierCache,
|
|
key_id: Option<Identifier>,
|
|
wallet_listener_url: Option<String>,
|
|
) -> Result<(core::Block, BlockFees), Error> {
|
|
let head = chain.head_header()?;
|
|
|
|
// prepare the block header timestamp
|
|
let mut now_sec = Utc::now().timestamp();
|
|
let head_sec = head.timestamp.timestamp();
|
|
if now_sec <= head_sec {
|
|
now_sec = head_sec + 1;
|
|
}
|
|
|
|
// Determine the difficulty our block should be at.
|
|
// Note: do not keep the difficulty_iter in scope (it has an active batch).
|
|
let difficulty = consensus::next_difficulty(head.height + 1, chain.difficulty_iter()?);
|
|
|
|
// Extract current "mineable" transactions from the pool.
|
|
// If this fails for *any* reason then fallback to an empty vec of txs.
|
|
// This will allow us to mine an "empty" block if the txpool is in an
|
|
// invalid (and unexpected) state.
|
|
let txs = match tx_pool.read().prepare_mineable_transactions() {
|
|
Ok(txs) => txs,
|
|
Err(e) => {
|
|
error!(
|
|
"build_block: Failed to prepare mineable txs from txpool: {:?}",
|
|
e
|
|
);
|
|
warn!("build_block: Falling back to mining empty block.");
|
|
vec![]
|
|
}
|
|
};
|
|
|
|
// build the coinbase and the block itself
|
|
let fees = txs.iter().map(|tx| tx.fee()).sum();
|
|
let height = head.height + 1;
|
|
let block_fees = BlockFees {
|
|
fees,
|
|
key_id,
|
|
height,
|
|
};
|
|
|
|
let (output, kernel, block_fees) = get_coinbase(wallet_listener_url, block_fees)?;
|
|
let mut b = core::Block::from_reward(&head, &txs, output, kernel, difficulty.difficulty)?;
|
|
|
|
// making sure we're not spending time mining a useless block
|
|
b.validate(&head.total_kernel_offset, verifier_cache)?;
|
|
|
|
b.header.pow.nonce = thread_rng().gen();
|
|
b.header.pow.secondary_scaling = difficulty.secondary_scaling;
|
|
b.header.timestamp = DateTime::<Utc>::from_utc(NaiveDateTime::from_timestamp(now_sec, 0), Utc);
|
|
|
|
debug!(
|
|
"Built new block with {} inputs and {} outputs, block difficulty: {}, cumulative difficulty {}",
|
|
b.inputs().len(),
|
|
b.outputs().len(),
|
|
difficulty.difficulty,
|
|
b.header.total_difficulty().to_num(),
|
|
);
|
|
|
|
// Now set txhashset roots and sizes on the header of the block being built.
|
|
match chain.set_txhashset_roots(&mut b) {
|
|
Ok(_) => Ok((b, block_fees)),
|
|
Err(e) => {
|
|
match e.kind() {
|
|
// If this is a duplicate commitment then likely trying to use
|
|
// a key that hass already been derived but not in the wallet
|
|
// for some reason, allow caller to retry.
|
|
chain::ErrorKind::DuplicateCommitment(e) => Err(Error::Chain(
|
|
chain::ErrorKind::DuplicateCommitment(e).into(),
|
|
)),
|
|
|
|
// Some other issue, possibly duplicate kernel
|
|
_ => {
|
|
error!("Error setting txhashset root to build a block: {:?}", e);
|
|
Err(Error::Chain(
|
|
chain::ErrorKind::Other(format!("{:?}", e)).into(),
|
|
))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
///
|
|
/// Probably only want to do this when testing.
|
|
///
|
|
fn burn_reward(block_fees: BlockFees) -> Result<(core::Output, core::TxKernel, BlockFees), Error> {
|
|
warn!("Burning block fees: {:?}", block_fees);
|
|
let keychain = ExtKeychain::from_random_seed(global::is_floonet())?;
|
|
let key_id = ExtKeychain::derive_key_id(1, 1, 0, 0, 0);
|
|
let (out, kernel) = crate::core::libtx::reward::output(
|
|
&keychain,
|
|
&ProofBuilder::new(&keychain),
|
|
&key_id,
|
|
block_fees.fees,
|
|
false,
|
|
)
|
|
.unwrap();
|
|
Ok((out, kernel, block_fees))
|
|
}
|
|
|
|
// Connect to the wallet listener and get coinbase.
|
|
// Warning: If a wallet listener URL is not provided the reward will be "burnt"
|
|
fn get_coinbase(
|
|
wallet_listener_url: Option<String>,
|
|
block_fees: BlockFees,
|
|
) -> Result<(core::Output, core::TxKernel, BlockFees), Error> {
|
|
match wallet_listener_url {
|
|
None => {
|
|
// Burn it
|
|
return burn_reward(block_fees);
|
|
}
|
|
Some(wallet_listener_url) => {
|
|
let res = create_coinbase(&wallet_listener_url, &block_fees)?;
|
|
let output = res.output;
|
|
let kernel = res.kernel;
|
|
let key_id = res.key_id;
|
|
let block_fees = BlockFees {
|
|
key_id: key_id,
|
|
..block_fees
|
|
};
|
|
|
|
debug!("get_coinbase: {:?}", block_fees);
|
|
return Ok((output, kernel, block_fees));
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Call the wallet API to create a coinbase output for the given block_fees.
|
|
/// Will retry based on default "retry forever with backoff" behavior.
|
|
fn create_coinbase(dest: &str, block_fees: &BlockFees) -> Result<CbData, Error> {
|
|
let url = format!("{}/v2/foreign", dest);
|
|
let req_body = json!({
|
|
"jsonrpc": "2.0",
|
|
"method": "build_coinbase",
|
|
"id": 1,
|
|
"params": {
|
|
"block_fees": block_fees
|
|
}
|
|
});
|
|
|
|
trace!("Sending build_coinbase request: {}", req_body);
|
|
let req = api::client::create_post_request(url.as_str(), None, &req_body)?;
|
|
let res: String = api::client::send_request(req).map_err(|e| {
|
|
let report = format!(
|
|
"Failed to get coinbase from {}. Is the wallet listening? {}",
|
|
dest, e
|
|
);
|
|
error!("{}", report);
|
|
Error::WalletComm(report)
|
|
})?;
|
|
|
|
let res: Value = serde_json::from_str(&res).unwrap();
|
|
trace!("Response: {}", res);
|
|
if res["error"] != json!(null) {
|
|
let report = format!(
|
|
"Failed to get coinbase from {}: Error: {}, Message: {}",
|
|
dest, res["error"]["code"], res["error"]["message"]
|
|
);
|
|
error!("{}", report);
|
|
return Err(Error::WalletComm(report));
|
|
}
|
|
|
|
let cb_data = res["result"]["Ok"].clone();
|
|
trace!("cb_data: {}", cb_data);
|
|
let ret_val = match serde_json::from_value::<CbData>(cb_data) {
|
|
Ok(r) => r,
|
|
Err(e) => {
|
|
let report = format!("Couldn't deserialize CbData: {}", e);
|
|
error!("{}", report);
|
|
return Err(Error::WalletComm(report));
|
|
}
|
|
};
|
|
|
|
Ok(ret_val)
|
|
}
|