1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
// This file is part of Substrate.
// Copyright (C) Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: Apache-2.0
// 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 std::{
future::Future,
io::{self, IsTerminal},
time::Instant,
};
use spinners::{Spinner, Spinners};
use super::Result;
// A simple helper to time a operation with a nice spinner, start message, and end message.
//
// The spinner is only displayed when stdout is a terminal.
pub(super) fn with_elapsed<F, R, EndMsg>(f: F, start_msg: &str, end_msg: EndMsg) -> Result<R>
where
F: FnOnce() -> Result<R>,
EndMsg: FnOnce(&R) -> String,
{
let timer = Instant::now();
let mut maybe_sp = start(start_msg);
Ok(end(f()?, timer, maybe_sp.as_mut(), end_msg))
}
// A simple helper to time an async operation with a nice spinner, start message, and end message.
//
// The spinner is only displayed when stdout is a terminal.
pub(super) async fn with_elapsed_async<F, Fut, R, EndMsg>(
f: F,
start_msg: &str,
end_msg: EndMsg,
) -> Result<R>
where
F: FnOnce() -> Fut,
Fut: Future<Output = Result<R>>,
EndMsg: FnOnce(&R) -> String,
{
let timer = Instant::now();
let mut maybe_sp = start(start_msg);
Ok(end(f().await?, timer, maybe_sp.as_mut(), end_msg))
}
fn start(start_msg: &str) -> Option<Spinner> {
let msg = format!("⏳ {start_msg}");
if io::stdout().is_terminal() {
Some(Spinner::new(Spinners::Dots, msg))
} else {
println!("{msg}");
None
}
}
fn end<T, EndMsg>(val: T, timer: Instant, maybe_sp: Option<&mut Spinner>, end_msg: EndMsg) -> T
where
EndMsg: FnOnce(&T) -> String,
{
let msg = format!("✅ {} in {:.2}s", end_msg(&val), timer.elapsed().as_secs_f32());
if let Some(sp) = maybe_sp {
sp.stop_with_message(msg);
} else {
println!("{msg}");
}
val
}