diff --git a/src/defs.rs b/src/defs.rs
index f1940d1..42eea22 100644
--- a/src/defs.rs
+++ b/src/defs.rs
@@ -1,5 +1,5 @@
//! ## Definitions (settings and in common types)
-//! - Group some types definitions in a directory
+//! - Group some types definitions in a directory
//! - Includes global **const** as **&str**
//! - Export / shared to other code files in crate or public
mod cli;
diff --git a/src/defs/cli.rs b/src/defs/cli.rs
index 1cfe7d7..d2db074 100644
--- a/src/defs/cli.rs
+++ b/src/defs/cli.rs
@@ -1,7 +1,7 @@
-//! ## Cli definitions, arguments parsing
+//! ## Cli definitions, arguments parsing
//! - It uses [clap](https://docs.rs/clap/latest/clap/)
//! - It includes a help options with **-h**
-//! - Alows to use differents **config paths** for batch processing
+//! - Alows to use differents **config paths** for batch processing
//! - Alows to be used interactively via terminal as a command for a single input / output file
use clap::Parser;
diff --git a/src/defs/config.rs b/src/defs/config.rs
index c772a3f..eca23fe 100644
--- a/src/defs/config.rs
+++ b/src/defs/config.rs
@@ -1,6 +1,6 @@
//! # Config settings definitions
//! To load config values from TOML file path, it can be provided via command-line arguments
-//! It use [serde](https://serde.rs/) via [`load_from_file`]
+//! It use [serde](https://serde.rs/) via [`load_from_file`]
//
use serde::de::DeserializeOwned;
@@ -28,12 +28,12 @@ fn default_config_targets() -> Vec {
vec![ConfigTarget::default()]
}
/// Settings for each target metric defined in **config path**
-/// **config.toml** content example:
+/// **config.toml** content example:
/// ```toml
-/// be_quiet = false
+/// be_quiet = false
/// [[targets]]
/// input = "input.txt"
-///
+///
/// [[targets]]
/// input = "input_2.txt"
/// ```
@@ -60,12 +60,12 @@ impl Default for ConfigTarget {
}
}
/// Config Settings with target metric settings [`ConfigTarget`]
-/// **config.toml** content example:
+/// **config.toml** content example:
/// ```toml
-/// be_quiet = false
+/// be_quiet = false
/// [[targets]]
/// input = "input.txt"
-///
+///
/// [[targets]]
/// input = "input_2.txt"
/// ```
@@ -109,7 +109,7 @@ impl Config {
}
/// To load config settings and **Deserialize** content to [`Config`] struct
/// It use **T** as generic, allowing to load from a file to a **T** type
-/// It use **fs::read_to_string** as it is expecting short files size, no need to buffers.
+/// It use **fs::read_to_string** as it is expecting short files size, no need to buffers.
pub fn load_from_file(file_cfg: &str) -> Result {
let file_path = if file_cfg.contains(CFG_FILE_EXTENSION) {
file_cfg.to_string()
diff --git a/src/defs/metric_data.rs b/src/defs/metric_data.rs
index 3b76582..4f9da6f 100644
--- a/src/defs/metric_data.rs
+++ b/src/defs/metric_data.rs
@@ -1,11 +1,11 @@
-//! ## Metrics Data for "Consumer Metrics"
+//! ## Metrics Data for "Consumer Metrics"
//! - Collecting, grouping and differentiate name, values, etc.
use chrono::{DateTime, Utc};
use std::{cmp::Ordering, time::SystemTime};
/// Associate type for metric name in [`MetricsConsumerData`]
-/// It allows to collect several **values** related with same **time**
+/// It allows to collect several **values** related with same **time**
#[derive(Clone, Debug)]
pub struct MetricTimeData {
pub time: SystemTime,
@@ -66,7 +66,7 @@ impl std::fmt::Display for MetricsConsumerData {
/// Some implementations:
/// - to get item **name** or **data** values
/// - to **add_data** to exiting [`MetricTimeData`] **values**
-/// - **from_values** allows create an **enum** item
+/// - **from_values** allows create an **enum** item
impl MetricsConsumerData {
pub fn name(&self) -> String {
match self {
diff --git a/src/defs/metrics.rs b/src/defs/metrics.rs
index 0cac393..daab51f 100644
--- a/src/defs/metrics.rs
+++ b/src/defs/metrics.rs
@@ -1,13 +1,17 @@
-//! ## metrics definitions generic models.
+//! ## metrics definitions generic models.
//! - Abstraction via **traits**
//! - Generic process / tasks / steps required for metrics traitment.
-use std::{error::Error, fs::File};
use regex::Regex;
+use std::{error::Error, fs::File};
-pub trait MetricParser {
- fn load_input(&mut self) -> Result>;
- fn parse(&mut self, file: File, reg_exp: Regex);
+pub trait MetricParser
+where
+ Self: Sized,
+{
+ fn input(&self) -> String;
+ fn load_input(&self) -> Result>;
+ fn parse(&self, file: File, reg_exp: Regex) -> Self;
fn collect_aggregates(&mut self);
- fn show_metrics(&mut self) -> Result, std::io::Error>;
+ fn show_metrics(&self) -> Result, std::io::Error>;
}
diff --git a/src/main.rs b/src/main.rs
index 30395be..78c926b 100644
--- a/src/main.rs
+++ b/src/main.rs
@@ -1,46 +1,46 @@
//! # Backend internal interview (rust) - improved
-//!
+//!
//! This **Improved** branch is a rather disruptive approach to the [initial proposal](https://repo.jesusperez.pro/NewRelic/be-technical-interview-rust)
-//! [Branch basic-fixed](https://repo.jesusperez.pro/NewRelic/be-technical-interview-rust/src/branch/basic-fixed) tried to solve proposal from initial code
+//! [Branch basic-fixed](https://repo.jesusperez.pro/NewRelic/be-technical-interview-rust/src/branch/basic-fixed) tried to solve proposal from initial code
//! as a **continuity effort** with the necessary changes and some improvement adjustments such as the **parallel input processing**
-//!
+//!
//! ## In summary
//! - [x] Define a basic model, easily to extend and modify. **Abstraction / Generic**.
//! - [x] Structs and implementations to specific metricis traitments. **Modular appoach**.
//! - [x] Settings and configuration for interactive and non interactive processing (batch mode) **Customize on context**.
-//!
+//!
//! ## Main improvements
-//! - Create abstractions / models to load and parse metrics, can be easily extended
+//! - Create abstractions / models to load and parse metrics, can be easily extended
//! - Define basic operations via [`MetricParser`] trait path **defs/metrics.rs**
//! - Use **structs** to group metrics attributes and implement operations like [`metrics_consumer::MetricsConsumerTarget`]
//! - Use **enums** with associated values like [`defs::metric_data::MetricsConsumerData`] path **defs/metric_data** to group attributes and values
//! - Remove **Maps collections**, use [Vectors](https://doc.rust-lang.org/std/vec/struct.Vec.html)
-//! - Use **const** for DEFAULT values on [`defs`] module
+//! - Use **const** for DEFAULT values on [`defs`] module
//! - Remove all hardcoded values to [`Config`] settings, a **TOML config file** (see config.toml) can be used to define several metrics input and customize output
//! - **Output** write to file (append o rewrite mode) is working (for now I/O is sync)
//! - **Command line** arguments are processed [`parse_args`] with [`defs::CliSettings`] via [clap](https://docs.rs/clap/latest/clap/) and **overide** the ones loaded in **config files**
//! - **Tests** have been accommodated from previous [Branch basic-fixed](https://repo.jesusperez.pro/NewRelic/be-technical-interview-rust/src/branch/basic-fixed) version to **imporved** approach
-//!
-//! ## Benefits
-//!
+//!
+//! ## Benefits
+//!
//! | Element | Benefit |
//! |----------------- |---------|
//! | Generic traits | Group generic task for metric processing, steps / tasks separation |
//! | Structs | Customize atributes and implementation for specific target or patterns |
//! | Enums with values| Associate attributes metrics and values, easy to add new attributes or combine with different values at once |
-//! | Vectors | Simplify types / grouped in structs, priorize vectors type, easy to iterate, filter, sort, etc |
+//! | Vectors | Simplify types / grouped in structs, priorize vectors type, easy to iterate, filter, sort, etc |
//! | Const and Config | Group main const, define metric targes and operations in declarative mode for non intective traitment |
//! | Command line args| Help to run in terminal as a cli |
//! | Unit Tests | Verify some operations results |
-//!
-//! ## Ideas not included
-//! - **Async I/O** to scale and performance ?
-//! - Other Thread alternatives like [Tokio](https://tokio.rs/) or/and [Coroutines](https://doc.rust-lang.org/std/ops/trait.Coroutine.html)
+//!
+//! ## Ideas not included
+//! - **Async I/O** to scale and performance ?
+//! - Other Thread alternatives like [Tokio](https://tokio.rs/) or/and [Coroutines](https://doc.rust-lang.org/std/ops/trait.Coroutine.html)
//! - Benchmarking for optimization
-//! - More **tests**
+//! - More **tests**
//! - Run as **API mode** not only as batch processing
-//!
-//!
+//!
+//!
//! Code is in a private repository with several branches[^note].
//! [^note]: Link to [branch repository improved](https://repo.jesusperez.pro/NewRelic/be-technical-interview-rust/src/branch/improved)
@@ -70,47 +70,40 @@ mod tests;
/// Main threads control flow for each `target_list` item
/// All process are collected and finally
-/// [`MetricsConsumerTarget`] to **show_metrics** is called to get **print** or **write** results
+/// [`MetricsConsumerTarget`] as **T** to **show_metrics** is called to get **print** or **write** results
/// **be_quiet** attribute is just to avoid (true) all messages around processing and parsing operations
-/// ## For paralellism
+/// ## For paralellism
/// This can be done with [Tokio](https://tokio.rs/) as alternative to [std::thread](https://doc.rust-lang.org/std/thread/)
/// It will require load other **crates** and feature customizations
/// Another alternative could be [Coroutines](https://doc.rust-lang.org/std/ops/trait.Coroutine.html)
/// As experimental features is a **nightly-only** (October 2024)
-fn generate_metrics(targets_list: Vec, be_quiet: bool) {
+fn generate_metrics(targets_list: Vec, be_quiet: bool) {
let n_items = targets_list.len();
let mut input_threads = Vec::with_capacity(n_items);
- let (tx, rx): (
- Sender,
- Receiver,
- ) = mpsc::channel();
- for mut metrics_consumer_data in targets_list.clone() {
+ let (tx, rx): (Sender