Skip to content

Commit

Permalink
feat: tracker checker command
Browse files Browse the repository at this point in the history
It runs some checks against a running tracker.
  • Loading branch information
josecelano committed Jan 22, 2024
1 parent 7ea6fb0 commit b2ef4e0
Show file tree
Hide file tree
Showing 12 changed files with 451 additions and 0 deletions.
12 changes: 12 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,8 @@ torrust-tracker-located-error = { version = "3.0.0-alpha.12-develop", path = "pa
torrust-tracker-primitives = { version = "3.0.0-alpha.12-develop", path = "packages/primitives" }
tower-http = { version = "0", features = ["compression-full"] }
uuid = { version = "1", features = ["v4"] }
colored = "2.1.0"
url = "2.5.0"

[dev-dependencies]
criterion = { version = "0.5.1", features = ["async_tokio"] }
Expand Down
11 changes: 11 additions & 0 deletions share/default/config/tracker_checker.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
{
"udp_trackers": [
"127.0.0.1:6969"
],
"http_trackers": [
"http://127.0.0.1:7070"
],
"health_checks": [
"http://127.0.0.1:1313/health_check"
]
}
11 changes: 11 additions & 0 deletions src/bin/tracker_checker.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
//! Program to run checks against running trackers.
//!
//! ```text
//! cargo run --bin tracker_checker "./share/default/config/tracker_checker.json"
//! ```
use torrust_tracker::checker::app;

#[tokio::main]
async fn main() {
app::run().await;
}
53 changes: 53 additions & 0 deletions src/checker/app.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
use std::sync::Arc;

use super::config::Configuration;
use super::console::Console;
use crate::checker::config::parse_from_json;
use crate::checker::service::Service;

pub const NUMBER_OF_ARGUMENTS: usize = 2;

/// # Panics
///
/// Will panic if:
///
/// - It can't read the json configuration file.
/// - The configuration file is invalid.
pub async fn run() {

Check warning on line 16 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L16

Added line #L16 was not covered by tests
let args = parse_arguments();
let config = setup_config(&args);
let console_printer = Console {};
let service = Service {
config: Arc::new(config),
console: console_printer,
};

service.run_checks().await;
}

Check warning on line 26 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L26

Added line #L26 was not covered by tests

pub struct Arguments {
pub config_path: String,
}

fn parse_arguments() -> Arguments {
let args: Vec<String> = std::env::args().collect();

Check warning on line 33 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L32-L33

Added lines #L32 - L33 were not covered by tests

if args.len() < NUMBER_OF_ARGUMENTS {
eprintln!("Usage: cargo run --bin tracker_checker <PATH_TO_CONFIG_FILE>");
eprintln!("For example: cargo run --bin tracker_checker ./share/default/config/tracker_checker.json");
std::process::exit(1);

Check warning on line 38 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L35-L38

Added lines #L35 - L38 were not covered by tests
}

let config_path = &args[1];

Check warning on line 41 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L41

Added line #L41 was not covered by tests

Arguments {
config_path: config_path.to_string(),
}
}

Check warning on line 46 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L43-L46

Added lines #L43 - L46 were not covered by tests

fn setup_config(args: &Arguments) -> Configuration {
let file_content = std::fs::read_to_string(args.config_path.clone())
.unwrap_or_else(|_| panic!("Can't read config file {}", args.config_path));

Check warning on line 50 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L48-L50

Added lines #L48 - L50 were not covered by tests

parse_from_json(&file_content).expect("Invalid config format")
}

Check warning on line 53 in src/checker/app.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/app.rs#L52-L53

Added lines #L52 - L53 were not covered by tests
152 changes: 152 additions & 0 deletions src/checker/config.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
use std::fmt;
use std::net::SocketAddr;

use reqwest::Url as ServiceUrl;
use serde::Deserialize;
use url;

/// It parses the configuration from a JSON format.
///
/// # Errors
///
/// Will return an error if the configuration is not valid.
///
/// # Panics
///
/// Will panic if unable to read the configuration file.
pub fn parse_from_json(json: &str) -> Result<Configuration, ConfigurationError> {
let plain_config: PlainConfiguration = serde_json::from_str(json).map_err(ConfigurationError::JsonParseError)?;
Configuration::try_from(plain_config)
}

Check warning on line 20 in src/checker/config.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/config.rs#L17-L20

Added lines #L17 - L20 were not covered by tests

/// DTO for the configuration to serialize/deserialize configuration.
///
/// Configuration does not need to be valid.
#[derive(Deserialize)]

Check warning on line 25 in src/checker/config.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/config.rs#L25

Added line #L25 was not covered by tests
struct PlainConfiguration {
pub udp_trackers: Vec<String>,
pub http_trackers: Vec<String>,
pub health_checks: Vec<String>,
}

/// Validated configuration
pub struct Configuration {
pub udp_trackers: Vec<SocketAddr>,
pub http_trackers: Vec<ServiceUrl>,
pub health_checks: Vec<ServiceUrl>,
}

#[derive(Debug)]

Check warning on line 39 in src/checker/config.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/config.rs#L39

Added line #L39 was not covered by tests
pub enum ConfigurationError {
JsonParseError(serde_json::Error),

Check warning on line 41 in src/checker/config.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/config.rs#L41

Added line #L41 was not covered by tests
InvalidUdpAddress(std::net::AddrParseError),
InvalidUrl(url::ParseError),
}

impl fmt::Display for ConfigurationError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
ConfigurationError::JsonParseError(e) => write!(f, "JSON parse error: {e}"),
ConfigurationError::InvalidUdpAddress(e) => write!(f, "Invalid UDP address: {e}"),
ConfigurationError::InvalidUrl(e) => write!(f, "Invalid URL: {e}"),

Check warning on line 51 in src/checker/config.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/config.rs#L47-L51

Added lines #L47 - L51 were not covered by tests
}
}

