mirror of
https://github.com/mimblewimble/grin.git
synced 2025-02-01 17:01:09 +03:00
added wallet info/outputs commands with pretty printing (#254)
* added wallet info/outputs commands with pretty printing * added confirmed but locked to display output
This commit is contained in:
parent
2238495d23
commit
b831192335
5 changed files with 155 additions and 30 deletions
|
@ -229,8 +229,11 @@ fn main() {
|
||||||
.default_value("1")
|
.default_value("1")
|
||||||
.takes_value(true)))
|
.takes_value(true)))
|
||||||
|
|
||||||
|
.subcommand(SubCommand::with_name("outputs")
|
||||||
|
.about("raw wallet info (list of outputs)"))
|
||||||
|
|
||||||
.subcommand(SubCommand::with_name("info")
|
.subcommand(SubCommand::with_name("info")
|
||||||
.about("basic wallet info (outputs)"))
|
.about("basic wallet contents summary"))
|
||||||
|
|
||||||
.subcommand(SubCommand::with_name("init")
|
.subcommand(SubCommand::with_name("init")
|
||||||
.about("Initialize a new wallet seed file.")))
|
.about("Initialize a new wallet seed file.")))
|
||||||
|
@ -431,6 +434,9 @@ fn wallet_command(wallet_args: &ArgMatches) {
|
||||||
("info", Some(_)) => {
|
("info", Some(_)) => {
|
||||||
wallet::show_info(&wallet_config, &keychain);
|
wallet::show_info(&wallet_config, &keychain);
|
||||||
}
|
}
|
||||||
|
("outputs", Some(_)) => {
|
||||||
|
wallet::show_outputs(&wallet_config, &keychain);
|
||||||
|
}
|
||||||
_ => panic!("Unknown wallet command, use 'grin help wallet' for details"),
|
_ => panic!("Unknown wallet command, use 'grin help wallet' for details"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,8 @@ hyper = "~0.11.4"
|
||||||
tokio-core="~0.1.1"
|
tokio-core="~0.1.1"
|
||||||
tokio-retry="~0.1.0"
|
tokio-retry="~0.1.0"
|
||||||
router = "~0.5.1"
|
router = "~0.5.1"
|
||||||
|
prettytable-rs = "^0.6"
|
||||||
|
term = "~0.4.6"
|
||||||
grin_api = { path = "../api" }
|
grin_api = { path = "../api" }
|
||||||
grin_core = { path = "../core" }
|
grin_core = { path = "../core" }
|
||||||
grin_keychain = { path = "../keychain" }
|
grin_keychain = { path = "../keychain" }
|
||||||
|
|
|
@ -14,17 +14,17 @@
|
||||||
|
|
||||||
use checker;
|
use checker;
|
||||||
use keychain::Keychain;
|
use keychain::Keychain;
|
||||||
use core::core;
|
use core::core::amount_to_hr_string;
|
||||||
use types::{WalletConfig, WalletData};
|
use types::{WalletConfig, WalletData, OutputStatus};
|
||||||
|
use prettytable;
|
||||||
|
use term;
|
||||||
|
use std::io::prelude::*;
|
||||||
|
|
||||||
pub fn show_info(config: &WalletConfig, keychain: &Keychain) {
|
pub fn show_info(config: &WalletConfig, keychain: &Keychain) {
|
||||||
let root_key_id = keychain.root_key_id();
|
|
||||||
let result = checker::refresh_outputs(&config, &keychain);
|
let result = checker::refresh_outputs(&config, &keychain);
|
||||||
|
|
||||||
// just read the wallet here, no need for a write lock
|
|
||||||
let _ = WalletData::read_wallet(&config.data_file_dir, |wallet_data| {
|
let _ = WalletData::read_wallet(&config.data_file_dir, |wallet_data| {
|
||||||
// get the current height via the api
|
|
||||||
// if we cannot get the current height use the max height known to the wallet
|
|
||||||
let current_height = match checker::get_tip_from_node(config) {
|
let current_height = match checker::get_tip_from_node(config) {
|
||||||
Ok(tip) => tip.height,
|
Ok(tip) => tip.height,
|
||||||
Err(_) => match wallet_data.outputs.values().map(|out| out.height).max() {
|
Err(_) => match wallet_data.outputs.values().map(|out| out.height).max() {
|
||||||
|
@ -32,32 +32,52 @@ pub fn show_info(config: &WalletConfig, keychain: &Keychain) {
|
||||||
None => 0,
|
None => 0,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
let mut unspent_total=0;
|
||||||
println!("Outputs - ");
|
let mut unspent_but_locked_total=0;
|
||||||
println!("key_id, height, lock_height, status, coinbase?, num_confs, value");
|
let mut unconfirmed_total=0;
|
||||||
println!("----------------------------------");
|
let mut locked_total=0;
|
||||||
|
for out in wallet_data
|
||||||
let mut outputs = wallet_data
|
|
||||||
.outputs
|
.outputs
|
||||||
.values()
|
.values()
|
||||||
.filter(|out| out.root_key_id == root_key_id)
|
.filter(|out| out.root_key_id == keychain.root_key_id())
|
||||||
.collect::<Vec<_>>();
|
{
|
||||||
outputs.sort_by_key(|out| out.n_child);
|
if out.status == OutputStatus::Unspent {
|
||||||
for out in outputs {
|
unspent_total+=out.value;
|
||||||
println!(
|
if out.lock_height > current_height {
|
||||||
"{}, {}, {}, {:?}, {}, {}, {}",
|
unspent_but_locked_total+=out.value;
|
||||||
out.key_id,
|
|
||||||
out.height,
|
|
||||||
out.lock_height,
|
|
||||||
out.status,
|
|
||||||
out.is_coinbase,
|
|
||||||
out.num_confirmations(current_height),
|
|
||||||
core::amount_to_hr_string(out.value),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
if out.status == OutputStatus::Unconfirmed && !out.is_coinbase {
|
||||||
|
unconfirmed_total+=out.value;
|
||||||
|
}
|
||||||
|
if out.status == OutputStatus::Locked {
|
||||||
|
locked_total+=out.value;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
println!();
|
||||||
|
let title=format!("Wallet Summary Info - Block Height: {}", current_height);
|
||||||
|
let mut t = term::stdout().unwrap();
|
||||||
|
t.fg(term::color::MAGENTA).unwrap();
|
||||||
|
writeln!(t, "{}", title).unwrap();
|
||||||
|
writeln!(t, "--------------------------").unwrap();
|
||||||
|
t.reset().unwrap();
|
||||||
|
|
||||||
|
let mut table = table!(
|
||||||
|
[bFG->"Total", FG->amount_to_hr_string(unspent_total+unconfirmed_total)],
|
||||||
|
[bFY->"Awaiting Confirmation", FY->amount_to_hr_string(unconfirmed_total)],
|
||||||
|
[bFY->"Confirmed but Still Locked", FY->amount_to_hr_string(unspent_but_locked_total)],
|
||||||
|
[bFG->"Currently Spendable", FG->amount_to_hr_string(unspent_total-unspent_but_locked_total)],
|
||||||
|
[Fw->"---------", Fw->"---------"],
|
||||||
|
[Fr->"(Locked by previous transaction)", Fr->amount_to_hr_string(locked_total)]
|
||||||
|
);
|
||||||
|
table.set_format(*prettytable::format::consts::FORMAT_NO_BORDER_LINE_SEPARATOR);
|
||||||
|
table.printstd();
|
||||||
|
println!();
|
||||||
});
|
});
|
||||||
|
|
||||||
if let Err(e) = result {
|
if let Err(_) = result {
|
||||||
println!("WARNING - Showing local data only - Wallet was unable to contact a node to update and verify the outputs shown here.");
|
println!("WARNING - Showing local data only - Wallet was unable to contact a node to update and verify the info shown here.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,9 @@ extern crate serde_derive;
|
||||||
extern crate serde_json;
|
extern crate serde_json;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate slog;
|
extern crate slog;
|
||||||
|
#[macro_use]
|
||||||
|
extern crate prettytable;
|
||||||
|
extern crate term;
|
||||||
|
|
||||||
extern crate bodyparser;
|
extern crate bodyparser;
|
||||||
extern crate futures;
|
extern crate futures;
|
||||||
|
@ -40,6 +43,7 @@ extern crate grin_util as util;
|
||||||
|
|
||||||
mod checker;
|
mod checker;
|
||||||
mod handlers;
|
mod handlers;
|
||||||
|
mod outputs;
|
||||||
mod info;
|
mod info;
|
||||||
mod receiver;
|
mod receiver;
|
||||||
mod sender;
|
mod sender;
|
||||||
|
@ -47,6 +51,7 @@ mod types;
|
||||||
pub mod client;
|
pub mod client;
|
||||||
pub mod server;
|
pub mod server;
|
||||||
|
|
||||||
|
pub use outputs::show_outputs;
|
||||||
pub use info::show_info;
|
pub use info::show_info;
|
||||||
pub use receiver::{receive_json_tx, receive_json_tx_str, WalletReceiver};
|
pub use receiver::{receive_json_tx, receive_json_tx_str, WalletReceiver};
|
||||||
pub use sender::{issue_burn_tx, issue_send_tx};
|
pub use sender::{issue_burn_tx, issue_send_tx};
|
||||||
|
|
92
wallet/src/outputs.rs
Normal file
92
wallet/src/outputs.rs
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
// Copyright 2017 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.
|
||||||
|
|
||||||
|
use checker;
|
||||||
|
use keychain::Keychain;
|
||||||
|
use core::core;
|
||||||
|
use types::{WalletConfig, WalletData};
|
||||||
|
use prettytable;
|
||||||
|
use term;
|
||||||
|
use std::io::prelude::*;
|
||||||
|
|
||||||
|
pub fn show_outputs(config: &WalletConfig, keychain: &Keychain) {
|
||||||
|
let root_key_id = keychain.root_key_id();
|
||||||
|
let result = checker::refresh_outputs(&config, &keychain);
|
||||||
|
|
||||||
|
// just read the wallet here, no need for a write lock
|
||||||
|
let _ = WalletData::read_wallet(&config.data_file_dir, |wallet_data| {
|
||||||
|
// get the current height via the api
|
||||||
|
// if we cannot get the current height use the max height known to the wallet
|
||||||
|
let current_height = match checker::get_tip_from_node(config) {
|
||||||
|
Ok(tip) => tip.height,
|
||||||
|
Err(_) => match wallet_data.outputs.values().map(|out| out.height).max() {
|
||||||
|
Some(height) => height,
|
||||||
|
None => 0,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut outputs = wallet_data
|
||||||
|
.outputs
|
||||||
|
.values()
|
||||||
|
.filter(|out| out.root_key_id == root_key_id)
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
outputs.sort_by_key(|out| out.n_child);
|
||||||
|
|
||||||
|
let title=format!("Wallet Outputs - Block Height: {}", current_height);
|
||||||
|
println!();
|
||||||
|
let mut t = term::stdout().unwrap();
|
||||||
|
t.fg(term::color::MAGENTA).unwrap();
|
||||||
|
writeln!(t, "{}", title).unwrap();
|
||||||
|
t.reset().unwrap();
|
||||||
|
|
||||||
|
let mut table = table!();
|
||||||
|
|
||||||
|
table.set_titles(row![
|
||||||
|
bMG->"Key Id",
|
||||||
|
bMG->"Block Height",
|
||||||
|
bMG->"Locked Until",
|
||||||
|
bMG->"Status",
|
||||||
|
bMG->"Is Coinbase?",
|
||||||
|
bMG->"Num. of Confirmations",
|
||||||
|
bMG->"Value"
|
||||||
|
]);
|
||||||
|
|
||||||
|
for out in outputs {
|
||||||
|
let key_id=format!("{}", out.key_id);
|
||||||
|
let height=format!("{}", out.height);
|
||||||
|
let lock_height=format!("{}", out.lock_height);
|
||||||
|
let status=format!("{:?}", out.status);
|
||||||
|
let is_coinbase=format!("{}", out.is_coinbase);
|
||||||
|
let num_confirmations=format!("{}", out.num_confirmations(current_height));
|
||||||
|
let value=format!("{}", core::amount_to_hr_string(out.value));
|
||||||
|
table.add_row(row![
|
||||||
|
bFC->key_id,
|
||||||
|
bFB->height,
|
||||||
|
bFB->lock_height,
|
||||||
|
bFR->status,
|
||||||
|
bFY->is_coinbase,
|
||||||
|
bFB->num_confirmations,
|
||||||
|
bFG->value
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
table.set_format(*prettytable::format::consts::FORMAT_NO_COLSEP);
|
||||||
|
table.printstd();
|
||||||
|
println!();
|
||||||
|
});
|
||||||
|
|
||||||
|
if let Err(_) = result {
|
||||||
|
println!("WARNING - Showing local data only - Wallet was unable to contact a node to update and verify the outputs shown here.");
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue