Compare commits
No commits in common. "2580886d04c3d55dfa45b78b30790a339b3842a6" and "ef9c2d4fd9a51aaa68174a7c9ba2e078b3479027" have entirely different histories.
2580886d04
...
ef9c2d4fd9
@ -1,4 +1,4 @@
|
|||||||
workspace = { members = ["config", "logging"] }
|
workspace = { members = ["config", "logging", "macros"] }
|
||||||
|
|
||||||
[package]
|
[package]
|
||||||
name = "wanessa"
|
name = "wanessa"
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
* A singleton, thread-safe struct used for accessing
|
* A singleton, thread-safe struct used for accessing
|
||||||
*/
|
*/
|
||||||
|
|
||||||
use std::path::Path;
|
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::sync::RwLock;
|
use std::sync::RwLock;
|
||||||
|
|
||||||
@ -38,7 +37,6 @@ pub struct Config {
|
|||||||
log_location: bool,
|
log_location: bool,
|
||||||
/// If `Some(path)` tries to also write the log to `path` in addition to stderr/stderr.<br>
|
/// If `Some(path)` tries to also write the log to `path` in addition to stderr/stderr.<br>
|
||||||
/// Default: [None]
|
/// Default: [None]
|
||||||
#[getset(skip)]
|
|
||||||
log_path: Option<PathBuf>,
|
log_path: Option<PathBuf>,
|
||||||
/// Logs to standard out, if true.<br>
|
/// Logs to standard out, if true.<br>
|
||||||
/// Default: `true`
|
/// Default: `true`
|
||||||
@ -71,26 +69,14 @@ impl Config {
|
|||||||
/// Getter for [`Self::log_time_format`].
|
/// Getter for [`Self::log_time_format`].
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn log_time_format(&self) -> &str {
|
pub fn log_time_format(&self) -> &str {
|
||||||
self.log_time_format.as_ref().map_or("%F-%T:%f", |fmt| fmt)
|
todo!();
|
||||||
|
// self.log_time_format.as_ref().map_or("%F-%T:%f", |fmt| fmt)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Setter for [`Self::log_time_format`].
|
/// Setter for [`Self::log_time_format`].
|
||||||
pub fn set_log_time_format(&mut self, format: impl Into<String>) {
|
pub fn set_log_time_format(&mut self, format: impl Into<String>) {
|
||||||
self.log_time_format = Some(format.into());
|
self.log_time_format = Some(format.into());
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Getter for [`Self::log_path`].
|
|
||||||
#[must_use]
|
|
||||||
pub fn log_path(&self) -> Option<&Path>
|
|
||||||
{
|
|
||||||
self.log_path.as_deref()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Setter for [`Self::log_path`].
|
|
||||||
pub fn set_log_path(&mut self, log_path: impl Into<Option<PathBuf>>)
|
|
||||||
{
|
|
||||||
self.log_path = log_path.into();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// See [`DEFAULTS`].
|
/// See [`DEFAULTS`].
|
||||||
@ -110,15 +96,15 @@ impl Default for Config {
|
|||||||
*
|
*
|
||||||
* defaults.set_log_verbosity(Warning);
|
* defaults.set_log_verbosity(Warning);
|
||||||
* defaults.set_log_time(false);
|
* defaults.set_log_time(false);
|
||||||
* assert_eq!(defaults.log_time_format(), "%F-%T:%f");
|
* //defaults.set_log_time_format(None);
|
||||||
* defaults.set_log_location(false);
|
* defaults.set_log_location(false);
|
||||||
* defaults.set_log_stdout(true);
|
* defaults.set_log_stdout(true);
|
||||||
* defaults.set_log_stderr(true);
|
* defaults.set_log_stderr(true);
|
||||||
* defaults.set_log_path(None);
|
* //defaults.set_log_path(None);
|
||||||
* assert_eq!(defaults.db_addr(), "localhost");
|
* //defaults.set_db_addr(None);
|
||||||
* defaults.set_db_port(6969);
|
* defaults.set_db_port(6969);
|
||||||
*
|
*
|
||||||
* # assert_eq!(defaults,config::DEFAULTS)
|
* # assert!(defaults == config::DEFAULTS)
|
||||||
* ```
|
* ```
|
||||||
*/
|
*/
|
||||||
pub const DEFAULTS: Config = Config {
|
pub const DEFAULTS: Config = Config {
|
||||||
|
@ -9,11 +9,6 @@ edition = "2021"
|
|||||||
chrono = { version = "0.4.34", default-features = false, features = ["now"] }
|
chrono = { version = "0.4.34", default-features = false, features = ["now"] }
|
||||||
config = { path = "../config"}
|
config = { path = "../config"}
|
||||||
|
|
||||||
[dev-dependencies]
|
|
||||||
uuid = { version = "1.7.0", default-features = false, features = ["v4", "fast-rng"] }
|
|
||||||
once_cell = { version = "1.19.0", default-features = false, features = ["std"] }
|
|
||||||
futures = { version = "0.3.30", default-features = false, features = ["alloc", "executor"] }
|
|
||||||
|
|
||||||
[lints.rust]
|
[lints.rust]
|
||||||
unsafe_code = "forbid"
|
unsafe_code = "forbid"
|
||||||
missing_docs = "warn"
|
missing_docs = "warn"
|
||||||
|
@ -3,14 +3,11 @@
|
|||||||
* It should mostly be called statically.
|
* It should mostly be called statically.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
mod test;
|
|
||||||
|
|
||||||
use std::fmt::Display;
|
use std::fmt::Display;
|
||||||
use std::fs::OpenOptions;
|
use std::fs::OpenOptions;
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
|
|
||||||
use config::LogVerbosity;
|
use config::LogVerbosity;
|
||||||
use config::Config;
|
|
||||||
|
|
||||||
use chrono::Utc;
|
use chrono::Utc;
|
||||||
|
|
||||||
@ -29,49 +26,14 @@ use LogMessageType::GenericWarn;
|
|||||||
* Panics if readlock on [`config::CONFIG`] could not be acquired
|
* Panics if readlock on [`config::CONFIG`] could not be acquired
|
||||||
* or if another error occurs, such as a full disk.
|
* or if another error occurs, such as a full disk.
|
||||||
*/
|
*/
|
||||||
pub fn log_message(msg: &LogMessage, conf: &Config, file: &str, line: u32, column: u32) {
|
// TODO: Create macro to make last three parameters optional.
|
||||||
|
pub fn log_message(msg: &LogMessage, file: &str, line: &str, column: &str) {
|
||||||
let Some(log_line) = log_to_str(msg, conf, file, line, column) else { return };
|
let conf = config::CONFIG.read().unwrap_or_else(|_| {
|
||||||
|
panic!("Failed aqcuire read lock on config\nTried to log message: {msg}")
|
||||||
// May panic if file cannot be opened or written to.
|
|
||||||
conf.log_path().as_ref().map_or_else(
|
|
||||||
|| {},
|
|
||||||
|path| {
|
|
||||||
let mut file = OpenOptions::new()
|
|
||||||
.write(true)
|
|
||||||
.append(true)
|
|
||||||
.create(true)
|
|
||||||
.open(path)
|
|
||||||
.unwrap_or_else(|_| {
|
|
||||||
panic!(
|
|
||||||
"Could not open log file: {path:#?}"
|
|
||||||
)
|
|
||||||
});
|
});
|
||||||
writeln!(file, "{log_line}")
|
|
||||||
.unwrap_or_else(|_| panic!("Could not write log to file: {path:#?}"));
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
if msg.1 <= Warning && *conf.log_stderr() {
|
|
||||||
// May panic if writing to stderr fails.
|
|
||||||
eprintln!("{log_line}");
|
|
||||||
} else if msg.1 >= Information && *conf.log_stdout() {
|
|
||||||
// May panic if writing to stdout fails.
|
|
||||||
println!("{log_line}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return log line, if message may be logged according to [`config::Config`].
|
|
||||||
* # Panics
|
|
||||||
* Panics if readlock on [`config::CONFIG`] could not be acquired
|
|
||||||
* or if another error occurs, such as a full disk.
|
|
||||||
*/
|
|
||||||
#[must_use]
|
|
||||||
pub fn log_to_str(msg: &LogMessage, conf: &Config, file: &str, line: u32, column: u32) -> Option<String>
|
|
||||||
{
|
|
||||||
if conf.log_verbosity() < &msg.1 {
|
if conf.log_verbosity() < &msg.1 {
|
||||||
return None;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut log_line = String::new();
|
let mut log_line = String::new();
|
||||||
@ -86,27 +48,41 @@ pub fn log_to_str(msg: &LogMessage, conf: &Config, file: &str, line: u32, column
|
|||||||
log_line += &format!("{file}:{line},{column} ");
|
log_line += &format!("{file}:{line},{column} ");
|
||||||
}
|
}
|
||||||
|
|
||||||
Some(log_line + &msg.to_string())
|
log_line += &msg.to_string();
|
||||||
|
|
||||||
|
// May panic if file cannot be opened or written to.
|
||||||
|
conf.log_path().as_ref().map_or_else(
|
||||||
|
|| {},
|
||||||
|
|path| {
|
||||||
|
let mut file = OpenOptions::new()
|
||||||
|
.write(true)
|
||||||
|
.append(true)
|
||||||
|
.open(path)
|
||||||
|
.unwrap_or_else(|_| {
|
||||||
|
panic!(
|
||||||
|
"Could not open log fil
|
||||||
|
e: {path:#?}"
|
||||||
|
)
|
||||||
|
});
|
||||||
|
writeln!(file, "{log_line}")
|
||||||
|
.unwrap_or_else(|_| panic!("Could not write log to file: {path:#?}"));
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if msg.1 <= Warning && *conf.log_stderr() {
|
||||||
|
// May panic if writing to stderr fails.
|
||||||
|
eprintln!("{log_line}");
|
||||||
|
} else if msg.1 >= Information && *conf.log_stdout() {
|
||||||
|
// May panic if writing to stdout fails.
|
||||||
|
println!("{log_line}");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
drop(conf); // remove read lock
|
||||||
* Shorthand version for [`log_message`], which does not require information about the [`config::Config::log_location`].
|
|
||||||
*/
|
|
||||||
#[macro_export]
|
|
||||||
macro_rules! log{
|
|
||||||
($msg:expr) => {
|
|
||||||
let conf = config::CONFIG.read().unwrap_or_else(|_| {
|
|
||||||
panic!("Failed aqcuire read lock on config!")
|
|
||||||
});
|
|
||||||
log_message($msg, &*conf, file!(), line!(), column!());
|
|
||||||
drop(conf);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A named typle assigning a log [`LogVerbosity`] to a [`LogMessageType`]
|
* A named typle assigning a log [`LogVerbosity`] to a [`LogMessageType`]
|
||||||
*/
|
*/
|
||||||
#[derive(PartialEq, Eq, Debug)]
|
|
||||||
pub struct LogMessage(LogMessageType, LogVerbosity);
|
pub struct LogMessage(LogMessageType, LogVerbosity);
|
||||||
|
|
||||||
impl Display for LogMessage {
|
impl Display for LogMessage {
|
||||||
@ -120,7 +96,7 @@ impl Display for LogMessage {
|
|||||||
*
|
*
|
||||||
* These groups correspond to the four levels of logging verbosity. See [`LogVerbosity`].
|
* These groups correspond to the four levels of logging verbosity. See [`LogVerbosity`].
|
||||||
*/
|
*/
|
||||||
#[derive(Clone, Debug, PartialEq, Eq)]
|
#[derive(Clone, Debug)]
|
||||||
pub enum LogMessageType {
|
pub enum LogMessageType {
|
||||||
/// Errors
|
/// Errors
|
||||||
/// An error of any type; please avoid using this.
|
/// An error of any type; please avoid using this.
|
||||||
|
@ -1,262 +0,0 @@
|
|||||||
#![cfg(test)]
|
|
||||||
/*!
|
|
||||||
* This test suite uses uuid to easily avoid race conditions when writing to the same log file.
|
|
||||||
*/
|
|
||||||
|
|
||||||
use std::{collections::HashSet, env, fs::{create_dir_all, read_to_string}, mem::take, path::PathBuf};
|
|
||||||
|
|
||||||
use futures::{executor::block_on, future::join_all};
|
|
||||||
use once_cell::sync::Lazy;
|
|
||||||
|
|
||||||
use uuid::Uuid;
|
|
||||||
|
|
||||||
use super::*;
|
|
||||||
|
|
||||||
const CONCURRENT_MESSAGE_COUNT: usize = 99999;
|
|
||||||
|
|
||||||
static LOG_DIR: Lazy<String> = Lazy::new(||
|
|
||||||
if cfg!(unix)
|
|
||||||
{
|
|
||||||
String::from("/tmp/WANessa/unit-tests/logging")
|
|
||||||
}
|
|
||||||
else if cfg !(windows) {
|
|
||||||
let tmp_path = env::var("TMP").unwrap_or_else(|_| env::var("TEMP").expect("Windows should have both TMP and TEMP, but you have neither, what did you do?"));
|
|
||||||
format!("{tmp_path}/WANessa/unit-tests/logging")
|
|
||||||
}
|
|
||||||
else { String::new() }
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/// Tests if [`log_message`] to file correctly.
|
|
||||||
#[test]
|
|
||||||
pub fn log_msg_file()
|
|
||||||
{
|
|
||||||
let log_path = &format!("{}/{}", *LOG_DIR, Uuid::new_v4());
|
|
||||||
println!("Log Path: {log_path}");
|
|
||||||
let message = LogMessageType::GenericWarn(String::from("Test Log")).log_message();
|
|
||||||
let mut config = config::DEFAULTS;
|
|
||||||
config.set_log_path(PathBuf::from(log_path));
|
|
||||||
|
|
||||||
create_dir_all(PathBuf::from(LOG_DIR.as_str()))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not create directory: {}", *LOG_DIR));
|
|
||||||
|
|
||||||
log_message(&message, &config, file!(), line!(), column!());
|
|
||||||
|
|
||||||
let log_file = read_to_string(PathBuf::from(log_path))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not read file: {log_path}"));
|
|
||||||
|
|
||||||
assert_eq!(message.to_string() + "\n", log_file);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Tests if [`log_message`] does not modify output from [`log_to_str`], when logging to file.
|
|
||||||
#[test]
|
|
||||||
pub fn log_str()
|
|
||||||
{
|
|
||||||
let log_path = &format!("{}/{}", *LOG_DIR, Uuid::new_v4());
|
|
||||||
println!("Log Path: {log_path}");
|
|
||||||
let message = LogMessageType::GenericWarn(String::from("Test Log")).log_message();
|
|
||||||
let mut config = config::DEFAULTS;
|
|
||||||
config.set_log_path(PathBuf::from(log_path));
|
|
||||||
|
|
||||||
create_dir_all(PathBuf::from(LOG_DIR.as_str()))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not create directory: {}", *LOG_DIR));
|
|
||||||
|
|
||||||
let log_line = log_to_str(&message, &config, file!(), line!(), column!()).expect("There should be a log line.") + "\n";
|
|
||||||
|
|
||||||
log_message(&message, &config, file!(), line!(), column!());
|
|
||||||
|
|
||||||
let log_file = read_to_string(PathBuf::from(log_path))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not read file: {log_path}"));
|
|
||||||
|
|
||||||
assert_eq!(log_line, log_file);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Tests if no messages are unintentionally filtered due to their verboisity.
|
|
||||||
#[test]
|
|
||||||
pub fn verbosity_no_filter()
|
|
||||||
{
|
|
||||||
let log_path = &format!("{}/{}", *LOG_DIR, Uuid::new_v4());
|
|
||||||
println!("Log Path: {log_path}");
|
|
||||||
let messages = vec![
|
|
||||||
LogMessageType::GenericErr(String::from("Test Err")).log_message(),
|
|
||||||
LogMessageType::GenericWarn(String::from("Test Warn")).log_message(),
|
|
||||||
LogMessageType::GenericInfo(String::from("Test Info")).log_message(),
|
|
||||||
LogMessageType::GenericDebug(String::from("Test Debug")).log_message(),
|
|
||||||
];
|
|
||||||
let mut config = config::DEFAULTS;
|
|
||||||
config.set_log_path(PathBuf::from(log_path));
|
|
||||||
config.set_log_verbosity(LogVerbosity::Error);
|
|
||||||
|
|
||||||
create_dir_all(PathBuf::from(LOG_DIR.as_str()))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not create directory: {}", *LOG_DIR));
|
|
||||||
|
|
||||||
let log_line = log_to_str(&messages[0], &config, file!(), line!(), column!()).expect("There should be a log line.") + "\n";
|
|
||||||
|
|
||||||
for msg in messages
|
|
||||||
{
|
|
||||||
log_message(&msg, &config, file!(), line!(), column!());
|
|
||||||
}
|
|
||||||
|
|
||||||
let log_file = read_to_string(PathBuf::from(log_path))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not read file: {log_path}"));
|
|
||||||
|
|
||||||
assert_eq!(log_file, log_line);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Tests if messages are properly filtered according to their verbosity.
|
|
||||||
#[test]
|
|
||||||
pub fn verbosity_filter()
|
|
||||||
{
|
|
||||||
let log_path = &format!("{}/{}", *LOG_DIR, Uuid::new_v4());
|
|
||||||
println!("Log Path: {log_path}");
|
|
||||||
let messages = vec![
|
|
||||||
LogMessageType::GenericErr(String::from("Test Err")).log_message(),
|
|
||||||
LogMessageType::GenericWarn(String::from("Test Warn")).log_message(),
|
|
||||||
LogMessageType::GenericInfo(String::from("Test Info")).log_message(),
|
|
||||||
LogMessageType::GenericDebug(String::from("Test Debug")).log_message(),
|
|
||||||
];
|
|
||||||
let mut config = config::DEFAULTS;
|
|
||||||
config.set_log_path(PathBuf::from(log_path));
|
|
||||||
|
|
||||||
create_dir_all(PathBuf::from(LOG_DIR.as_str()))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not create directory: {}", *LOG_DIR));
|
|
||||||
|
|
||||||
let mut log_line = log_to_str(&messages[0], &config, file!(), line!(), column!()).expect("There should be a log line.") + "\n";
|
|
||||||
log_line += &(log_to_str(&messages[1], &config, file!(), line!(), column!()).expect("There should be a log line.") + "\n");
|
|
||||||
|
|
||||||
for msg in messages
|
|
||||||
{
|
|
||||||
log_message(&msg, &config, file!(), line!(), column!());
|
|
||||||
}
|
|
||||||
|
|
||||||
let log_file = read_to_string(PathBuf::from(log_path))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not read file: {log_path}"));
|
|
||||||
|
|
||||||
assert_eq!(log_file, log_line);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* All testing concurrency in a controlled manner.
|
|
||||||
*
|
|
||||||
* When a test modifies the config according to it's requirements, another test might panic, because
|
|
||||||
* it might read the result from the changed config.
|
|
||||||
*/
|
|
||||||
#[test]
|
|
||||||
pub fn concurrency_tests()
|
|
||||||
{
|
|
||||||
log_macro_file();
|
|
||||||
log_concurrently_any_order();
|
|
||||||
log_concurrently_correct_order();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests if log macro logs to file correctly.
|
|
||||||
* [`config::CONFIG`]
|
|
||||||
*/
|
|
||||||
fn log_macro_file()
|
|
||||||
{
|
|
||||||
let log_path = &format!("{}/{}", *LOG_DIR, Uuid::new_v4());
|
|
||||||
println!("Log Path: {log_path}");
|
|
||||||
let message = LogMessageType::GenericWarn(String::from("Test Log")).log_message();
|
|
||||||
let mut config = config::CONFIG.write().expect("Could not acquire write lock on config!");
|
|
||||||
take(&mut *config);
|
|
||||||
config.set_log_path(PathBuf::from(log_path));
|
|
||||||
drop(config);
|
|
||||||
|
|
||||||
create_dir_all(PathBuf::from(LOG_DIR.as_str()))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not create directory: {}", *LOG_DIR));
|
|
||||||
|
|
||||||
log!(&message);
|
|
||||||
|
|
||||||
let log_file = read_to_string(PathBuf::from(log_path))
|
|
||||||
.unwrap_or_else(|_| panic!("Could not read file: {log_path}"));
|
|
||||||
|
|
||||||
assert_eq!(message.to_string()+"\n", log_file);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests concurrent logging. Log lines may be in any order.
|
|
||||||
*/
|
|
||||||
fn log_concurrently_any_order()
|
|
||||||
{
|
|
||||||
let log_path = &format!("{}/{}", *LOG_DIR, Uuid::new_v4());
|
|
||||||
println!("Log Path: {log_path}");
|
|
||||||
let mut config = config::CONFIG.write().expect("Could not acquire write lock on config!");
|
|
||||||
take(&mut *config);
|
|
||||||
let mut messages = Vec::with_capacity(CONCURRENT_MESSAGE_COUNT);
|
|
||||||
config.set_log_path(PathBuf::from(log_path));
|
|
||||||
drop(config);
|
|
||||||
|
|
||||||
for i in 0..CONCURRENT_MESSAGE_COUNT
|
|
||||||
{
|
|
||||||
let msg = i.to_string();
|
|
||||||
messages.push(
|
|
||||||
async {
|
|
||||||
log!(&LogMessageType::GenericWarn(msg).log_message());
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
block_on(join_all(messages));
|
|
||||||
|
|
||||||
let mut num_set = HashSet::with_capacity(CONCURRENT_MESSAGE_COUNT);
|
|
||||||
for i in 0..CONCURRENT_MESSAGE_COUNT
|
|
||||||
{
|
|
||||||
num_set.insert(i);
|
|
||||||
}
|
|
||||||
|
|
||||||
for line in read_to_string(PathBuf::from(log_path)).unwrap_or_else(|_| panic!("Could not read file: {log_path}")).lines()
|
|
||||||
{
|
|
||||||
let num_str = line.split_whitespace().last().unwrap_or_else(|| panic!("Could not get message number from line: {line}"));
|
|
||||||
let num = usize::from_str_radix(num_str, 10).unwrap_or_else(|_| panic!("Could not parse number: {num_str}"));
|
|
||||||
assert!(num_set.remove(&num));
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_eq!(num_set.len(), 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests concurrent logging. Log lines must be in order.
|
|
||||||
*/
|
|
||||||
fn log_concurrently_correct_order()
|
|
||||||
{
|
|
||||||
let log_path = &format!("{}/{}", *LOG_DIR, Uuid::new_v4());
|
|
||||||
println!("Log Path: {log_path}");
|
|
||||||
let mut config = config::CONFIG.write().expect("Could not acquire write lock on config!");
|
|
||||||
take(&mut *config);
|
|
||||||
let mut messages = Vec::with_capacity(CONCURRENT_MESSAGE_COUNT);
|
|
||||||
config.set_log_path(PathBuf::from(log_path));
|
|
||||||
drop(config);
|
|
||||||
|
|
||||||
for i in 0..CONCURRENT_MESSAGE_COUNT
|
|
||||||
{
|
|
||||||
let msg = i.to_string();
|
|
||||||
messages.push(
|
|
||||||
async {
|
|
||||||
log!(&LogMessageType::GenericWarn(msg).log_message());
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
block_on(join_all(messages));
|
|
||||||
|
|
||||||
let mut num_set = HashSet::with_capacity(CONCURRENT_MESSAGE_COUNT);
|
|
||||||
for i in 0..CONCURRENT_MESSAGE_COUNT
|
|
||||||
{
|
|
||||||
num_set.insert(i);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i, line) in read_to_string(PathBuf::from(log_path)).unwrap_or_else(|_| panic!("Could not read file: {log_path}")).lines().enumerate()
|
|
||||||
{
|
|
||||||
let num_str = line.split_whitespace().last().unwrap_or_else(|| panic!("Could not get message number from line: {line}"));
|
|
||||||
let num = usize::from_str_radix(num_str, 10).unwrap_or_else(|_| panic!("Could not parse number: {num_str}"));
|
|
||||||
assert!(num_set.remove(&num));
|
|
||||||
assert_eq!(i, num);
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_eq!(num_set.len(), 0);
|
|
||||||
|
|
||||||
}
|
|
24
macros/Cargo.toml
Normal file
24
macros/Cargo.toml
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
[package]
|
||||||
|
name = "macros"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
proc-macro = true
|
||||||
|
|
||||||
|
[lints.rust]
|
||||||
|
unsafe_code = "forbid"
|
||||||
|
missing_docs = "warn"
|
||||||
|
|
||||||
|
[lints.clippy]
|
||||||
|
enum_glob_use = "deny" # https://rust-lang.github.io/rust-clippy/master/index.html#/enum_glob_use
|
||||||
|
pedantic = "deny" # https://rust-lang.github.io/rust-clippy/master/index.html#/?groups=pedantic
|
||||||
|
nursery = "deny" # wip lints: https://rust-lang.github.io/rust-clippy/master/index.html#/?groups=nursery
|
||||||
|
unwrap_used = "forbid" # https://rust-lang.github.io/rust-clippy/master/index.html#/unwrap_used
|
||||||
|
missing_const_for_fn = "warn" # https://rust-lang.github.io/rust-clippy/master/index.html#/missing_const_for_fn
|
||||||
|
missing_assert_message = "warn" # https://rust-lang.github.io/rust-clippy/master/index.html#/missing_assert_message
|
||||||
|
missing_errors_doc = "warn" # https://rust-lang.github.io/rust-clippy/master/index.html#/missing_assert_message
|
1
macros/src/lib.rs
Normal file
1
macros/src/lib.rs
Normal file
@ -0,0 +1 @@
|
|||||||
|
|
Reference in New Issue
Block a user