1
1
mirror of https://github.com/theoludwig/billion_row_challenge.git synced 2024-10-29 22:18:25 +01:00

feat: add basic implementation

This commit is contained in:
Théo LUDWIG 2024-06-23 11:24:03 +02:00
parent a55dfa791f
commit 4d7efe67f7
Signed by: theoludwig
GPG Key ID: ADFE5A563D718F3B
5 changed files with 262 additions and 1 deletions

16
src/config.rs Normal file
View File

@ -0,0 +1,16 @@
#[derive(Debug, PartialEq)]
pub struct Config {
pub input_file_path: String,
}
impl Config {
pub fn build(mut arguments: impl Iterator<Item = String>) -> Result<Config, &'static str> {
arguments.next();
let error_message = "Usage: billion_row_challenge <input_file_path>";
let input_file_path = match arguments.next() {
Some(argument) => argument,
None => return Err(error_message),
};
Ok(Config { input_file_path })
}
}

27
src/error.rs Normal file
View File

@ -0,0 +1,27 @@
use std::error::Error;
use std::fmt::{Display, Formatter};
#[derive(Debug)]
pub enum RunError {
InputOutputError(std::io::Error),
Other(Box<dyn Error>),
}
impl Display for RunError {
fn fmt(&self, formatter: &mut Formatter<'_>) -> std::fmt::Result {
match self {
RunError::InputOutputError(error) => {
write!(formatter, "{error}")
}
RunError::Other(error) => write!(formatter, "{error}"),
}
}
}
impl Error for RunError {}
impl From<std::io::Error> for RunError {
fn from(error: std::io::Error) -> Self {
RunError::InputOutputError(error)
}
}

19
src/lib.rs Normal file
View File

@ -0,0 +1,19 @@
use config::Config;
use error::RunError;
use std::fs;
use weather::WeatherStations;
pub mod config;
pub mod error;
pub mod weather;
pub fn run(config: &Config) -> Result<(), RunError> {
let file_content = fs::read_to_string(&config.input_file_path)?;
let mut lines: Vec<String> = file_content.lines().map(|line| line.to_string()).collect();
let mut weather_stations = WeatherStations::default();
for line in lines.iter_mut() {
weather_stations.add_measurement(line);
}
println!("{}", weather_stations.output());
Ok(())
}

View File

@ -1,3 +1,31 @@
use std::env;
use std::io::ErrorKind;
use std::process;
use billion_row_challenge::config::Config;
use billion_row_challenge::error::RunError;
fn main() {
println!("Hello, world!");
let config = Config::build(env::args()).unwrap_or_else(|error| {
eprintln!("{error}");
process::exit(1);
});
match billion_row_challenge::run(&config) {
Ok(_) => (),
Err(error) => match error {
RunError::InputOutputError(error) => {
match error.kind() {
ErrorKind::NotFound => {
eprintln!("Error: File `{}` not found.", config.input_file_path)
}
_ => eprintln!("Error: {error}"),
}
process::exit(1);
}
RunError::Other(error) => {
eprintln!("Error: {error}");
process::exit(1);
}
},
}
}

171
src/weather.rs Normal file
View File

