nautilus_common/testing.rs
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 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
// -------------------------------------------------------------------------------------------------
// Copyright (C) 2015-2025 Nautech Systems Pty Ltd. All rights reserved.
// https://nautechsystems.io
//
// Licensed under the GNU Lesser General Public License Version 3.0 (the "License");
// You may not use this file except in compliance with the License.
// You may obtain a copy of the License at https://www.gnu.org/licenses/lgpl-3.0.en.html
//
// 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.
// -------------------------------------------------------------------------------------------------
//! Common test related helper functions.
use std::{
future::Future,
thread,
time::{Duration, Instant},
};
use nautilus_core::UUID4;
use nautilus_model::identifiers::TraderId;
use crate::logging::{init_logging, logger::LoggerConfig, writer::FileWriterConfig};
pub fn init_logger_for_testing(stdout_level: Option<log::LevelFilter>) {
let mut config = LoggerConfig::default();
config.stdout_level = stdout_level.unwrap_or(log::LevelFilter::Trace);
init_logging(
TraderId::default(),
UUID4::new(),
config,
FileWriterConfig::default(),
);
}
/// Repeatedly evaluates a condition with a delay until it becomes true or a timeout occurs.
///
/// * `condition`: A closure that represents the condition to be met. This closure should return `true`
/// when the condition is met and `false` otherwise.
/// * `timeout`: The maximum amount of time to wait for the condition to be met. If this duration is
/// exceeded, the function will panic.
///
/// # Panics
///
/// This function will panic if the timeout duration is exceeded without the condition being met.
///
/// # Examples
///
/// ```
/// use std::time::Duration;
/// use std::thread;
/// use nautilus_common::testing::wait_until;
///
/// let start_time = std::time::Instant::now();
/// let timeout = Duration::from_secs(5);
///
/// wait_until(|| {
/// if start_time.elapsed().as_secs() > 2 {
/// true
/// } else {
/// false
/// }
/// }, timeout);
/// ```
///
/// In the above example, the `wait_until` function will block for at least 2 seconds, as that's how long
/// it takes for the condition to be met. If the condition was not met within 5 seconds, it would panic.
pub fn wait_until<F>(mut condition: F, timeout: Duration)
where
F: FnMut() -> bool,
{
let start_time = Instant::now();
loop {
if condition() {
break;
}
assert!(
start_time.elapsed() <= timeout,
"Timeout waiting for condition"
);
thread::sleep(Duration::from_millis(100));
}
}
pub async fn wait_until_async<F, Fut>(mut condition: F, timeout: Duration)
where
F: FnMut() -> Fut,
Fut: Future<Output = bool>,
{
let start_time = Instant::now();
loop {
if condition().await {
break;
}
assert!(
start_time.elapsed() <= timeout,
"Timeout waiting for condition"
);
tokio::time::sleep(Duration::from_millis(100)).await;
}
}