Check warning on line 53 in src/checker/config.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/config.rs#L53

Added line #L53 was not covered by tests
}

impl TryFrom<PlainConfiguration> for Configuration {
type Error = ConfigurationError;

fn try_from(plain_config: PlainConfiguration) -> Result<Self, Self::Error> {
let udp_trackers = plain_config
.udp_trackers
.into_iter()
.map(|s| s.parse::<SocketAddr>().map_err(ConfigurationError::InvalidUdpAddress))
.collect::<Result<Vec<_>, _>>()?;

let http_trackers = plain_config
.http_trackers
.into_iter()
.map(|s| s.parse::<ServiceUrl>().map_err(ConfigurationError::InvalidUrl))
.collect::<Result<Vec<_>, _>>()?;

let health_checks = plain_config
.health_checks
.into_iter()
.map(|s| s.parse::<ServiceUrl>().map_err(ConfigurationError::InvalidUrl))
.collect::<Result<Vec<_>, _>>()?;

Ok(Configuration {
udp_trackers,
http_trackers,
health_checks,
})
}
}

#[cfg(test)]
mod tests {
use std::net::{IpAddr, Ipv4Addr, SocketAddr};

use super::*;

#[test]
fn configuration_should_be_build_from_plain_serializable_configuration() {
let dto = PlainConfiguration {
udp_trackers: vec!["127.0.0.1:8080".to_string()],
http_trackers: vec!["http://127.0.0.1:8080".to_string()],
health_checks: vec!["http://127.0.0.1:8080/health".to_string()],
};

let config = Configuration::try_from(dto).expect("A valid configuration");

assert_eq!(
config.udp_trackers,
vec![SocketAddr::new(IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), 8080)]
);
assert_eq!(
config.http_trackers,
vec![ServiceUrl::parse("http://127.0.0.1:8080").unwrap()]
);
assert_eq!(
config.health_checks,
vec![ServiceUrl::parse("http://127.0.0.1:8080/health").unwrap()]
);
}

mod building_configuration_from_plan_configuration {
use crate::checker::config::{Configuration, PlainConfiguration};

#[test]
fn it_should_fail_when_a_tracker_udp_address_is_invalid() {
let plain_config = PlainConfiguration {
udp_trackers: vec!["invalid_address".to_string()],
http_trackers: vec![],
health_checks: vec![],
};

assert!(Configuration::try_from(plain_config).is_err());
}

#[test]
fn it_should_fail_when_a_tracker_http_address_is_invalid() {
let plain_config = PlainConfiguration {
udp_trackers: vec![],
http_trackers: vec!["not_a_url".to_string()],
health_checks: vec![],
};

assert!(Configuration::try_from(plain_config).is_err());
}

#[test]
fn it_should_fail_when_a_health_check_http_address_is_invalid() {
let plain_config = PlainConfiguration {
udp_trackers: vec![],
http_trackers: vec![],
health_checks: vec!["not_a_url".to_string()],
};

assert!(Configuration::try_from(plain_config).is_err());
}
}
}
38 changes: 38 additions & 0 deletions src/checker/console.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
use super::printer::{Printer, CLEAR_SCREEN};