@ -0,0 +1,171 @@
use std::collections::HashMap;
use std::str::FromStr;
fn round_towards_positive(value: f64, decimal_places: u32) -> f64 {
let factor = 10_f64.powi(decimal_places as i32);
(value * factor).ceil() / factor
}
#[derive(Debug, Default, PartialEq, Clone)]
pub struct WeatherStationMeasurement {
pub name: String,
pub temperature: f64,
}
impl FromStr for WeatherStationMeasurement {
type Err = &'static str;
/// Parses a string `string` to return a value of [`WeatherStationMeasurement`]
///
/// If parsing succeeds, return the value inside [`Ok`], otherwise
/// when the string is ill-formatted return an error specific to the
/// inside [`Err`].
///
/// # Examples
///
/// ```
/// use std::str::FromStr;
/// use billion_row_challenge::weather::WeatherStationMeasurement;
///
/// let string = "Kunming;19.8";
/// let expected_result = WeatherStationMeasurement {
/// name: String::from("Kunming"),
/// temperature: 19.8,
/// };
/// let actual_result = WeatherStationMeasurement::from_str(string).unwrap();
/// assert_eq!(actual_result, expected_result);
/// ```
fn from_str(string: &str) -> Result<Self, Self::Err> {
let mut result = WeatherStationMeasurement::default();
let mut parts = string.split(';');
result.name = parts.next().unwrap_or("station").to_string();
result.temperature = parts.next().unwrap_or("0").trim().parse().unwrap_or(0.0);
Ok(result)
}
}
#[derive(Debug, Default, PartialEq, Clone)]
pub struct WeatherStationMeasurements {
pub name: String,
pub minimum: f64,
pub maximum: f64,
pub count: usize,
pub sum: f64,
}
impl WeatherStationMeasurements {
pub fn average(&self) -> f64 {
if self.count == 0 {
return 0.0;
}
self.sum / (self.count as f64)
}
/// Returns a string representation of the [`WeatherStationMeasurements`] instance in the format:
/// ```text
/// name=minimum/average/maximum
/// ```
/// # Examples
/// ```
/// use billion_row_challenge::weather::WeatherStationMeasurements;
///
/// let weather_station = WeatherStationMeasurements {
/// name: String::from("Bosaso"),
/// minimum: -15.0,
/// maximum: 20.3,
/// count: 10,
/// sum: 100.0,
/// };
/// let expected_output = "Bosaso=-15.0/10.0/20.3";
/// let actual_output = weather_station.output();
/// assert_eq!(actual_output, expected_output);
pub fn output(&self) -> String {
format!(
"{}={:.1}/{:.1}/{:.1}",
self.name,
self.minimum,
round_towards_positive(self.average(), 1),
self.maximum
)
}
}
#[derive(Debug, Default, PartialEq, Clone)]
pub struct WeatherStations {
pub stations: HashMap<String, WeatherStationMeasurements>,
}
impl WeatherStations {
pub fn add_measurement(&mut self, line: &str) {
let weather_station_measurement =
WeatherStationMeasurement::from_str(line).unwrap_or_default();
match self.stations.get_mut(&weather_station_measurement.name) {
Some(value) => {
value.maximum = value.maximum.max(weather_station_measurement.temperature);
value.minimum = value.minimum.min(weather_station_measurement.temperature);
value.count += 1;
value.sum += weather_station_measurement.temperature;
}
None => {
let name = weather_station_measurement.name;
self.stations.insert(
name.clone(),
WeatherStationMeasurements {
name,
minimum: weather_station_measurement.temperature,
maximum: weather_station_measurement.temperature,
count: 1,
sum: weather_station_measurement.temperature,
},
);
}
}
}
/// Returns a string representation of the [`WeatherStations`] instance in the format:
/// ```text
/// {station1, station2, station3}
/// ```
/// # Examples
/// ```
/// use billion_row_challenge::weather::WeatherStations;
/// use billion_row_challenge::weather::WeatherStationMeasurements;
/// use std::collections::HashMap;
///
/// let mut weather_stations = WeatherStations {
/// stations: HashMap::new(),
/// };
/// weather_stations.stations.insert(
/// String::from("Bosaso"),
/// WeatherStationMeasurements {
/// name: String::from("Bosaso"),
/// minimum: -15.0,
/// maximum: 20.0,
/// count: 10,
/// sum: 100.0,
/// },
/// );
/// weather_stations.stations.insert(
/// String::from("Petropavlovsk-Kamchatsky"),
/// WeatherStationMeasurements {
/// name: String::from("Petropavlovsk-Kamchatsky"),
/// minimum: -10.0,
/// maximum: 10.0,
/// count: 0,
/// sum: 0.0,
/// },
/// );
/// let expected_output = "{Bosaso=-15.0/10.0/20.0, Petropavlovsk-Kamchatsky=-10.0/0.0/10.0}";
/// let actual_output = weather_stations.output();
/// assert_eq!(actual_output, expected_output);
pub fn output(&self) -> String {
let mut outputs: Vec<String> = vec![];
let mut station_names: Vec<&String> = self.stations.keys().collect();
station_names.sort_unstable();
for station_name in station_names.into_iter() {
let weather_station = self.stations.get(station_name).unwrap();
outputs.push(weather_station.output());
}
format!("{{{}}}", outputs.join(", "))
}
}