2018-03-02 23:47:27 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2017-06-01 01:52:43 +03:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
|
|
|
//! Utilities to check the status of all the outputs we have stored in
|
|
|
|
//! the wallet storage and update them.
|
|
|
|
|
2018-03-04 03:19:54 +03:00
|
|
|
use failure::ResultExt;
|
2018-05-24 03:14:34 +03:00
|
|
|
use std::collections::hash_map::Entry;
|
2018-05-30 23:57:13 +03:00
|
|
|
use std::collections::HashMap;
|
2017-10-25 20:57:48 +03:00
|
|
|
|
2017-06-01 01:52:43 +03:00
|
|
|
use api;
|
2018-05-30 19:48:32 +03:00
|
|
|
use keychain::Identifier;
|
|
|
|
use libwallet::types::*;
|
2017-06-01 01:52:43 +03:00
|
|
|
use util;
|
2018-05-24 03:14:34 +03:00
|
|
|
use util::secp::pedersen;
|
2018-05-30 23:57:13 +03:00
|
|
|
use util::LOGGER;
|
2017-06-01 01:52:43 +03:00
|
|
|
|
2018-05-30 19:48:32 +03:00
|
|
|
pub fn refresh_outputs<T>(wallet: &mut T) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: WalletBackend,
|
|
|
|
{
|
|
|
|
let tip = get_tip_from_node(&wallet.node_url())?;
|
|
|
|
refresh_output_state(wallet, &tip)?;
|
|
|
|
refresh_missing_block_hashes(wallet, &tip)?;
|
2018-01-17 06:03:40 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
2017-10-25 20:57:48 +03:00
|
|
|
|
2018-03-04 03:19:54 +03:00
|
|
|
// TODO - this might be slow if we have really old outputs that have never been
|
|
|
|
// refreshed
|
2018-05-30 19:48:32 +03:00
|
|
|
fn refresh_missing_block_hashes<T>(wallet: &mut T, tip: &api::Tip) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: WalletBackend,
|
|
|
|
{
|
2018-01-17 06:03:40 +03:00
|
|
|
// build a local map of wallet outputs keyed by commit
|
2017-11-10 18:12:15 +03:00
|
|
|
// and a list of outputs we want to query the node for
|
2018-05-30 19:48:32 +03:00
|
|
|
let wallet_outputs = map_wallet_outputs_missing_block(wallet)?;
|
2017-10-25 20:57:48 +03:00
|
|
|
|
2018-01-17 06:03:40 +03:00
|
|
|
// nothing to do so return (otherwise we hit the api with a monster query...)
|
|
|
|
if wallet_outputs.is_empty() {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
debug!(
|
|
|
|
LOGGER,
|
|
|
|
"Refreshing missing block hashes (and heights) for {} outputs",
|
|
|
|
wallet_outputs.len(),
|
|
|
|
);
|
|
|
|
|
2018-04-27 16:16:30 +03:00
|
|
|
let id_params: Vec<String> = wallet_outputs
|
2018-01-17 06:03:40 +03:00
|
|
|
.keys()
|
2018-05-24 03:14:34 +03:00
|
|
|
.map(|commit| format!("id={}", util::to_hex(commit.as_ref().to_vec())))
|
2017-10-26 00:09:34 +03:00
|
|
|
.collect();
|
2017-10-25 20:57:48 +03:00
|
|
|
|
2018-05-24 03:14:34 +03:00
|
|
|
let height_params = [format!("start_height={}&end_height={}", 0, tip.height)];
|
2018-01-17 06:03:40 +03:00
|
|
|
|
|
|
|
let mut api_blocks: HashMap<pedersen::Commitment, api::BlockHeaderInfo> = HashMap::new();
|
2018-03-02 23:47:27 +03:00
|
|
|
let mut api_merkle_proofs: HashMap<pedersen::Commitment, MerkleProofWrapper> = HashMap::new();
|
2018-04-27 16:16:30 +03:00
|
|
|
|
|
|
|
// Split up into separate requests, to avoid hitting http limits
|
2018-05-24 03:14:34 +03:00
|
|
|
for mut query_chunk in id_params.chunks(1000) {
|
2018-04-27 16:16:30 +03:00
|
|
|
let url = format!(
|
|
|
|
"{}/v1/chain/outputs/byheight?{}",
|
2018-05-30 19:48:32 +03:00
|
|
|
wallet.node_url(),
|
2018-05-24 03:14:34 +03:00
|
|
|
[&height_params, query_chunk].concat().join("&"),
|
2018-04-27 16:16:30 +03:00
|
|
|
);
|
|
|
|
|
|
|
|
match api::client::get::<Vec<api::BlockOutputs>>(url.as_str()) {
|
|
|
|
Ok(blocks) => for block in blocks {
|
|
|
|
for out in block.outputs {
|
|
|
|
api_blocks.insert(out.commit, block.header.clone());
|
|
|
|
if let Some(merkle_proof) = out.merkle_proof {
|
|
|
|
let wrapper = MerkleProofWrapper(merkle_proof);
|
|
|
|
api_merkle_proofs.insert(out.commit, wrapper);
|
|
|
|
}
|
2018-01-17 06:03:40 +03:00
|
|
|
}
|
2018-04-27 16:16:30 +03:00
|
|
|
},
|
|
|
|
Err(e) => {
|
|
|
|
// if we got anything other than 200 back from server, bye
|
|
|
|
return Err(e).context(ErrorKind::Node)?;
|
2018-01-06 00:03:53 +03:00
|
|
|
}
|
2018-01-17 06:03:40 +03:00
|
|
|
}
|
2018-01-06 00:03:53 +03:00
|
|
|
}
|
2017-10-25 20:57:48 +03:00
|
|
|
|
2017-10-26 00:09:34 +03:00
|
|
|
// now for each commit, find the output in the wallet and
|
2018-01-06 00:03:53 +03:00
|
|
|
// the corresponding api output (if it exists)
|
|
|
|
// and refresh it in-place in the wallet.
|
|
|
|
// Note: minimizing the time we spend holding the wallet lock.
|
2018-05-30 19:48:32 +03:00
|
|
|
wallet.with_wallet(|wallet_data| {
|
2018-01-17 06:03:40 +03:00
|
|
|
for commit in wallet_outputs.keys() {
|
2018-01-06 00:03:53 +03:00
|
|
|
let id = wallet_outputs.get(&commit).unwrap();
|
2018-05-30 19:48:32 +03:00
|
|
|
if let Entry::Occupied(mut output) = wallet_data.outputs().entry(id.to_hex()) {
|
2018-01-17 06:03:40 +03:00
|
|
|
if let Some(b) = api_blocks.get(&commit) {
|
|
|
|
let output = output.get_mut();
|
2018-03-02 23:47:27 +03:00
|
|
|
output.block = Some(BlockIdentifier::from_hex(&b.hash).unwrap());
|
2018-01-17 06:03:40 +03:00
|
|
|
output.height = b.height;
|
2018-03-02 23:47:27 +03:00
|
|
|
if let Some(merkle_proof) = api_merkle_proofs.get(&commit) {
|
|
|
|
output.merkle_proof = Some(merkle_proof.clone());
|
|
|
|
}
|
2018-01-17 06:03:40 +03:00
|
|
|
}
|
2018-01-06 00:03:53 +03:00
|
|
|
}
|
2017-06-01 01:52:43 +03:00
|
|
|
}
|
2017-09-22 19:44:12 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-05-16 15:18:09 +03:00
|
|
|
/// build a local map of wallet outputs keyed by commit
|
|
|
|
/// and a list of outputs we want to query the node for
|
2018-05-30 19:48:32 +03:00
|
|
|
pub fn map_wallet_outputs<T>(
|
|
|
|
wallet: &mut T,
|
|
|
|
) -> Result<HashMap<pedersen::Commitment, Identifier>, Error>
|
|
|
|
where
|
|
|
|
T: WalletBackend,
|
|
|
|
{
|
2018-01-17 06:03:40 +03:00
|
|
|
let mut wallet_outputs: HashMap<pedersen::Commitment, Identifier> = HashMap::new();
|
2018-05-30 19:48:32 +03:00
|
|
|
let _ = wallet.read_wallet(|wallet_data| {
|
|
|
|
let keychain = wallet_data.keychain().clone();
|
|
|
|
let root_key_id = keychain.root_key_id().clone();
|
2018-05-24 03:14:34 +03:00
|
|
|
let unspents = wallet_data
|
2018-05-30 19:48:32 +03:00
|
|
|
.outputs()
|
2018-05-24 03:14:34 +03:00
|
|
|
.values()
|
2018-05-30 19:48:32 +03:00
|
|
|
.filter(|x| x.root_key_id == root_key_id && x.status != OutputStatus::Spent);
|
2018-05-24 03:14:34 +03:00
|
|
|
for out in unspents {
|
|
|
|
let commit = keychain
|
|
|
|
.commit_with_key_index(out.value, out.n_child)
|
|
|
|
.context(ErrorKind::Keychain)?;
|
|
|
|
wallet_outputs.insert(commit, out.key_id.clone());
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
});
|
2018-05-16 15:18:09 +03:00
|
|
|
Ok(wallet_outputs)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// As above, but only return unspent outputs with missing block hashes
|
|
|
|
/// and a list of outputs we want to query the node for
|
2018-05-30 19:48:32 +03:00
|
|
|
pub fn map_wallet_outputs_missing_block<T>(
|
|
|
|
wallet: &mut T,
|
|
|
|
) -> Result<HashMap<pedersen::Commitment, Identifier>, Error>
|
|
|
|
where
|
|
|
|
T: WalletBackend,
|
|
|
|
{
|
2018-05-16 15:18:09 +03:00
|
|
|
let mut wallet_outputs: HashMap<pedersen::Commitment, Identifier> = HashMap::new();
|
2018-05-30 19:48:32 +03:00
|
|
|
let _ = wallet.read_wallet(|wallet_data| {
|
|
|
|
let keychain = wallet_data.keychain().clone();
|
|
|
|
for out in wallet_data.outputs().clone().values().filter(|x| {
|
|
|
|
x.root_key_id == wallet_data.keychain().root_key_id() && x.block.is_none()
|
2018-05-16 15:18:09 +03:00
|
|
|
&& x.status == OutputStatus::Unspent
|
|
|
|
}) {
|
|
|
|
let commit = keychain
|
|
|
|
.commit_with_key_index(out.value, out.n_child)
|
|
|
|
.context(ErrorKind::Keychain)?;
|
|
|
|
wallet_outputs.insert(commit, out.key_id.clone());
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
});
|
|
|
|
Ok(wallet_outputs)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Apply refreshed API output data to the wallet
|
2018-05-30 19:48:32 +03:00
|
|
|
pub fn apply_api_outputs<T>(
|
|
|
|
wallet: &mut T,
|
2018-05-16 15:18:09 +03:00
|
|
|
wallet_outputs: &HashMap<pedersen::Commitment, Identifier>,
|
|
|
|
api_outputs: &HashMap<pedersen::Commitment, api::Output>,
|
2018-05-30 19:48:32 +03:00
|
|
|
) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: WalletBackend,
|
|
|
|
{
|
2018-05-24 03:14:34 +03:00
|
|
|
// now for each commit, find the output in the wallet and the corresponding
|
|
|
|
// api output (if it exists) and refresh it in-place in the wallet.
|
2018-05-16 15:18:09 +03:00
|
|
|
// Note: minimizing the time we spend holding the wallet lock.
|
2018-05-30 19:48:32 +03:00
|
|
|
wallet.with_wallet(|wallet_data| {
|
2018-05-16 15:18:09 +03:00
|
|
|
for commit in wallet_outputs.keys() {
|
|
|
|
let id = wallet_outputs.get(&commit).unwrap();
|
2018-05-30 19:48:32 +03:00
|
|
|
if let Entry::Occupied(mut output) = wallet_data.outputs().entry(id.to_hex()) {
|
2018-05-16 15:18:09 +03:00
|
|
|
match api_outputs.get(&commit) {
|
2018-05-22 23:39:32 +03:00
|
|
|
Some(_) => output.get_mut().mark_unspent(),
|
|
|
|
None => output.get_mut().mark_spent(),
|
2018-05-16 15:18:09 +03:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Builds a single api query to retrieve the latest output data from the node.
|
|
|
|
/// So we can refresh the local wallet outputs.
|
2018-05-30 19:48:32 +03:00
|
|
|
fn refresh_output_state<T>(wallet: &mut T, tip: &api::Tip) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: WalletBackend,
|
|
|
|
{
|
2018-05-16 15:18:09 +03:00
|
|
|
debug!(LOGGER, "Refreshing wallet outputs");
|
2018-01-17 06:03:40 +03:00
|
|
|
|
2018-05-16 15:18:09 +03:00
|
|
|
// build a local map of wallet outputs keyed by commit
|
|
|
|
// and a list of outputs we want to query the node for
|
2018-05-30 19:48:32 +03:00
|
|
|
let wallet_outputs = map_wallet_outputs(wallet)?;
|
2018-05-24 03:14:34 +03:00
|
|
|
|
2018-01-17 06:03:40 +03:00
|
|
|
// build the necessary query params -
|
|
|
|
// ?id=xxx&id=yyy&id=zzz
|
|
|
|
let query_params: Vec<String> = wallet_outputs
|
|
|
|
.keys()
|
2018-05-24 03:14:34 +03:00
|
|
|
.map(|commit| format!("id={}", util::to_hex(commit.as_ref().to_vec())))
|
2018-01-17 06:03:40 +03:00
|
|
|
.collect();
|
|
|
|
|
|
|
|
// build a map of api outputs by commit so we can look them up efficiently
|
2018-03-05 22:33:44 +03:00
|
|
|
let mut api_outputs: HashMap<pedersen::Commitment, api::Output> = HashMap::new();
|
2018-01-17 06:03:40 +03:00
|
|
|
|
2018-05-24 03:14:34 +03:00
|
|
|
for query_chunk in query_params.chunks(1000) {
|
2018-04-27 16:16:30 +03:00
|
|
|
let url = format!(
|
|
|
|
"{}/v1/chain/outputs/byids?{}",
|
2018-05-30 19:48:32 +03:00
|
|
|
wallet.node_url(),
|
2018-05-24 03:14:34 +03:00
|
|
|
query_chunk.join("&"),
|
2018-04-27 16:16:30 +03:00
|
|
|
);
|
|
|
|
|
|
|
|
match api::client::get::<Vec<api::Output>>(url.as_str()) {
|
|
|
|
Ok(outputs) => for out in outputs {
|
|
|
|
api_outputs.insert(out.commit.commit(), out);
|
|
|
|
},
|
|
|
|
Err(e) => {
|
|
|
|
// if we got anything other than 200 back from server, don't attempt to refresh
|
|
|
|
// the wallet data after
|
|
|
|
return Err(e).context(ErrorKind::Node)?;
|
|
|
|
}
|
2018-05-24 03:14:34 +03:00
|
|
|
}
|
2018-04-27 16:16:30 +03:00
|
|
|
}
|
2018-05-24 03:14:34 +03:00
|
|
|
|
2018-05-30 19:48:32 +03:00
|
|
|
apply_api_outputs(wallet, &wallet_outputs, &api_outputs)?;
|
|
|
|
clean_old_unconfirmed(wallet, tip)?;
|
2018-05-16 15:18:09 +03:00
|
|
|
Ok(())
|
2018-01-17 06:03:40 +03:00
|
|
|
}
|
|
|
|
|
2018-05-30 19:48:32 +03:00
|
|
|
fn clean_old_unconfirmed<T>(wallet: &mut T, tip: &api::Tip) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: WalletBackend,
|
|
|
|
{
|
2018-05-24 18:27:26 +03:00
|
|
|
if tip.height < 500 {
|
|
|
|
return Ok(());
|
|
|
|
}
|
2018-05-30 19:48:32 +03:00
|
|
|
wallet.with_wallet(|wallet_data| {
|
|
|
|
wallet_data.outputs().retain(|_, ref mut out| {
|
2018-05-24 03:14:34 +03:00
|
|
|
!(out.status == OutputStatus::Unconfirmed && out.height > 0
|
|
|
|
&& out.height < tip.height - 500)
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-05-30 19:48:32 +03:00
|
|
|
pub fn get_tip_from_node(addr: &str) -> Result<api::Tip, Error> {
|
|
|
|
let url = format!("{}/v1/chain", addr);
|
2018-03-04 03:19:54 +03:00
|
|
|
api::client::get::<api::Tip>(url.as_str())
|
|
|
|
.context(ErrorKind::Node)
|
|
|
|
.map_err(|e| e.into())
|
2017-06-01 01:52:43 +03:00
|
|
|
}
|