pub struct Console {}

impl Default for Console {
fn default() -> Self {
Self::new()
}

Check warning on line 8 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L6-L8

Added lines #L6 - L8 were not covered by tests
}

impl Console {
#[must_use]
pub fn new() -> Self {

Check warning on line 13 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L13

Added line #L13 was not covered by tests
Self {}
}

Check warning on line 15 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L15

Added line #L15 was not covered by tests
}

impl Printer for Console {
fn clear(&self) {
self.print(CLEAR_SCREEN);
}

Check warning on line 21 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L19-L21

Added lines #L19 - L21 were not covered by tests

fn print(&self, output: &str) {
print!("{}", &output);
}

Check warning on line 25 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L23-L25

Added lines #L23 - L25 were not covered by tests

fn eprint(&self, output: &str) {
eprint!("{}", &output);
}

Check warning on line 29 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L27-L29

Added lines #L27 - L29 were not covered by tests

fn println(&self, output: &str) {
println!("{}", &output);
}

Check warning on line 33 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L31-L33

Added lines #L31 - L33 were not covered by tests

fn eprintln(&self, output: &str) {
eprintln!("{}", &output);
}

Check warning on line 37 in src/checker/console.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/console.rs#L35-L37

Added lines #L35 - L37 were not covered by tests
}
72 changes: 72 additions & 0 deletions src/checker/logger.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
use std::cell::RefCell;

use super::printer::{Printer, CLEAR_SCREEN};

pub struct Logger {
output: RefCell<String>,
}

impl Default for Logger {
fn default() -> Self {
Self::new()
}

Check warning on line 12 in src/checker/logger.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/logger.rs#L10-L12

Added lines #L10 - L12 were not covered by tests
}

impl Logger {
#[must_use]
pub fn new() -> Self {
Self {
output: RefCell::new(String::new()),
}
}

pub fn log(&self) -> String {
self.output.borrow().clone()
}
}

impl Printer for Logger {
fn clear(&self) {
self.print(CLEAR_SCREEN);
}

fn print(&self, output: &str) {
*self.output.borrow_mut() = format!("{}{}", self.output.borrow(), &output);
}

fn eprint(&self, output: &str) {
*self.output.borrow_mut() = format!("{}{}", self.output.borrow(), &output);
}

Check warning on line 39 in src/checker/logger.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/logger.rs#L37-L39

Added lines #L37 - L39 were not covered by tests

fn println(&self, output: &str) {
self.print(&format!("{}/n", &output));
}

Check warning on line 43 in src/checker/logger.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/logger.rs#L41-L43

Added lines #L41 - L43 were not covered by tests

fn eprintln(&self, output: &str) {
self.eprint(&format!("{}/n", &output));
}

Check warning on line 47 in src/checker/logger.rs

View check run for this annotation

Codecov / codecov/patch

src/checker/logger.rs#L45-L47

Added lines #L45 - L47 were not covered by tests
}

#[cfg(test)]
mod tests {
use crate::checker::logger::Logger;
use crate::checker::printer::{Printer, CLEAR_SCREEN};

#[test]
fn should_capture_the_clear_screen_command() {
let console_logger = Logger::new();

console_logger.clear();

assert_eq!(CLEAR_SCREEN, console_logger.log());
}

#[test]
fn should_capture_the_print_command_output() {
let console_logger = Logger::new();

console_logger.print("OUTPUT");

assert_eq!("OUTPUT", console_logger.log());
}
}
6 changes: 6 additions & 0 deletions src/checker/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
pub mod app;
pub mod config;
pub mod console;
pub mod logger;
pub mod printer;
pub mod service;
Loading

0 comments on commit b2ef4e0

Please sign in to comment.