2018-03-13 21:10:45 +03:00
|
|
|
// Copyright 2018 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.
|
|
|
|
|
|
|
|
//! Basic status view definition
|
|
|
|
|
2018-10-13 05:12:13 +03:00
|
|
|
use chrono::prelude::Utc;
|
2018-06-14 15:16:14 +03:00
|
|
|
use cursive::direction::Orientation;
|
|
|
|
use cursive::traits::Identifiable;
|
2018-03-16 12:53:02 +03:00
|
|
|
use cursive::view::View;
|
2018-03-13 21:10:45 +03:00
|
|
|
use cursive::views::{BoxView, LinearLayout, TextView};
|
2018-07-09 20:01:19 +03:00
|
|
|
use cursive::Cursive;
|
2018-03-13 21:10:45 +03:00
|
|
|
|
2018-06-14 15:16:14 +03:00
|
|
|
use tui::constants::VIEW_BASIC_STATUS;
|
|
|
|
use tui::types::TUIStatusListener;
|
2018-03-13 21:10:45 +03:00
|
|
|
|
2018-07-02 02:08:39 +03:00
|
|
|
use servers::common::types::SyncStatus;
|
2018-04-12 12:11:24 +03:00
|
|
|
use servers::ServerStats;
|
2018-03-13 21:10:45 +03:00
|
|
|
|
2018-10-13 05:12:13 +03:00
|
|
|
const NANO_TO_MILLIS: f64 = 1.0 / 1_000_000.0;
|
|
|
|
|
2018-03-13 21:10:45 +03:00
|
|
|
pub struct TUIStatusView;
|
|
|
|
|
|
|
|
impl TUIStatusListener for TUIStatusView {
|
|
|
|
/// Create basic status view
|
2018-03-16 12:53:02 +03:00
|
|
|
fn create() -> Box<View> {
|
2018-03-13 21:10:45 +03:00
|
|
|
let basic_status_view = BoxView::with_full_screen(
|
|
|
|
LinearLayout::new(Orientation::Vertical)
|
|
|
|
.child(
|
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("Current Status: "))
|
|
|
|
.child(TextView::new("Starting").with_id("basic_current_status")),
|
2018-09-27 22:45:48 +03:00
|
|
|
).child(
|
2018-03-13 21:10:45 +03:00
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("Connected Peers: "))
|
|
|
|
.child(TextView::new("0").with_id("connected_peers")),
|
2018-10-05 10:29:33 +03:00
|
|
|
).child(
|
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("------------------------")),
|
2018-11-09 18:14:25 +03:00
|
|
|
).child(
|
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("Header Tip Hash: "))
|
|
|
|
.child(TextView::new(" ").with_id("basic_header_tip_hash")),
|
2018-10-05 10:29:33 +03:00
|
|
|
).child(
|
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("Header Chain Height: "))
|
|
|
|
.child(TextView::new(" ").with_id("basic_header_chain_height")),
|
|
|
|
).child(
|
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("Header Cumulative Difficulty: "))
|
|
|
|
.child(TextView::new(" ").with_id("basic_header_total_difficulty")),
|
|
|
|
).child(
|
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("------------------------")),
|
2018-11-09 18:14:25 +03:00
|
|
|
).child(
|
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("Tip Hash: "))
|
|
|
|
.child(TextView::new(" ").with_id("tip_hash")),
|
2018-09-27 22:45:48 +03:00
|
|
|
).child(
|
2018-03-13 21:10:45 +03:00
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("Chain Height: "))
|
|
|
|
.child(TextView::new(" ").with_id("chain_height")),
|
2018-09-27 22:45:48 +03:00
|
|
|
).child(
|
2018-03-29 14:29:17 +03:00
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
2018-10-05 10:29:33 +03:00
|
|
|
.child(TextView::new("Cumulative Difficulty: "))
|
2018-03-29 14:29:17 +03:00
|
|
|
.child(TextView::new(" ").with_id("basic_total_difficulty")),
|
2018-09-27 22:45:48 +03:00
|
|
|
).child(
|
2018-03-13 21:10:45 +03:00
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new("------------------------")),
|
2018-09-27 22:45:48 +03:00
|
|
|
).child(
|
2018-03-13 21:10:45 +03:00
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new(" ").with_id("basic_mining_config_status")),
|
2018-09-27 22:45:48 +03:00
|
|
|
).child(
|
2018-03-13 21:10:45 +03:00
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new(" ").with_id("basic_mining_status")),
|
2018-09-27 22:45:48 +03:00
|
|
|
).child(
|
2018-03-13 21:10:45 +03:00
|
|
|
LinearLayout::new(Orientation::Horizontal)
|
|
|
|
.child(TextView::new(" ").with_id("basic_network_info")),
|
|
|
|
), //.child(logo_view)
|
|
|
|
);
|
|
|
|
Box::new(basic_status_view.with_id(VIEW_BASIC_STATUS))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// update
|
|
|
|
fn update(c: &mut Cursive, stats: &ServerStats) {
|
|
|
|
//find and update here as needed
|
|
|
|
let basic_status = {
|
2018-10-23 08:50:13 +03:00
|
|
|
match stats.sync_status {
|
|
|
|
SyncStatus::Initial => "Initializing".to_string(),
|
|
|
|
SyncStatus::NoSync => "Running".to_string(),
|
|
|
|
SyncStatus::AwaitingPeers(_) => "Waiting for peers".to_string(),
|
|
|
|
SyncStatus::HeaderSync {
|
|
|
|
current_height,
|
|
|
|
highest_height,
|
|
|
|
} => {
|
|
|
|
let percent = if highest_height == 0 {
|
|
|
|
0
|
|
|
|
} else {
|
|
|
|
current_height * 100 / highest_height
|
|
|
|
};
|
|
|
|
format!("Downloading headers: {}%, step 1/4", percent)
|
|
|
|
}
|
|
|
|
SyncStatus::TxHashsetDownload {
|
|
|
|
start_time,
|
|
|
|
downloaded_size,
|
|
|
|
total_size,
|
|
|
|
} => {
|
|
|
|
if total_size > 0 {
|
|
|
|
let percent = if total_size > 0 {
|
|
|
|
downloaded_size * 100 / total_size
|
2018-07-02 02:08:39 +03:00
|
|
|
} else {
|
2018-10-23 08:50:13 +03:00
|
|
|
0
|
2018-07-02 02:08:39 +03:00
|
|
|
};
|
2018-10-23 08:50:13 +03:00
|
|
|
let start = start_time.timestamp_nanos();
|
|
|
|
let fin = Utc::now().timestamp_nanos();
|
|
|
|
let dur_ms = (fin - start) as f64 * NANO_TO_MILLIS;
|
2018-10-13 05:12:13 +03:00
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
format!("Downloading {}(MB) chain state for state sync: {}% at {:.1?}(kB/s), step 2/4",
|
2018-10-23 08:50:13 +03:00
|
|
|
total_size / 1_000_000,
|
|
|
|
percent,
|
|
|
|
if dur_ms > 1.0f64 { downloaded_size as f64 / dur_ms as f64 } else { 0f64 },
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
let start = start_time.timestamp_millis();
|
|
|
|
let fin = Utc::now().timestamp_millis();
|
|
|
|
let dur_secs = (fin - start) / 1000;
|
2018-10-13 05:12:13 +03:00
|
|
|
|
2018-11-10 06:27:52 +03:00
|
|
|
format!("Downloading chain state for state sync. Waiting remote peer to start: {}s, step 2/4",
|
2018-10-23 08:50:13 +03:00
|
|
|
dur_secs,
|
|
|
|
)
|
2018-07-02 02:08:39 +03:00
|
|
|
}
|
|
|
|
}
|
2018-10-23 08:50:13 +03:00
|
|
|
SyncStatus::TxHashsetSetup => {
|
|
|
|
"Preparing chain state for validation, step 3/4".to_string()
|
|
|
|
}
|
|
|
|
SyncStatus::TxHashsetValidation {
|
|
|
|
kernels,
|
|
|
|
kernel_total,
|
|
|
|
rproofs,
|
|
|
|
rproof_total,
|
|
|
|
} => {
|
|
|
|
// 10% of overall progress is attributed to kernel validation
|
|
|
|
// 90% to range proofs (which are much longer)
|
|
|
|
let mut percent = if kernel_total > 0 {
|
|
|
|
kernels * 10 / kernel_total
|
|
|
|
} else {
|
|
|
|
0
|
|
|
|
};
|
|
|
|
percent += if rproof_total > 0 {
|
|
|
|
rproofs * 90 / rproof_total
|
|
|
|
} else {
|
|
|
|
0
|
|
|
|
};
|
|
|
|
format!("Validating chain state: {}%, step 3/4", percent)
|
|
|
|
}
|
|
|
|
SyncStatus::TxHashsetSave => {
|
2018-11-10 06:27:52 +03:00
|
|
|
"Finalizing chain state for state sync, step 3/4".to_string()
|
|
|
|
}
|
|
|
|
SyncStatus::TxHashsetDone => {
|
|
|
|
"Finalized chain state for state sync, step 3/4".to_string()
|
2018-10-23 08:50:13 +03:00
|
|
|
}
|
|
|
|
SyncStatus::BodySync {
|
|
|
|
current_height,
|
|
|
|
highest_height,
|
|
|
|
} => {
|
|
|
|
let percent = if highest_height == 0 {
|
|
|
|
0
|
|
|
|
} else {
|
|
|
|
current_height * 100 / highest_height
|
|
|
|
};
|
|
|
|
format!("Downloading blocks: {}%, step 4/4", percent)
|
|
|
|
}
|
2018-03-13 21:10:45 +03:00
|
|
|
}
|
|
|
|
};
|
2018-04-24 11:18:24 +03:00
|
|
|
/*let basic_mining_config_status = {
|
2018-03-13 21:10:45 +03:00
|
|
|
if stats.mining_stats.is_enabled {
|
|
|
|
"Configured as mining node"
|
|
|
|
} else {
|
|
|
|
"Configured as validating node only (not mining)"
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let (basic_mining_status, basic_network_info) = {
|
|
|
|
if stats.mining_stats.is_enabled {
|
|
|
|
if stats.is_syncing {
|
|
|
|
(
|
|
|
|
"Mining Status: Paused while syncing".to_string(),
|
|
|
|
" ".to_string(),
|
|
|
|
)
|
|
|
|
} else if stats.mining_stats.combined_gps == 0.0 {
|
|
|
|
(
|
2018-03-27 21:06:54 +03:00
|
|
|
"Mining Status: Starting miner and awaiting first solution...".to_string(),
|
2018-03-13 21:10:45 +03:00
|
|
|
" ".to_string(),
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
(
|
|
|
|
format!(
|
|
|
|
"Mining Status: Mining at height {} at {:.*} GPS",
|
|
|
|
stats.mining_stats.block_height, 4, stats.mining_stats.combined_gps
|
|
|
|
),
|
|
|
|
format!(
|
|
|
|
"Cuckoo {} - Network Difficulty {}",
|
2018-10-16 02:14:23 +03:00
|
|
|
stats.mining_stats.edge_bits,
|
2018-03-13 21:10:45 +03:00
|
|
|
stats.mining_stats.network_difficulty.to_string()
|
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
(" ".to_string(), " ".to_string())
|
|
|
|
}
|
2018-04-24 11:18:24 +03:00
|
|
|
};*/
|
2018-03-13 21:10:45 +03:00
|
|
|
c.call_on_id("basic_current_status", |t: &mut TextView| {
|
|
|
|
t.set_content(basic_status);
|
|
|
|
});
|
|
|
|
c.call_on_id("connected_peers", |t: &mut TextView| {
|
|
|
|
t.set_content(stats.peer_count.to_string());
|
|
|
|
});
|
2018-11-09 18:14:25 +03:00
|
|
|
c.call_on_id("tip_hash", |t: &mut TextView| {
|
|
|
|
t.set_content(stats.head.last_block_h.to_string());
|
|
|
|
});
|
2018-03-13 21:10:45 +03:00
|
|
|
c.call_on_id("chain_height", |t: &mut TextView| {
|
|
|
|
t.set_content(stats.head.height.to_string());
|
|
|
|
});
|
2018-03-29 14:29:17 +03:00
|
|
|
c.call_on_id("basic_total_difficulty", |t: &mut TextView| {
|
|
|
|
t.set_content(stats.head.total_difficulty.to_string());
|
|
|
|
});
|
2018-11-09 18:14:25 +03:00
|
|
|
c.call_on_id("basic_header_tip_hash", |t: &mut TextView| {
|
|
|
|
t.set_content(stats.header_head.last_block_h.to_string());
|
|
|
|
});
|
2018-10-05 10:29:33 +03:00
|
|
|
c.call_on_id("basic_header_chain_height", |t: &mut TextView| {
|
|
|
|
t.set_content(stats.header_head.height.to_string());
|
|
|
|
});
|
|
|
|
c.call_on_id("basic_header_total_difficulty", |t: &mut TextView| {
|
|
|
|
t.set_content(stats.header_head.total_difficulty.to_string());
|
|
|
|
});
|
2018-04-24 11:18:24 +03:00
|
|
|
/*c.call_on_id("basic_mining_config_status", |t: &mut TextView| {
|
2018-03-13 21:10:45 +03:00
|
|
|
t.set_content(basic_mining_config_status);
|
|
|
|
});
|
|
|
|
c.call_on_id("basic_mining_status", |t: &mut TextView| {
|
|
|
|
t.set_content(basic_mining_status);
|
|
|
|
});
|
|
|
|
c.call_on_id("basic_network_info", |t: &mut TextView| {
|
|
|
|
t.set_content(basic_network_info);
|
2018-04-24 11:18:24 +03:00
|
|
|
});*/
|
2018-03-13 21:10:45 +03:00
|
|
|
}
|
|
|
|
}
|